mirror of
https://repo.or.cz/socat.git
synced 2025-01-08 22:12:33 +00:00
269 lines
7.3 KiB
C
269 lines
7.3 KiB
C
/* source: xioshutdown.c */
|
|
/* Copyright Gerhard Rieger 2001-2009 */
|
|
/* Published under the GNU General Public License V.2, see file COPYING */
|
|
|
|
/* this is the source of the extended shutdown function */
|
|
|
|
|
|
#include "xiosysincludes.h"
|
|
#include "xioopen.h"
|
|
|
|
|
|
static int xioshut_sleep_kill(pid_t sub, unsigned long usec, int sig);
|
|
|
|
static pid_t socat_kill_pid; /* here we pass the pid to be killed in sighandler */
|
|
|
|
static void signal_kill_pid(int dummy) {
|
|
Notice("SIGALRM while waiting for w/o child process to die, killing it now");
|
|
Kill(socat_kill_pid, SIGTERM);
|
|
}
|
|
|
|
/* how: SHUT_RD, SHUT_WR, or SHUT_RDWR */
|
|
int xioshutdown(xiofile_t *sock, int how) {
|
|
int fd;
|
|
int result = 0;
|
|
|
|
Debug2("xioshutdown(%p, %d)", sock, how);
|
|
Debug2("xioshutdown(): dtype=0x%x, howtoshut=0x%04x",
|
|
sock->stream.dtype, sock->stream.howtoshut);
|
|
|
|
if (sock->tag == XIO_TAG_INVALID) {
|
|
Error("xioshutdown(): invalid file descriptor");
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
|
|
/*Debug3("xioshutdown: flags=%d, dtype=%d, howtoclose=%d", sock->stream.flags, sock->stream.dtype, sock->stream.howtoclose);*/
|
|
if (sock->tag == XIO_TAG_DUAL) {
|
|
if ((how+1)&1) {
|
|
result = xioshutdown((xiofile_t *)sock->dual.stream[0], 0);
|
|
}
|
|
if ((how+1)&2) {
|
|
result |= xioshutdown((xiofile_t *)sock->dual.stream[1], 1);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
fd = XIO_GETWRFD(sock);
|
|
|
|
/* let us bring how nearer to the resulting action */
|
|
if ((sock->stream.flags&XIO_ACCMODE) == XIO_WRONLY) {
|
|
how = ((how+1) & ~(SHUT_RD+1)) - 1;
|
|
} else if ((sock->stream.flags&XIO_ACCMODE) == XIO_RDONLY) {
|
|
how = ((how+1) & ~(SHUT_WR+1)) - 1;
|
|
}
|
|
|
|
switch (sock->stream.howtoshut) {
|
|
#if WITH_PTY
|
|
case XIOSHUT_PTYEOF:
|
|
{
|
|
struct termios termarg;
|
|
int result;
|
|
Debug1("tcdrain(%d)", sock->stream.wfd);
|
|
result = tcdrain(sock->stream.wfd);
|
|
Debug1("tcdrain() -> %d", result);
|
|
if (Tcgetattr(sock->stream.wfd, &termarg) < 0) {
|
|
Error3("tcgetattr(%d, %p): %s",
|
|
sock->stream.wfd, &termarg, strerror(errno));
|
|
}
|
|
#if 0
|
|
/* these settings might apply to data still in the buff (despite the
|
|
TCSADRAIN */
|
|
termarg.c_iflag |= (IGNBRK | BRKINT | PARMRK | ISTRIP
|
|
| INLCR | IGNCR | ICRNL | IXON);
|
|
termarg.c_oflag |= OPOST;
|
|
termarg.c_lflag |= (/*ECHO | ECHONL |*/ ICANON | ISIG | IEXTEN);
|
|
//termarg.c_cflag |= (PARENB);
|
|
#else
|
|
termarg.c_lflag |= ICANON;
|
|
#endif
|
|
if (Tcsetattr(sock->stream.wfd, TCSADRAIN, &termarg) < 0) {
|
|
Error3("tcsetattr(%d, TCSADRAIN, %p): %s",
|
|
sock->stream.wfd, &termarg, strerror(errno));
|
|
}
|
|
if (Write(sock->stream.wfd, &termarg.c_cc[VEOF], 1) < 1) {
|
|
Warn3("write(%d, 0%o, 1): %s",
|
|
sock->stream.wfd, termarg.c_cc[VEOF], strerror(errno));
|
|
}
|
|
}
|
|
return 0;
|
|
#endif /* WITH_PTY */
|
|
#if WITH_OPENSSL
|
|
case XIOSHUT_OPENSSL:
|
|
sycSSL_shutdown(sock->stream.para.openssl.ssl);
|
|
/*! what about half/full close? */
|
|
return 0;
|
|
#endif /* WITH_OPENSSL */
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* here handle special shutdown functions */
|
|
switch (sock->stream.howtoshut & XIOSHUTWR_MASK) {
|
|
char writenull;
|
|
case XIOSHUTWR_NONE:
|
|
return 0;
|
|
case XIOSHUTWR_CLOSE:
|
|
if (Close(fd) < 0) {
|
|
Info2("close(%d): %s", fd, strerror(errno));
|
|
}
|
|
return 0;
|
|
case XIOSHUTWR_DOWN:
|
|
if ((result = Shutdown(fd, how)) < 0) {
|
|
Info3("shutdown(%d, %d): %s", fd, how, strerror(errno));
|
|
}
|
|
return 0;
|
|
#if _WITH_SOCKET
|
|
case XIOSHUTWR_NULL:
|
|
/* send an empty packet; only useful on datagram sockets? */
|
|
xiowrite(sock, &writenull, 0);
|
|
return 0;
|
|
#endif /* _WITH_SOCKET */
|
|
default: break;
|
|
}
|
|
|
|
#if 0
|
|
if (how == SHUT_RDWR) {
|
|
/* in this branch we handle only shutdown actions where read and write
|
|
shutdown are not independent */
|
|
|
|
switch (sock->stream.howtoshut) {
|
|
#if _WITH_SOCKET
|
|
case XIOSHUT_DOWN:
|
|
if ((result = Shutdown(fd, how)) < 0) {
|
|
Info3("shutdown(%d, %d): %s", fd, how, strerror(errno));
|
|
}
|
|
break;
|
|
case XIOSHUT_KILL:
|
|
if ((result = Shutdown(fd, how)) < 0) {
|
|
Info3("shutdown(%d, %d): %s", fd, how, strerror(errno));
|
|
}
|
|
break;
|
|
#endif /* _WITH_SOCKET */
|
|
case XIOSHUT_CLOSE:
|
|
Close(fd);
|
|
#if WITH_TERMIOS
|
|
if (sock->stream.ttyvalid) {
|
|
if (Tcsetattr(fd, 0, &sock->stream.savetty) < 0) {
|
|
Warn2("cannot restore terminal settings on fd %d: %s",
|
|
fd, strerror(errno));
|
|
}
|
|
}
|
|
#endif /* WITH_TERMIOS */
|
|
/*PASSTHROUGH*/
|
|
case XIOSHUT_NONE:
|
|
break;
|
|
default:
|
|
Error1("xioshutdown(): bad shutdown action 0x%x", sock->stream.howtoshut);
|
|
return -1;
|
|
}
|
|
|
|
#if 0 && _WITH_SOCKET
|
|
case XIODATA_RECVFROM:
|
|
if (how >= 1) {
|
|
if (Close(fd) < 0) {
|
|
Info2("close(%d): %s",
|
|
fd, strerror(errno));
|
|
}
|
|
sock->stream.eof = 2;
|
|
sock->stream.fd1 = -1;
|
|
}
|
|
break;
|
|
#endif /* _WITH_SOCKET */
|
|
}
|
|
#endif
|
|
|
|
if ((how+1) & 1) { /* contains SHUT_RD */
|
|
switch (sock->stream.dtype & XIODATA_READMASK) {
|
|
/* shutdown read channel */
|
|
|
|
case XIOREAD_STREAM:
|
|
case XIODATA_2PIPE:
|
|
if (Close(fd) < 0) {
|
|
Info2("close(%d): %s", fd, strerror(errno));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((how+1) & 2) { /* contains SHUT_WR */
|
|
/* shutdown write channel */
|
|
|
|
switch (sock->stream.howtoshut & XIOSHUTWR_MASK) {
|
|
|
|
case XIOSHUTWR_CLOSE:
|
|
if (Close(fd) < 0) {
|
|
Info2("close(%d): %s", fd, strerror(errno));
|
|
}
|
|
/*PASSTHROUGH*/
|
|
case XIOSHUTWR_NONE:
|
|
break;
|
|
|
|
#if _WITH_SOCKET
|
|
case XIOSHUTWR_DOWN:
|
|
if (Shutdown(fd, SHUT_WR) < 0) {
|
|
Info2("shutdown(%d, SHUT_WR): %s", fd, strerror(errno));
|
|
}
|
|
break;
|
|
#endif /* _WITH_SOCKET */
|
|
|
|
#if 0
|
|
case XIOSHUTWR_DOWN_KILL:
|
|
if (Shutdown(fd, SHUT_WR) < 0) {
|
|
Info2("shutdown(%d, SHUT_WR): %s", fd, strerror(errno));
|
|
}
|
|
/*!!!*/
|
|
#endif
|
|
case XIOSHUTWR_SIGHUP:
|
|
/* the child process might want to flush some data before
|
|
terminating */
|
|
xioshut_sleep_kill(sock->stream.child.pid, 0, SIGHUP);
|
|
break;
|
|
case XIOSHUTWR_SIGTERM:
|
|
/* the child process might want to flush some data before
|
|
terminating */
|
|
xioshut_sleep_kill(sock->stream.child.pid, 1000000, SIGTERM);
|
|
break;
|
|
case XIOSHUTWR_SIGKILL:
|
|
/* the child process might want to flush some data before
|
|
terminating */
|
|
xioshut_sleep_kill(sock->stream.child.pid, 1000000, SIGKILL);
|
|
break;
|
|
|
|
default:
|
|
Error1("xioshutdown(): unhandled howtoshut=0x%x during SHUT_WR",
|
|
sock->stream.howtoshut&XIOSHUTWR_MASK);
|
|
}
|
|
sock->stream.wfd = -1;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/* wait some time and then send signal to sub process. This is useful after
|
|
shutting down the connection to give process some time to flush its output
|
|
data */
|
|
static int xioshut_sleep_kill(pid_t sub, unsigned long usec, int sig) {
|
|
struct sigaction act;
|
|
int status = 0;
|
|
|
|
/* we wait for the child process to die, but to prevent timeout
|
|
we raise an alarm after some time. */
|
|
/* NOTE: the alarm does not terminate waitpid() on Linux/glibc
|
|
(BUG?),
|
|
therefore we have to do the kill in the signal handler */
|
|
Signal(SIGALRM, signal_kill_pid);
|
|
socat_kill_pid = sub;
|
|
#if HAVE_SETITIMER
|
|
/*! with next feature release, we get usec resolution and an option
|
|
*/
|
|
#else
|
|
Alarm(1 /*! sock->stream.child.waitdie */);
|
|
#endif /* !HAVE_SETITIMER */
|
|
if (Waitpid(sub, &status, 0) < 0) {
|
|
Warn3("waitpid("F_pid", %p, 0): %s",
|
|
sub, &status, strerror(errno));
|
|
}
|
|
Alarm(0);
|
|
return 0;
|
|
}
|