1
0
mirror of https://github.com/madler/zlib synced 2025-03-28 21:13:15 +00:00

Merge df4df935d6b0eebfc3174ff73c453f36d3831021 into 5a82f71ed1dfc0bec044d9702463dbdf84ea3b71

This commit is contained in:
Brian Carcich 2025-03-10 22:28:41 +03:00 committed by GitHub
commit e4ad9dc835
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 624 additions and 5 deletions

1
.gitignore vendored
View File

@ -22,6 +22,7 @@
/minigzip
/minigzip64
/minigzipsh
/gznonblk
/zlib.pc
/configure.log
/build

View File

@ -1,8 +1,8 @@
ChangeLog file for zlib
Changes in 1.3.1.1 (xx Jan 2024)
-
Changes in 1.3.1.1 (xx Jul 2024)
- Make non-blocking I/O work in gzread.c (Issue #992)
Changes in 1.3.1 (22 Jan 2024)
- Reject overflows of zip header fields in minizip

View File

@ -79,11 +79,13 @@ shared: examplesh$(EXE) minigzipsh$(EXE)
all64: example64$(EXE) minigzip64$(EXE)
staticgznb: gznonblk$(EXE)
check: test
test: all teststatic testshared
teststatic: static
teststatic: static testgznonblk
@TMPST=tmpst_$$; \
if echo hello world | ${QEMU_RUN} ./minigzip | ${QEMU_RUN} ./minigzip -d && ${QEMU_RUN} ./example $$TMPST ; then \
echo ' *** zlib test OK ***'; \
@ -92,6 +94,13 @@ teststatic: static
fi
@rm -f tmpst_$$
testgznonblk: staticgznb
@if ./gznonblk 44444 --client-fork 127.0.0.1 first second --delay third --delay -stopserver- ; then \
echo ' *** gznonblk test OK ***'; \
else \
echo ' *** gznonblk test FAILED ***'; false; \
fi
testshared: shared
@LD_LIBRARY_PATH=`pwd`:$(LD_LIBRARY_PATH) ; export LD_LIBRARY_PATH; \
LD_LIBRARYN32_PATH=`pwd`:$(LD_LIBRARYN32_PATH) ; export LD_LIBRARYN32_PATH; \
@ -151,6 +160,9 @@ example.o: $(SRCDIR)test/example.c $(SRCDIR)zlib.h zconf.h
minigzip.o: $(SRCDIR)test/minigzip.c $(SRCDIR)zlib.h zconf.h
$(CC) $(CFLAGS) $(ZINCOUT) -c -o $@ $(SRCDIR)test/minigzip.c
gznonblk.o: $(SRCDIR)test/gznonblk.c $(SRCDIR)zlib.h zconf.h
$(CC) $(CFLAGS) $(ZINCOUT) -c -o $@ $(SRCDIR)test/gznonblk.c
example64.o: $(SRCDIR)test/example.c $(SRCDIR)zlib.h zconf.h
$(CC) $(CFLAGS) $(ZINCOUT) -D_FILE_OFFSET_BITS=64 -c -o $@ $(SRCDIR)test/example.c
@ -293,6 +305,9 @@ example$(EXE): example.o $(STATICLIB)
minigzip$(EXE): minigzip.o $(STATICLIB)
$(CC) $(CFLAGS) $(LDFLAGS) -o $@ minigzip.o $(TEST_LIBS)
gznonblk$(EXE): gznonblk.o $(STATICLIB)
$(CC) $(CFLAGS) $(LDFLAGS) -o $@ gznonblk.o $(TEST_LIBS)
examplesh$(EXE): example.o $(SHAREDLIBV)
$(CC) $(CFLAGS) -o $@ example.o $(LDFLAGS) -L. $(SHAREDLIBV)
@ -377,6 +392,7 @@ clean: minizip-clean
rm -f *.o *.lo *~ \
example$(EXE) minigzip$(EXE) examplesh$(EXE) minigzipsh$(EXE) \
example64$(EXE) minigzip64$(EXE) \
gznonblk$(EXE) \
infcover \
libz.* foo.gz so_locations \
_match.s maketree contrib/infback9/*.o
@ -400,6 +416,7 @@ adler32.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
zutil.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
gzclose.o gzlib.o gzread.o gzwrite.o: $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
compress.o example.o minigzip.o uncompr.o: $(SRCDIR)zlib.h zconf.h
gznonblk.o: $(SRCDIR)zlib.h zconf.h
crc32.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)crc32.h
deflate.o: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
infback.o inflate.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h $(SRCDIR)inffixed.h
@ -411,6 +428,7 @@ adler32.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
zutil.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
gzclose.lo gzlib.lo gzread.lo gzwrite.lo: $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
compress.lo example.lo minigzip.lo uncompr.lo: $(SRCDIR)zlib.h zconf.h
gznonblk.lo: $(SRCDIR)zlib.h zconf.h
crc32.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)crc32.h
deflate.lo: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
infback.lo inflate.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h $(SRCDIR)inffixed.h

View File

@ -24,7 +24,7 @@ local int gz_load(gz_statep state, unsigned char *buf, unsigned len,
break;
*have += (unsigned)ret;
} while (*have < len);
if (ret < 0) {
if (ret < 0 && errno != EAGAIN && errno != EWOULDBLOCK) {
gz_error(state, Z_ERRNO, zstrerror());
return -1;
}
@ -268,6 +268,7 @@ local int gz_skip(gz_statep state, z_off64_t len) {
local z_size_t gz_read(gz_statep state, voidp buf, z_size_t len) {
z_size_t got;
unsigned n;
int load_errno;
/* if len is zero, avoid unnecessary operations */
if (len == 0)
@ -283,6 +284,7 @@ local z_size_t gz_read(gz_statep state, voidp buf, z_size_t len) {
/* get len bytes to buf, or less than len if at the end */
got = 0;
do {
load_errno = 0;
/* set n to the maximum amount of len that fits in an unsigned int */
n = (unsigned)-1;
if (n > len)
@ -307,8 +309,10 @@ local z_size_t gz_read(gz_statep state, voidp buf, z_size_t len) {
buffer */
else if (state->how == LOOK || n < (state->size << 1)) {
/* get more output, looking for header if required */
errno = 0;
if (gz_fetch(state) == -1)
return 0;
if (!state->x.have) { load_errno = errno; }
continue; /* no progress yet -- go back to copy above */
/* the copy above assures that we will leave with space in the
output buffer, allowing at least one gzungetc() to succeed */
@ -316,16 +320,20 @@ local z_size_t gz_read(gz_statep state, voidp buf, z_size_t len) {
/* large len -- read directly into user buffer */
else if (state->how == COPY) { /* read directly */
errno = 0;
if (gz_load(state, (unsigned char *)buf, n, &n) == -1)
return 0;
load_errno = errno;
}
/* large len -- decompress directly into user buffer */
else { /* state->how == GZIP */
state->strm.avail_out = n;
state->strm.next_out = (unsigned char *)buf;
errno = 0;
if (gz_decomp(state) == -1)
return 0;
load_errno = errno;
n = state->x.have;
state->x.have = 0;
}
@ -335,7 +343,7 @@ local z_size_t gz_read(gz_statep state, voidp buf, z_size_t len) {
buf = (char *)buf + n;
got += n;
state->x.pos += n;
} while (len);
} while (len && load_errno != EAGAIN && load_errno != EWOULDBLOCK);
/* return number of bytes read into user buffer */
return got;

592
test/gznonblk.c Normal file
View File

@ -0,0 +1,592 @@
/* gznonblk.c -- test non-blocking reads w/zlib compression library
* Copyright (C) 2024 Brian T. Carcich
* For conditions of distribution and use, see copyright notice in zlib.h
*/
#include <zlib.h>
#include <fcntl.h>
#include <netdb.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#define BUF_SIZE 128
const struct timeval tvfixed = { 1, 0 };
/* Wrapper to make file non-blocking */
int
make_fd_nonblocking(int fd)
{
int existing_flags = fcntl(fd, F_GETFL);
return (-1 == existing_flags)
? -2
: fcntl(fd, F_SETFL, existing_flags | O_NONBLOCK);
}
/* Server code to use gzread.c/zlib.h library:
* - Listen for, and accept, socket connection(s)
* - Make connection non-blocking
* - select(2), and (gz)read data when available
* - Exit when "-stopserver-" is received
*/
int
servermain(int argc, char** argv)
{
struct addrinfo *rp; /* getaddrinfo(3) items */
struct addrinfo hints;
struct addrinfo *result;
struct sockaddr_storage peer_addr;
socklen_t peerlen = sizeof peer_addr;
int rtn; /* Return value from many routines */
int sfd; /* Accepted socket file descriptor, for sending data */
int listenfd; /* Bound socket file descriptor, for listening */
int stopserver = 0; /* Flag for when to exit server code */
char buf[BUF_SIZE]; /* Read (gzread) buffer */
gzFile gzfi = NULL; /* gzread "file" information */
if (argc != 2) {
fprintf(stderr, "Server usage: %s port|service\n", argv[0]);
return EXIT_FAILURE;
}
/* Setup for getaddrinfo(3) call */
memset(&hints, 0, sizeof(hints));
//hints.ai_family = AF_UNSPEC; /* Allow IPv4 or IPv6 */
//hints.ai_family = AF_INET6; /* IPv6 */
hints.ai_family = AF_INET; /* IPv4 */
hints.ai_socktype = SOCK_STREAM; /* Stream socket */
hints.ai_flags = AI_PASSIVE; /* For wildcard IP address */
hints.ai_protocol = 0; /* Any protocol */
hints.ai_canonname = NULL; /* Server; nobbut else needed */
hints.ai_addr = NULL;
hints.ai_next = NULL;
/* Get address info: NULL => server; argv[1] is the port */
rtn = getaddrinfo(NULL, argv[1], &hints, &result);
if (rtn) {
fprintf(stderr, "Server %d=getaddrinfo(\"%s\",\"%s\",...)"
" failed: %s\n"
, rtn, "<null>", argv[1], gai_strerror(rtn)
);
return EXIT_FAILURE;
}
/* getaddrinfo() returns a list of address structures.
* Try each address structure until we successfully bind(2).
* If socket(2) or bind(2) or listen(2) fails, then we try the next
* address structure after closing the socket if socket(2) succeeded
*/
for (rp = result; rp != NULL; rp = rp->ai_next) {
listenfd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol);
if (listenfd == -1) { continue; }
if (!bind(listenfd, rp->ai_addr, rp->ai_addrlen)
&& !listen(listenfd, 10)
)
{
break; /* Success */
}
close(listenfd);
}
freeaddrinfo(result); /* No longer needed */
if (rp == NULL) { /* No address succeeded */
fprintf(stderr, "Server could not socket/bind/listen\n");
return EXIT_FAILURE;
}
sfd=-1;
/* Loop:
* - select(2) on EITHER listening OR connected socket
* - accept a new socket connection if listening socket has data
* - select(2)/gzread data from connected socket, print to stderr
* - Close when socket is closed
*
* zs is the number of consecutive times select(2) returns 0
*/
for (int zs = 0; !stopserver; ++zs)
{
int nfd;
fd_set rfds;
struct timeval tv;
char host[NI_MAXHOST];
char service[NI_MAXSERV];
/* Clear FD set */
FD_ZERO(&rfds);
if (gzfi)
{
/* Set accepted socket's bit if gzfi is not NULL */
FD_SET(sfd, &rfds);
nfd = sfd + 1;
}
else
{
/* Set listening socket's bit if gzfi is NULL */
FD_SET(listenfd, &rfds);
nfd = listenfd + 1;
}
/* Wait for data from selected file */
errno = 0;
tv = tvfixed;
rtn = select(nfd, &rfds, NULL, NULL, &tv);
if (rtn)
{
/* Log non-zero return values from select(2) */
fprintf(stderr, "Server %d=select(nfd,%llx,,,tv)"
"; errno=%d[%s]\n"
, rtn, *((long long*)&rfds)
, errno, strerror(errno)
);
zs = 0;
}
else
{
/* Count and log consecutive passes with no incoming data */
const char terms[] = { "|/-\\" };
fprintf(stderr, "%d%c\r", zs, terms[zs&3]);
continue;
}
if (rtn < 0) /* select failed */
{
select(0, NULL, NULL, NULL, &tv); /* finish delay */
continue; /* Ignore failed select */
}
/* to here, select(2) returned 1 */
if (gzfi) /* if accepted socket is active, handle I/O */
{
char* p;
char* pend;
int ipos;
//not necessary: if (!FD_ISSET(sfd, &rfds)) continue;
/* Read data */
errno = 0;
gzclearerr(gzfi);
rtn = gzread(gzfi, buf, sizeof buf);
fprintf(stderr, "Server %d=gzread(%d,...)"
"; errno=%d[%s]\n"
, rtn, sfd, errno, strerror(errno)
);
/* Handle EOF (rtn==0) or error (rtn<0) */
if (rtn < 1)
{
int igzerr;
const char* pgzerr = gzerror(gzfi, &igzerr);
fprintf(stderr, "Server %d=gzerror[%s]\n"
, igzerr, pgzerr ? pgzerr : "<null>"
);
gzclose(gzfi);
gzfi = NULL;
sfd = -1;
continue;
}
/* Log data read by gzread */
fprintf(stderr, "%s", "Server buf=>[");
for (pend = (p=buf) + rtn; p<pend; ++p)
{
fprintf(stderr, (32<=*p&&*p<127) ? "%c" : "<0x%02x>", *p);
}
fprintf(stderr, "%s", "]\n");
/* If "-stopserver-" was received, then set flag */
stopserver = (rtn == 12 || (rtn == 13 && !buf[12]))
&& !strncmp("-stopserver-", buf, 12);
continue; /* Done with gzread over socket; skip listenfd */
} /* if (gzfi) */
/* To here, there is no active accepted socket; select(2) result
* was for bound listening socket (listenfd), indicating a new
* connection request
*/
//not necessary: if (!FD_ISSET(listenfd, &rfds)) { continue; }
/* Accept the new connection */
errno = 0;
sfd = accept(listenfd, (struct sockaddr *)&peer_addr, &peerlen);
if (sfd < 0)
{
fprintf(stderr, "Server %d=accept(listenfd,...)"
"; errno=%d[%s]\n"
, rtn, errno, strerror(errno)
);
continue;
}
/* Make the new socket non-blocking */
errno = 0;
if (0 > make_fd_nonblocking(sfd))
{
fprintf(stderr, "Server %d=make_fd_nonblocking(%d,...)"
"; errno=%d[%s]\n"
, rtn, sfd, errno, strerror(errno)
);
close(sfd);
sfd = -1;
continue;
}
/* Log information about the newly-accepted socket */
rtn = getnameinfo((struct sockaddr *)&peer_addr, peerlen
, host, NI_MAXHOST
, service, NI_MAXSERV, NI_NUMERICSERV);
if (rtn == 0)
{
fprintf(stderr, "Server accepted connection from %s:%s\n"
, host, service
);
}
else
{
fprintf(stderr, "Server getnameinfo failed: %s\n"
, gai_strerror(rtn)
);
}
/* Allocate/open the gzFile pointer, and set its buffer size */
if (!(gzfi = gzdopen(sfd, "r")))
{
fprintf(stderr, "Server gzdopen(%d, \"r\") failed\n" , sfd);
close(sfd);
sfd = -1;
continue;
}
if (gzbuffer(gzfi, 16))
{
fprintf(stderr, "%s\n", "Server gzbuffer(gzfi, 16) failed");
gzclose(gzfi);
gzfi = NULL;
sfd = -1;
continue;
}
} // for (int zs = 0; !stopserver; ++zs)
return EXIT_SUCCESS;
}
/* Client code to use gzwrite.c/zlib.h library:
* - Open socket connection to server above
* - Make connection non-blocking
* - gzwrite command-line arguments (argv)
* - Exit after last argument has been "gzwritten"
*/
int
clientmain(int argc, char** argv)
{
struct addrinfo *rp; /* getaddrinfo(3) items */
struct addrinfo hints;
struct addrinfo *result;
int rtn; /* Return value from many routines */
int sfd; /* Socket file descriptor */
size_t len; /* Length of data to send */
char* serverhost; /* Name of server (argv[2] or argv[3] */
gzFile gzfi = NULL; /* gzread "file" information */
int final_rtn = EXIT_SUCCESS; /* Exit code; assume success */
int clientfork = argc > 2 && !strcmp(argv[2], "--client-fork");
if (argc < 3 || (argc == 3 && clientfork))
{
fprintf(stderr, "Client usage: %s port|service%s"
" serverhost msg...\n"
, argv[0], clientfork ? " --client-fork" : ""
);
return EXIT_FAILURE;
}
/* Extract server hostname apropo the command line:
*
* gznonblk portnumber serverhost ...
*
* OR
*
* gznonblk portnumber --client-fork serverhost ...
*/
serverhost = clientfork ? argv[3] : argv[2];
/* Obtain address(es) matching host/port */
memset(&hints, 0, sizeof(hints));
//hints.ai_family = AF_UNSPEC; /* Allow IPv4 or IPv6 */
//hints.ai_family = AF_INET6; /* Allow IPv6 only */
hints.ai_family = AF_INET; /* Allow IPv4 only */
hints.ai_socktype = SOCK_STREAM; /* Stream socket */
hints.ai_flags = 0;
hints.ai_protocol = 0; /* Any protocol */
hints.ai_canonname = NULL;
hints.ai_addr = NULL;
hints.ai_next = NULL;
/* Get server address info: argv[1] is the port */
errno = 0;
rtn = getaddrinfo(serverhost, argv[1], &hints, &result);
if (rtn) {
fprintf(stderr, "Client %d=getaddrinfo(\"%s\",\"%s\",...)"
" failed: %s\n"
, rtn, serverhost, argv[1], gai_strerror(rtn)
);
return EXIT_FAILURE;
}
/* getaddrinfo() returns a list of address structures.
Try each address until we successfully connect(2).
If socket(2) (or connect(2)) fails, we (close the socket
and) try the next address. */
for (rp = result; rp != NULL; rp = rp->ai_next) {
sfd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol);
if (sfd == -1) { continue; } /* Ignore failed socket creation*/
if (connect(sfd, rp->ai_addr, rp->ai_addrlen) != -1) { break; }
/* ^ Exit loop on Successful connection */
/* v Close socket and continue on Failed connection */
close(sfd);
}
freeaddrinfo(result); /* getaddrinfo result is no longer needed */
if (rp == NULL) /* No address means all failed */
{
fprintf(stderr, "Client could not connect\n");
return EXIT_FAILURE;
}
/* Allocate/open the gzFile pointer */
if (!(gzfi = gzdopen(sfd, "w")))
{
fprintf(stderr, "Client gzdopen(%d, \"w\") failed\n" , sfd);
close(sfd);
sfd = -1;
return EXIT_FAILURE;
}
/* Send remaining command-line arguments as separate gzwrites */
for (int iarg = clientfork ? 4 : 3
; final_rtn == EXIT_SUCCESS && iarg < argc
; ++iarg
)
{
int itmp; /* Unused value from gzerror */
int save_err; /* cache for errno */
if (!strcmp(argv[iarg],"--delay")) /* argument is "--delay" */
{
struct timeval tv;
tv = tvfixed;
select(0,NULL,NULL,NULL,&tv); /* Delay, then get next arg */
continue;
}
len = strlen(argv[iarg]) + 1; /* +1 for terminating null byte */
/* Write argument using gzwrite; on error log and exit */
errno = 0;
if ((rtn=gzwrite(gzfi, argv[iarg], len)) != len)
{
save_err = errno;
fprintf(stderr, "Client partial/failed %d=gzwrite[%s]"
"; %d=errno[%s]"
, rtn, gzerror(gzfi, &itmp)
, save_err, strerror(save_err)
);
final_rtn = EXIT_FAILURE;
continue;
}
/* Flush data to socket */
errno = 0;
if ((rtn=gzflush(gzfi, Z_SYNC_FLUSH)) != Z_OK)
{
save_err = errno;
fprintf(stderr, "Client partial/failed %d=gzbuffer[%s]"
"; %d=errno[%s]"
, rtn, gzerror(gzfi, &itmp)
, save_err, strerror(save_err)
);
final_rtn = EXIT_FAILURE;
continue;
}
/* End of argument loop */
} // for (...; final_return == EXIT_SUCCESS ...; ++iarg)
/* Execute final flush, close socket, return status */
gzflush(gzfi, Z_FINISH);
gzclose(gzfi);
return final_rtn;
}
int Usage(int argc, char** argv)
{
const char* arrUsage[] =
{ "Usage:"
, " gznonblk pn[[ --client-fork] srvrhost[ msg1|--delay[ msg2...]]]"
, " gznonblk --help[-long]"
, ""
, "where"
, " pn = port# or service where server will be listening"
, " --client-fork = directive to run server and fork client"
, " srvrhost = hostname of server for client to use"
, " msgN|--delay = client messages to send or delays between them"
, NULL
, ""
, "Examples:"
, ""
, " gznonblk 4444"
, " - Start server only, listening on port 4444"
, ""
, " gznonblk 4444 srvrhost message1 --delay message2 message3"
, " - Start client only, connect to server at port 4444 on srvrhost"
, " - Client"
, " - sends \"message1\""
, " - delays"
, " - sends \"message2\" and \"message3\""
, ""
, " gznonblk 4444 --client-fork 127.0.0.1 msg1 --delay -stopserver-"
, " - Fork client, connect to server at port 4444 on 127.0.0.1"
, " - Client"
, " - delays for server to start (forced when forking client)"
, " - sends \"msg1\""
, " - delays"
, " - sends \"-stopserver-\""
, " - which will stop server later"
, " - Start server, listening on port 4444"
, NULL
};
int helplong = 0;
while (--argc)
{
if (!strcmp(argv[argc], "--help")) { break; }
if (!strcmp(argv[argc], "--help-long")) { ++helplong; break; }
}
if (!argc) { return 0; }
for (char** p=(char**)arrUsage; helplong || *p; ++p)
{
if (!*p) { --helplong; continue; }
fprintf(stdout, "%s\n", *p);
}
return 1;
}
int
main(int argc, char** argv)
{
/* Check if "--client-fork" is at argument offset 2 */
int clientfork = argc > 2 && !strcmp(argv[2], "--client-fork");
if (Usage(argc, argv)) { return EXIT_SUCCESS; }
/* If command line is:
*
* gznonblk portnum serverhost ...
*
* then run client only
*/
if (argc > 2 && !clientfork) { return clientmain(argc, argv); }
/* If command line is:
*
* gznonblk portnum --clienthost serverhost ...
*
* then fork client, run server, wait for client
*/
errno = 0;
if (clientfork)
{
int rtn; /* Return value from many routines */
int wstat; /* Client return status from waitpid(2) */
pid_t pidwaited; /* Return value from waitpid(2) */
int badchild = 0; /* Non-zero flag for any client failure */
int wopts = WNOHANG; /* Options for waitpid(2) */
struct timeval tv = {3, 0}; /* Delay 3s for client to exit */
pid_t pidforked = fork(); /* Fork the client */
int save_err = errno; /* Save the fork(2) result */
/* On fork(2) error, log and exit */
if (pidforked < 0)
{
fprintf(stderr, "Server %d=fork() of client failed"
"; %d=errno[%s]\n"
, pidforked, save_err, strerror(save_err)
);
return EXIT_FAILURE;
}
/* If this is now forked child, then run client and exit */
if (!pidforked) { return clientmain(argc, argv); }
/* To here, this is the server */
fprintf(stderr, "Server %d=fork()=PID of client succeeded"
"; %d=errno[%s]\n"
, pidforked, save_err, strerror(save_err)
);
rtn = servermain(2, argv); /* Run server */
select(0, NULL, NULL, NULL, &tv); /* Delay for client to exit */
errno = 0;
pidwaited = waitpid(-1, &wstat, wopts); /* Get client status */
if (pidwaited < 1) /* Non-positive return is a failure */
{
++badchild;
fprintf(stderr, "Server child %d=waitpid(-1,...) failed"
"; %d=errno[%s]\n"
, pidwaited, errno, strerror(errno)
);
return EXIT_FAILURE;
}
if (pidwaited != pidforked) /* Wait PID must match fork PID */
{
++badchild;
fprintf(stderr, "Server child %d=waitpid(-1,...) not equal"
" to forked pid (%d)\n"
, pidwaited, pidforked
);
return EXIT_FAILURE;
}
/* Combine server status and child/client status */
return rtn | (WIFEXITED(wstat) ? WEXITSTATUS(wstat) : 1);
}
/* If command line is:
*
* gznonblk portnum
*
* then run server only
*/
if (argc == 2) { return servermain(2, argv); }
return -1;
}