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:
commit
e4ad9dc835
1
.gitignore
vendored
1
.gitignore
vendored
@ -22,6 +22,7 @@
|
||||
/minigzip
|
||||
/minigzip64
|
||||
/minigzipsh
|
||||
/gznonblk
|
||||
/zlib.pc
|
||||
/configure.log
|
||||
/build
|
||||
|
@ -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
|
||||
|
20
Makefile.in
20
Makefile.in
@ -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
|
||||
|
12
gzread.c
12
gzread.c
@ -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
592
test/gznonblk.c
Normal 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;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user