X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~ianmdlvl/git?a=blobdiff_plain;f=src%2Fshared%2Fptyfwd.c;h=31274a141880079655499adf94654802ec7ef6f2;hb=374c22b351e43ce4ef70ef0ad1bd1e4e520f9a28;hp=11356e2def5900deb9afe8b287084870f5a1b809;hpb=9b15b7846d4de01bb5d9700a24077787e984e8ab;p=elogind.git diff --git a/src/shared/ptyfwd.c b/src/shared/ptyfwd.c index 11356e2de..31274a141 100644 --- a/src/shared/ptyfwd.c +++ b/src/shared/ptyfwd.c @@ -53,7 +53,8 @@ struct PTYForward { bool master_writable:1; bool master_hangup:1; - bool repeat:1; + /* Continue reading after hangup? */ + bool ignore_vhangup:1; bool last_char_set:1; char last_char; @@ -171,13 +172,13 @@ static int shovel(PTYForward *f) { if (k < 0) { /* Note that EIO on the master device - * might be cause by vhangup() or + * might be caused by vhangup() or * temporary closing of everything on * the other side, we treat it like * EAGAIN here and try again, unless - * repeat is off. */ + * ignore_vhangup is off. */ - if (errno == EAGAIN || (errno == EIO && f->repeat)) + if (errno == EAGAIN || (errno == EIO && f->ignore_vhangup)) f->master_readable = false; else if (errno == EPIPE || errno == ECONNRESET || errno == EIO) { f->master_readable = f->master_writable = false; @@ -292,12 +293,12 @@ static int on_sigwinch_event(sd_event_source *e, const struct signalfd_siginfo * /* The window size changed, let's forward that. */ if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) >= 0) - (void)ioctl(f->master, TIOCSWINSZ, &ws); + (void) ioctl(f->master, TIOCSWINSZ, &ws); return 0; } -int pty_forward_new(sd_event *event, int master, bool repeat, PTYForward **ret) { +int pty_forward_new(sd_event *event, int master, bool ignore_vhangup, PTYForward **ret) { _cleanup_(pty_forward_freep) PTYForward *f = NULL; struct winsize ws; int r; @@ -306,7 +307,7 @@ int pty_forward_new(sd_event *event, int master, bool repeat, PTYForward **ret) if (!f) return -ENOMEM; - f->repeat = repeat; + f->ignore_vhangup = ignore_vhangup; if (event) f->event = sd_event_ref(event); @@ -372,6 +373,8 @@ int pty_forward_new(sd_event *event, int master, bool repeat, PTYForward **ret) return r; r = sd_event_add_signal(f->event, &f->sigwinch_event_source, SIGWINCH, on_sigwinch_event, f); + if (r < 0) + return r; *ret = f; f = NULL; @@ -403,7 +406,7 @@ PTYForward *pty_forward_free(PTYForward *f) { return NULL; } -int pty_forward_last_char(PTYForward *f, char *ch) { +int pty_forward_get_last_char(PTYForward *f, char *ch) { assert(f); assert(ch); @@ -413,3 +416,32 @@ int pty_forward_last_char(PTYForward *f, char *ch) { *ch = f->last_char; return 0; } + +int pty_forward_set_ignore_vhangup(PTYForward *f, bool ignore_vhangup) { + int r; + + assert(f); + + if (f->ignore_vhangup == ignore_vhangup) + return 0; + + f->ignore_vhangup = ignore_vhangup; + if (!f->ignore_vhangup) { + + /* We shall now react to vhangup()s? Let's check + * immediately if we might be in one */ + + f->master_readable = true; + r = shovel(f); + if (r < 0) + return r; + } + + return 0; +} + +int pty_forward_get_ignore_vhangup(PTYForward *f) { + assert(f); + + return f->ignore_vhangup; +}