/* source: xio-pipe.c */ /* Copyright Gerhard Rieger and contributors (see file CHANGES) */ /* Published under the GNU General Public License V.2, see file COPYING */ /* this file contains the source for opening addresses of pipe type */ #include "xiosysincludes.h" #include "xioopen.h" #include "xio-named.h" #include "xio-pipe.h" #if WITH_PIPE static int xioopen_fifo(int argc, const char *argv[], struct opt *opts, int xioflags, xiofile_t *fd, const struct addrdesc *addrdesc); static int xioopen_fifo_unnamed(xiofile_t *sock, struct opt *opts); const struct addrdesc xioaddr_pipe = { "PIPE", 3, xioopen_fifo, GROUP_FD|GROUP_NAMED|GROUP_OPEN|GROUP_FIFO, 0, 0, 0 HELP("[:]") }; #if defined(F_SETPIPE_SZ) const struct optdesc opt_f_setpipe_sz = { "f-setpipe-sz", "pipesz", OPT_F_SETPIPE_SZ, GROUP_FD, PH_FD, TYPE_INT, OFUNC_FCNTL, F_SETPIPE_SZ, 0 }; #endif /* process an unnamed bidirectional "pipe" or "fifo" or "echo" argument with options */ static int xioopen_fifo_unnamed(xiofile_t *sock, struct opt *opts) { struct single *sfd = &sock->stream; struct opt *opts2; int filedes[2]; int numleft; int result; if (applyopts_single(sfd, opts, PH_INIT) < 0) return -1; applyopts(sfd, -1, opts, PH_INIT); if (Pipe(filedes) != 0) { Error2("pipe(%p): %s", filedes, strerror(errno)); return -1; } /*0 Info2("pipe({%d,%d})", filedes[0], filedes[1]);*/ sock->common.tag = XIO_TAG_RDWR; sfd->dtype = XIODATA_PIPE; sfd->fd = filedes[0]; sfd->para.bipipe.fdout = filedes[1]; sfd->para.bipipe.socktype = SOCK_STREAM; /* due to socketpair reuse */ applyopts_cloexec(sfd->fd, opts); applyopts_cloexec(sfd->para.bipipe.fdout, opts); /* one-time and input-direction options, no second application */ retropt_bool(opts, OPT_IGNOREEOF, &sfd->ignoreeof); /* here we copy opts! */ if ((opts2 = copyopts(opts, GROUP_FIFO)) == NULL) { return STAT_NORETRY; } /* apply options to first FD */ if ((result = applyopts(sfd, -1, opts, PH_ALL)) < 0) { return result; } if ((result = applyopts_single(sfd, opts, PH_ALL)) < 0) { return result; } /* apply options to second FD */ if (applyopts(&sock->stream, sfd->para.bipipe.fdout, opts2, PH_ALL) < 0) return -1; if ((numleft = leftopts(opts)) > 0) { showleft(opts); Error1("INTERNAL: %d option(s) remained unused", numleft); } xio_chk_pipesz(sfd->fd); Notice("writing to and reading from unnamed pipe"); return 0; } /* open a named or unnamed pipe/fifo */ static int xioopen_fifo( int argc, const char *argv[], struct opt *opts, int xioflags, xiofile_t *xfd, const struct addrdesc *addrdesc) { struct single *sfd = &xfd->stream; const char *pipename = argv[1]; int rw = (xioflags & XIO_ACCMODE); #if HAVE_STAT64 struct stat64 pipstat; #else struct stat pipstat; #endif /* !HAVE_STAT64 */ bool opt_unlink_early = false; bool opt_unlink_close = true; mode_t mode = 0666; int result; if (argc == 1) { return xioopen_fifo_unnamed(xfd, sfd->opts); } if (argc != 2) { xio_syntax(argv[0], 1, argc-1,addrdesc->syntax); return STAT_NORETRY; } if (applyopts_single(sfd, opts, PH_INIT) < 0) return -1; applyopts(sfd, -1, opts, PH_INIT); retropt_bool(opts, OPT_UNLINK_EARLY, &opt_unlink_early); applyopts_named(pipename, opts, PH_EARLY); /* umask! */ applyopts(sfd, -1, opts, PH_EARLY); if (opt_unlink_early) { if (Unlink(pipename) < 0) { return STAT_RETRYLATER; } } retropt_bool(opts, OPT_UNLINK_CLOSE, &opt_unlink_close); retropt_modet(opts, OPT_PERM, &mode); if (applyopts_named(pipename, opts, PH_EARLY) < 0) { return STAT_RETRYLATER; } if (applyopts_named(pipename, opts, PH_PREOPEN) < 0) { return STAT_RETRYLATER; } if ( #if HAVE_STAT64 Stat64(pipename, &pipstat) < 0 #else Stat(pipename, &pipstat) < 0 #endif /* !HAVE_STAT64 */ ) { if (errno != ENOENT) { Error3("stat(\"%s\", %p): %s", pipename, &pipstat, strerror(errno)); } else { Debug1("xioopen_fifo(\"%s\"): does not exist, creating fifo", pipename); #if 0 result = Mknod(pipename, S_IFIFO|mode, 0); if (result < 0) { Error3("mknod(%s, %d, 0): %s", pipename, mode, strerror(errno)); return STAT_RETRYLATER; } #else result = Mkfifo(pipename, mode); if (result < 0) { Error3("mkfifo(%s, %d): %s", pipename, mode, strerror(errno)); return STAT_RETRYLATER; } #endif Notice2("created named pipe \"%s\" for %s", pipename, ddirection[rw]); applyopts_named(pipename, opts, PH_ALL); } if (opt_unlink_close) { if ((sfd->unlink_close = strdup(pipename)) == NULL) { Error1("strdup(\"%s\"): out of memory", pipename); } sfd->opt_unlink_close = true; } } else { /* exists */ Info1("xioopen_fifo(\"%s\"): already exist, opening it", pipename); Notice3("opening %s \"%s\" for %s", filetypenames[(pipstat.st_mode&S_IFMT)>>12], pipename, ddirection[rw]); applyopts_named(pipename, opts, PH_EARLY); } if ((result = _xioopen_open(pipename, rw, opts)) < 0) { return result; } sfd->fd = result; applyopts_named(pipename, opts, PH_FD); applyopts(sfd, -1, opts, PH_FD); applyopts_cloexec(sfd->fd, opts); xio_chk_pipesz(sfd->fd); return _xio_openlate(sfd, opts); } /* Checks if fd is a pipe and if its buffer is at least the blksiz. returns 0 if ok; returns 1 if unknown; returns -1 if not */ int xio_chk_pipesz( int fd) { struct stat st; int pipesz; if (fstat(fd, &st) < 0) { Warn2("fstat(%d, ...): %s", fd, strerror(errno)); return 1; } if ((st.st_mode&S_IFMT) != S_IFIFO) { return 0; } #if defined(F_GETPIPE_SZ) if ((pipesz = Fcntl(fd, F_GETPIPE_SZ)) < 0) { Warn2("fcntl(%d, F_GETPIPE_SZ): %s", fd, strerror(errno)); return 1; } if (pipesz >= xioparms.bufsiz) return 0; Warn3("xio_chk_pipesz(%d, ...): Socat block size "F_Zu" is larger than pipe size %d, might block; use option f-setpipe-sz!", fd, xioparms.bufsiz, pipesz); return -1; #else /* !defined(F_GETPIPE_SZ) */ return 1; #endif /* !defined(F_GETPIPE_SZ) */ } #endif /* WITH_PIPE */