socat/xioclose.c

142 lines
3.5 KiB
C
Raw Permalink Normal View History

2008-01-28 21:37:16 +00:00
/* source: xioclose.c */
2008-01-29 06:59:12 +00:00
/* Copyright Gerhard Rieger 2001-2008 */
2008-01-27 12:00:08 +00:00
/* Published under the GNU General Public License V.2, see file COPYING */
/* this is the source of the extended close function */
#include "xiosysincludes.h"
#include "xioopen.h"
#include "xiolockfile.h"
#include "xio-termios.h"
2008-01-29 06:59:12 +00:00
/* close the xio fd; must be valid and "simple" (not dual) */
2008-01-27 12:00:08 +00:00
int xioclose1(struct single *pipe) {
if (pipe->tag == XIO_TAG_INVALID) {
Notice("xioclose1(): invalid file descriptor");
errno = EINVAL;
return -1;
}
2008-02-17 13:59:16 +00:00
switch (pipe->howtoclose) {
2008-01-27 12:00:08 +00:00
#if WITH_READLINE
2008-02-17 13:59:16 +00:00
case XIOCLOSE_READLINE:
2008-01-27 12:00:08 +00:00
Write_history(pipe->para.readline.history_file);
/*xiotermios_setflag(pipe->fd, 3, ECHO|ICANON);*/ /* error when pty closed */
2008-02-17 13:59:16 +00:00
break;
2008-01-27 12:00:08 +00:00
#endif /* WITH_READLINE */
2008-02-17 13:59:16 +00:00
2008-01-27 12:00:08 +00:00
#if WITH_OPENSSL
2008-02-17 13:59:16 +00:00
case XIOCLOSE_OPENSSL:
2008-01-27 12:00:08 +00:00
if (pipe->para.openssl.ssl) {
/* e.g. on TCP connection refused, we do not yet have this set */
sycSSL_shutdown(pipe->para.openssl.ssl);
sycSSL_free(pipe->para.openssl.ssl);
pipe->para.openssl.ssl = NULL;
}
2008-02-17 13:59:16 +00:00
Close(pipe->fd1); pipe->fd1 = -1;
Close(pipe->fd2); pipe->fd2 = -1;
2008-01-27 12:00:08 +00:00
if (pipe->para.openssl.ctx) {
sycSSL_CTX_free(pipe->para.openssl.ctx);
pipe->para.openssl.ctx = NULL;
}
2008-02-17 13:59:16 +00:00
break;
2008-01-27 12:00:08 +00:00
#endif /* WITH_OPENSSL */
2008-02-17 13:59:16 +00:00
case XIOCLOSE_SIGTERM:
if (pipe->child.pid > 0) {
if (Kill(pipe->child.pid, SIGTERM) < 0) {
Msg2(errno==ESRCH?E_INFO:E_WARN, "kill(%d, SIGTERM): %s",
pipe->child.pid, strerror(errno));
}
}
break;
case XIOCLOSE_CLOSE_SIGTERM:
if (pipe->child.pid > 0) {
if (Kill(pipe->child.pid, SIGTERM) < 0) {
2008-01-27 12:00:08 +00:00
Msg2(errno==ESRCH?E_INFO:E_WARN, "kill(%d, SIGTERM): %s",
2008-02-17 13:59:16 +00:00
pipe->child.pid, strerror(errno));
2008-01-27 12:00:08 +00:00
}
2008-02-17 13:59:16 +00:00
}
/*PASSTHROUGH*/
case XIOCLOSE_CLOSE:
if (pipe->fd1 >= 0) {
if (Close(pipe->fd1) < 0) {
Info2("close(%d): %s", pipe->fd1, strerror(errno));
2008-01-27 12:00:08 +00:00
}
}
2008-02-17 13:59:16 +00:00
break;
case XIOCLOSE_SLEEP_SIGTERM:
Sleep(1);
if (pipe->child.pid > 0) {
if (Kill(pipe->child.pid, SIGTERM) < 0) {
Msg2(errno==ESRCH?E_INFO:E_WARN, "kill(%d, SIGTERM): %s",
pipe->child.pid, strerror(errno));
}
2008-01-27 12:00:08 +00:00
}
2008-02-17 13:59:16 +00:00
break;
case XIOCLOSE_NONE:
break;
default:
2008-10-22 00:33:23 +00:00
Error2("xioclose(): bad close action 0x%x on 0x%x", pipe->howtoclose, pipe);
2008-02-17 13:59:16 +00:00
break;
2008-01-27 12:00:08 +00:00
}
2008-10-22 00:33:23 +00:00
#if WITH_TERMIOS
if (pipe->ttyvalid) {
if (Tcsetattr(pipe->fd1, 0, &pipe->savetty) < 0) {
Warn2("cannot restore terminal settings on fd %d: %s",
pipe->fd1, strerror(errno));
2008-01-27 12:00:08 +00:00
}
}
2008-10-22 00:33:23 +00:00
#endif /* WITH_TERMIOS */
2008-01-27 12:00:08 +00:00
/* unlock */
if (pipe->havelock) {
xiounlock(pipe->lock.lockfile);
pipe->havelock = false;
}
if (pipe->opt_unlink_close && pipe->unlink_close) {
if (Unlink(pipe->unlink_close) < 0) {
Info2("unlink(\"%s\"): %s", pipe->unlink_close, strerror(errno));
}
free(pipe->unlink_close);
}
pipe->tag = XIO_TAG_INVALID;
return 0; /*! */
}
/* close the xio fd */
int xioclose(xiofile_t *file) {
2008-02-17 13:59:16 +00:00
xiofile_t *xfd = file;
2008-01-27 12:00:08 +00:00
int result;
if (file->tag == XIO_TAG_INVALID) {
Error("xioclose(): invalid file descriptor");
errno = EINVAL;
return -1;
}
if (file->tag == XIO_TAG_DUAL) {
result = xioclose1(file->dual.stream[0]);
result |= xioclose1(file->dual.stream[1]);
file->tag = XIO_TAG_INVALID;
} else {
result = xioclose1(&file->stream);
}
2008-02-17 13:59:16 +00:00
if (xfd->stream.subthread != 0) {
Pthread_join(xfd->stream.subthread, NULL);
}
2008-01-27 12:00:08 +00:00
return result;
}