X-Recipient: archive-cygwin AT delorie DOT com X-Original-To: cygwin AT cygwin DOT com Delivered-To: cygwin AT cygwin DOT com DMARC-Filter: OpenDMARC Filter v1.4.1 sourceware.org 2CFAD3858D39 Authentication-Results: sourceware.org; dmarc=none (p=none dis=none) header.from=lyx.org Authentication-Results: sourceware.org; spf=none smtp.mailfrom=lyx.org Date: Sun, 18 Sep 2022 23:45:43 +0200 From: Enrico Forestieri To: cygwin AT cygwin DOT com Subject: FIFO issues Message-ID: Mail-Followup-To: cygwin AT cygwin DOT com MIME-Version: 1.0 Content-Type: multipart/mixed; boundary="adlsL66GMvd2USfy" Content-Disposition: inline X-Spam-Status: No, score=1.0 required=5.0 tests=BAYES_00, KAM_DMARC_STATUS, KAM_LAZY_DOMAIN_SECURITY, RCVD_IN_BARRACUDACENTRAL, SPF_HELO_NONE, SPF_NONE, TXREP autolearn=no autolearn_force=no version=3.4.6 X-Spam-Level: * X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on server2.sourceware.org X-BeenThere: cygwin AT cygwin DOT com X-Mailman-Version: 2.1.29 List-Id: General Cygwin discussions and problem reports List-Archive: List-Post: List-Help: List-Subscribe: , Sender: "Cygwin" --adlsL66GMvd2USfy Content-Type: text/plain; charset=us-ascii Content-Disposition: inline Hi, I think I am experiencing a problem with fifos on cygwin. The attached C source (fifocomm.c) creates two pipes (/tmp/pipe.{in,out}), expecting to receive inputs from /tmp/pipe.in and replying to /tmp/pipe.out. Compiling this source on linux and launching it produces on the terminal 1) Opening pipes and then the program waits for someone to write to /tmp/pipe.in. Opening another terminal and launching the fifotest.sh script (also attached) produces on the first terminal 1) Closing pipes 2) Opening pipes and the program continues waiting for another input, while the other terminal shows "You sent: foo" Instead, on cygwin, after launching the program one gets: 1) Opening pipes 1) Closing pipes 2) Opening pipes 2) Closing pipes 3) Opening pipes 3) Closing pipes 4) Opening pipes .... .... meaning that the pipes are continually closed and reopened even if nobody is writing to /tmp/pipe.in. Seemingly, the problem is the return value of read() on line 55. As O_NONBLOCK is set, until no data is available for reading, read() shouldn't block but should return -1 and set errno to EAGAIN. After a client that opened the pipe for writing, closes it (and no other client is using the pipe), read() should return 0 and only at this point the pipes have to be closed and reopened. However, on cygwin, read() returns 0 also when nobody is writing to the input pipe causing the above ping pong. As already said, it works as it should on linux. -- Enrico --adlsL66GMvd2USfy Content-Type: text/plain; charset=us-ascii Content-Disposition: attachment; filename="fifocomm.c" #include #include #include #include #include #include #include #include #include #define FIFO_IN "/tmp/pipe.in" #define FIFO_OUT "/tmp/pipe.out" #define FALSE 0 #define TRUE 1 typedef int bool; void openpipes(void); void closepipes(void); int startpipe(char *name, bool inputpipe); void endpipe(char *name, int fd); int infd; int outfd; int count = 0; int main(void) { int nsel; bool done = FALSE; char const *msg = "You sent: "; int mlen = strlen(msg); fd_set readfds; FD_ZERO(&readfds); openpipes(); do { FD_SET(infd, &readfds); do { nsel = select(infd + 1, &readfds, 0, 0, 0); } while (nsel == -1 && (errno == EINTR || errno == EAGAIN)); if (nsel == -1) { perror("select"); exit(4); } if (FD_ISSET(infd, &readfds)) { char buf[100] = "You sent: "; int status; int count = 0; strcpy(buf, msg); while ((status = read(infd, buf + mlen, sizeof(buf) - mlen - 1))) { if (status > 0) { buf[mlen + status] = '\0'; if (write(outfd, buf, strlen(buf)) < 0) { if (buf[mlen + status - 1] == '\n') buf[mlen + status - 1] = '\0'; fprintf(stderr, "Error sending message: '%s': %s\n", buf, strerror(errno)); } if (strncasecmp(buf + mlen, "quit", 4) == 0) done = TRUE; } else if (errno == EAGAIN) { /* Nothing to read, continue */ break; } else { /* An error occurred, bail out */ perror("read"); done = TRUE; break; } } if (!done) { sleep(3); closepipes(); openpipes(); errno = 0; } } } while (!done); closepipes(); return 0; } void openpipes(void) { fprintf(stderr, "%d) Opening pipes\n", ++count); infd = startpipe(FIFO_IN, TRUE); if (infd == -1) exit(1); outfd = startpipe(FIFO_OUT, FALSE); if (outfd == -1) exit(2); if (fcntl(outfd, F_SETFL, O_NONBLOCK) < 0) { fprintf(stderr, "Could not set flags on pipe %s: %s\n", FIFO_OUT, strerror(errno)); exit(3); } } void closepipes() { fprintf(stderr, "%d) Closing pipes\n", count); endpipe(FIFO_IN, infd); endpipe(FIFO_OUT, outfd); } int startpipe(char *name, bool inputpipe) { static bool stalepipe = FALSE; int fd; if (access(name, F_OK) == 0) { if (inputpipe) { /* Let's see whether we have a stale pipe */ fd = open(name, O_WRONLY | O_NONBLOCK); if (fd >= 0) { /* pipe exists and is used by another process */ close(fd); } else if (errno == ENXIO) { /* No process is reading from the other end */ stalepipe = TRUE; endpipe(name, -1); } } else if (stalepipe) { endpipe(name, -1); stalepipe = FALSE; } if (access(name, F_OK) == 0) { fprintf(stderr, "Pipe %s already exists and is in use.\n", name); return -1; } } if (mkfifo(name, 0600) < 0) { fprintf(stderr, "Could not create pipe %s: %s\n", name, strerror(errno)); return -1; } fd = open(name, inputpipe ? (O_RDONLY | O_NONBLOCK) : O_RDWR); if (fd < 0) { fprintf(stderr, "Could not open pipe %s: %s\n", name, strerror(errno)); endpipe(name, -1); return -1; } return fd; } void endpipe(char *name, int fd) { if (fd >= 0 && close(fd) < 0) fprintf(stderr, "Could not close pipe %s: %s\n", name, strerror(errno)); if (remove(name) < 0) fprintf(stderr, "Could not remove pipe %s: %s\n", name, strerror(errno)); } --adlsL66GMvd2USfy Content-Type: application/x-sh Content-Disposition: attachment; filename="fifotest.sh" Content-Transfer-Encoding: quoted-printable #!/bin/sh=0Aecho "foo" > /tmp/pipe.in=0Acat /tmp/pipe.out=0A=0A --adlsL66GMvd2USfy Content-Type: text/plain; charset="us-ascii" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Content-Disposition: inline -- Problem reports: https://cygwin.com/problems.html FAQ: https://cygwin.com/faq/ Documentation: https://cygwin.com/docs.html Unsubscribe info: https://cygwin.com/ml/#unsubscribe-simple --adlsL66GMvd2USfy--