688 lines
17 KiB
C
688 lines
17 KiB
C
/*
|
|
* test-simple-ipc.c: verify that the Inter-Process Communication works.
|
|
*/
|
|
|
|
#include "test-tool.h"
|
|
#include "gettext.h"
|
|
#include "simple-ipc.h"
|
|
#include "parse-options.h"
|
|
#include "thread-utils.h"
|
|
#include "strvec.h"
|
|
#include "run-command.h"
|
|
#include "trace2.h"
|
|
|
|
#ifndef SUPPORTS_SIMPLE_IPC
|
|
int cmd__simple_ipc(int argc, const char **argv)
|
|
{
|
|
die("simple IPC not available on this platform");
|
|
}
|
|
#else
|
|
|
|
/*
|
|
* The test daemon defines an "application callback" that supports a
|
|
* series of commands (see `test_app_cb()`).
|
|
*
|
|
* Unknown commands are caught here and we send an error message back
|
|
* to the client process.
|
|
*/
|
|
static int app__unhandled_command(const char *command,
|
|
ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
struct strbuf buf = STRBUF_INIT;
|
|
int ret;
|
|
|
|
strbuf_addf(&buf, "unhandled command: %s", command);
|
|
ret = reply_cb(reply_data, buf.buf, buf.len);
|
|
strbuf_release(&buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Reply with a single very large buffer. This is to ensure that
|
|
* long response are properly handled -- whether the chunking occurs
|
|
* in the kernel or in the (probably pkt-line) layer.
|
|
*/
|
|
#define BIG_ROWS (10000)
|
|
static int app__big_command(ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
struct strbuf buf = STRBUF_INIT;
|
|
int row;
|
|
int ret;
|
|
|
|
for (row = 0; row < BIG_ROWS; row++)
|
|
strbuf_addf(&buf, "big: %.75d\n", row);
|
|
|
|
ret = reply_cb(reply_data, buf.buf, buf.len);
|
|
strbuf_release(&buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Reply with a series of lines. This is to ensure that we can incrementally
|
|
* compute the response and chunk it to the client.
|
|
*/
|
|
#define CHUNK_ROWS (10000)
|
|
static int app__chunk_command(ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
struct strbuf buf = STRBUF_INIT;
|
|
int row;
|
|
int ret;
|
|
|
|
for (row = 0; row < CHUNK_ROWS; row++) {
|
|
strbuf_setlen(&buf, 0);
|
|
strbuf_addf(&buf, "big: %.75d\n", row);
|
|
ret = reply_cb(reply_data, buf.buf, buf.len);
|
|
}
|
|
|
|
strbuf_release(&buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Slowly reply with a series of lines. This is to model an expensive to
|
|
* compute chunked response (which might happen if this callback is running
|
|
* in a thread and is fighting for a lock with other threads).
|
|
*/
|
|
#define SLOW_ROWS (1000)
|
|
#define SLOW_DELAY_MS (10)
|
|
static int app__slow_command(ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
struct strbuf buf = STRBUF_INIT;
|
|
int row;
|
|
int ret;
|
|
|
|
for (row = 0; row < SLOW_ROWS; row++) {
|
|
strbuf_setlen(&buf, 0);
|
|
strbuf_addf(&buf, "big: %.75d\n", row);
|
|
ret = reply_cb(reply_data, buf.buf, buf.len);
|
|
sleep_millisec(SLOW_DELAY_MS);
|
|
}
|
|
|
|
strbuf_release(&buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* The client sent a command followed by a (possibly very) large buffer.
|
|
*/
|
|
static int app__sendbytes_command(const char *received, size_t received_len,
|
|
ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
struct strbuf buf_resp = STRBUF_INIT;
|
|
const char *p = "?";
|
|
int len_ballast = 0;
|
|
int k;
|
|
int errs = 0;
|
|
int ret;
|
|
|
|
/*
|
|
* The test is setup to send:
|
|
* "sendbytes" SP <n * char>
|
|
*/
|
|
if (received_len < strlen("sendbytes "))
|
|
BUG("received_len is short in app__sendbytes_command");
|
|
|
|
if (skip_prefix(received, "sendbytes ", &p))
|
|
len_ballast = strlen(p);
|
|
|
|
/*
|
|
* Verify that the ballast is n copies of a single letter.
|
|
* And that the multi-threaded IO layer didn't cross the streams.
|
|
*/
|
|
for (k = 1; k < len_ballast; k++)
|
|
if (p[k] != p[0])
|
|
errs++;
|
|
|
|
if (errs)
|
|
strbuf_addf(&buf_resp, "errs:%d\n", errs);
|
|
else
|
|
strbuf_addf(&buf_resp, "rcvd:%c%08d\n", p[0], len_ballast);
|
|
|
|
ret = reply_cb(reply_data, buf_resp.buf, buf_resp.len);
|
|
|
|
strbuf_release(&buf_resp);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* An arbitrary fixed address to verify that the application instance
|
|
* data is handled properly.
|
|
*/
|
|
static int my_app_data = 42;
|
|
|
|
static ipc_server_application_cb test_app_cb;
|
|
|
|
/*
|
|
* This is the "application callback" that sits on top of the
|
|
* "ipc-server". It completely defines the set of commands supported
|
|
* by this application.
|
|
*/
|
|
static int test_app_cb(void *application_data,
|
|
const char *command, size_t command_len,
|
|
ipc_server_reply_cb *reply_cb,
|
|
struct ipc_server_reply_data *reply_data)
|
|
{
|
|
/*
|
|
* Verify that we received the application-data that we passed
|
|
* when we started the ipc-server. (We have several layers of
|
|
* callbacks calling callbacks and it's easy to get things mixed
|
|
* up (especially when some are "void*").)
|
|
*/
|
|
if (application_data != (void*)&my_app_data)
|
|
BUG("application_cb: application_data pointer wrong");
|
|
|
|
if (command_len == 4 && !strncmp(command, "quit", 4)) {
|
|
/*
|
|
* The client sent a "quit" command. This is an async
|
|
* request for the server to shutdown.
|
|
*
|
|
* We DO NOT send the client a response message
|
|
* (because we have nothing to say and the other
|
|
* server threads have not yet stopped).
|
|
*
|
|
* Tell the ipc-server layer to start shutting down.
|
|
* This includes: stop listening for new connections
|
|
* on the socket/pipe and telling all worker threads
|
|
* to finish/drain their outgoing responses to other
|
|
* clients.
|
|
*
|
|
* This DOES NOT force an immediate sync shutdown.
|
|
*/
|
|
return SIMPLE_IPC_QUIT;
|
|
}
|
|
|
|
if (command_len == 4 && !strncmp(command, "ping", 4)) {
|
|
const char *answer = "pong";
|
|
return reply_cb(reply_data, answer, strlen(answer));
|
|
}
|
|
|
|
if (command_len == 3 && !strncmp(command, "big", 3))
|
|
return app__big_command(reply_cb, reply_data);
|
|
|
|
if (command_len == 5 && !strncmp(command, "chunk", 5))
|
|
return app__chunk_command(reply_cb, reply_data);
|
|
|
|
if (command_len == 4 && !strncmp(command, "slow", 4))
|
|
return app__slow_command(reply_cb, reply_data);
|
|
|
|
if (command_len >= 10 && starts_with(command, "sendbytes "))
|
|
return app__sendbytes_command(command, command_len,
|
|
reply_cb, reply_data);
|
|
|
|
return app__unhandled_command(command, reply_cb, reply_data);
|
|
}
|
|
|
|
struct cl_args
|
|
{
|
|
const char *subcommand;
|
|
const char *path;
|
|
const char *token;
|
|
|
|
int nr_threads;
|
|
int max_wait_sec;
|
|
int bytecount;
|
|
int batchsize;
|
|
|
|
char bytevalue;
|
|
};
|
|
|
|
static struct cl_args cl_args = {
|
|
.subcommand = NULL,
|
|
.path = "ipc-test",
|
|
.token = NULL,
|
|
|
|
.nr_threads = 5,
|
|
.max_wait_sec = 60,
|
|
.bytecount = 1024,
|
|
.batchsize = 10,
|
|
|
|
.bytevalue = 'x',
|
|
};
|
|
|
|
/*
|
|
* This process will run as a simple-ipc server and listen for IPC commands
|
|
* from client processes.
|
|
*/
|
|
static int daemon__run_server(void)
|
|
{
|
|
int ret;
|
|
|
|
struct ipc_server_opts opts = {
|
|
.nr_threads = cl_args.nr_threads,
|
|
};
|
|
|
|
/*
|
|
* Synchronously run the ipc-server. We don't need any application
|
|
* instance data, so pass an arbitrary pointer (that we'll later
|
|
* verify made the round trip).
|
|
*/
|
|
ret = ipc_server_run(cl_args.path, &opts, test_app_cb, (void*)&my_app_data);
|
|
if (ret == -2)
|
|
error("socket/pipe already in use: '%s'", cl_args.path);
|
|
else if (ret == -1)
|
|
error_errno("could not start server on: '%s'", cl_args.path);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static start_bg_wait_cb bg_wait_cb;
|
|
|
|
static int bg_wait_cb(const struct child_process *cp UNUSED,
|
|
void *cb_data UNUSED)
|
|
{
|
|
int s = ipc_get_active_state(cl_args.path);
|
|
|
|
switch (s) {
|
|
case IPC_STATE__LISTENING:
|
|
/* child is "ready" */
|
|
return 0;
|
|
|
|
case IPC_STATE__NOT_LISTENING:
|
|
case IPC_STATE__PATH_NOT_FOUND:
|
|
/* give child more time */
|
|
return 1;
|
|
|
|
default:
|
|
case IPC_STATE__INVALID_PATH:
|
|
case IPC_STATE__OTHER_ERROR:
|
|
/* all the time in world won't help */
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static int daemon__start_server(void)
|
|
{
|
|
struct child_process cp = CHILD_PROCESS_INIT;
|
|
enum start_bg_result sbgr;
|
|
|
|
strvec_push(&cp.args, "test-tool");
|
|
strvec_push(&cp.args, "simple-ipc");
|
|
strvec_push(&cp.args, "run-daemon");
|
|
strvec_pushf(&cp.args, "--name=%s", cl_args.path);
|
|
strvec_pushf(&cp.args, "--threads=%d", cl_args.nr_threads);
|
|
|
|
cp.no_stdin = 1;
|
|
cp.no_stdout = 1;
|
|
cp.no_stderr = 1;
|
|
|
|
sbgr = start_bg_command(&cp, bg_wait_cb, NULL, cl_args.max_wait_sec);
|
|
|
|
switch (sbgr) {
|
|
case SBGR_READY:
|
|
return 0;
|
|
|
|
default:
|
|
case SBGR_ERROR:
|
|
case SBGR_CB_ERROR:
|
|
return error("daemon failed to start");
|
|
|
|
case SBGR_TIMEOUT:
|
|
return error("daemon not online yet");
|
|
|
|
case SBGR_DIED:
|
|
return error("daemon terminated");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* This process will run a quick probe to see if a simple-ipc server
|
|
* is active on this path.
|
|
*
|
|
* Returns 0 if the server is alive.
|
|
*/
|
|
static int client__probe_server(void)
|
|
{
|
|
enum ipc_active_state s;
|
|
|
|
s = ipc_get_active_state(cl_args.path);
|
|
switch (s) {
|
|
case IPC_STATE__LISTENING:
|
|
return 0;
|
|
|
|
case IPC_STATE__NOT_LISTENING:
|
|
return error("no server listening at '%s'", cl_args.path);
|
|
|
|
case IPC_STATE__PATH_NOT_FOUND:
|
|
return error("path not found '%s'", cl_args.path);
|
|
|
|
case IPC_STATE__INVALID_PATH:
|
|
return error("invalid pipe/socket name '%s'", cl_args.path);
|
|
|
|
case IPC_STATE__OTHER_ERROR:
|
|
default:
|
|
return error("other error for '%s'", cl_args.path);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Send an IPC command token to an already-running server daemon and
|
|
* print the response.
|
|
*
|
|
* This is a simple 1 word command/token that `test_app_cb()` (in the
|
|
* daemon process) will understand.
|
|
*/
|
|
static int client__send_ipc(void)
|
|
{
|
|
const char *command = "(no-command)";
|
|
struct strbuf buf = STRBUF_INIT;
|
|
struct ipc_client_connect_options options
|
|
= IPC_CLIENT_CONNECT_OPTIONS_INIT;
|
|
|
|
if (cl_args.token && *cl_args.token)
|
|
command = cl_args.token;
|
|
|
|
options.wait_if_busy = 1;
|
|
options.wait_if_not_found = 0;
|
|
|
|
if (!ipc_client_send_command(cl_args.path, &options,
|
|
command, strlen(command),
|
|
&buf)) {
|
|
if (buf.len) {
|
|
printf("%s\n", buf.buf);
|
|
fflush(stdout);
|
|
}
|
|
strbuf_release(&buf);
|
|
|
|
return 0;
|
|
}
|
|
|
|
return error("failed to send '%s' to '%s'", command, cl_args.path);
|
|
}
|
|
|
|
/*
|
|
* Send an IPC command to an already-running server and ask it to
|
|
* shutdown. "send quit" is an async request and queues a shutdown
|
|
* event in the server, so we spin and wait here for it to actually
|
|
* shutdown to make the unit tests a little easier to write.
|
|
*/
|
|
static int client__stop_server(void)
|
|
{
|
|
int ret;
|
|
time_t time_limit, now;
|
|
enum ipc_active_state s;
|
|
|
|
time(&time_limit);
|
|
time_limit += cl_args.max_wait_sec;
|
|
|
|
cl_args.token = "quit";
|
|
|
|
ret = client__send_ipc();
|
|
if (ret)
|
|
return ret;
|
|
|
|
for (;;) {
|
|
sleep_millisec(100);
|
|
|
|
s = ipc_get_active_state(cl_args.path);
|
|
|
|
if (s != IPC_STATE__LISTENING) {
|
|
/*
|
|
* The socket/pipe is gone and/or has stopped
|
|
* responding. Lets assume that the daemon
|
|
* process has exited too.
|
|
*/
|
|
return 0;
|
|
}
|
|
|
|
time(&now);
|
|
if (now > time_limit)
|
|
return error("daemon has not shutdown yet");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Send an IPC command followed by ballast to confirm that a large
|
|
* message can be sent and that the kernel or pkt-line layers will
|
|
* properly chunk it and that the daemon receives the entire message.
|
|
*/
|
|
static int do_sendbytes(int bytecount, char byte, const char *path,
|
|
const struct ipc_client_connect_options *options)
|
|
{
|
|
struct strbuf buf_send = STRBUF_INIT;
|
|
struct strbuf buf_resp = STRBUF_INIT;
|
|
|
|
strbuf_addstr(&buf_send, "sendbytes ");
|
|
strbuf_addchars(&buf_send, byte, bytecount);
|
|
|
|
if (!ipc_client_send_command(path, options,
|
|
buf_send.buf, buf_send.len,
|
|
&buf_resp)) {
|
|
strbuf_rtrim(&buf_resp);
|
|
printf("sent:%c%08d %s\n", byte, bytecount, buf_resp.buf);
|
|
fflush(stdout);
|
|
strbuf_release(&buf_send);
|
|
strbuf_release(&buf_resp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
return error("client failed to sendbytes(%d, '%c') to '%s'",
|
|
bytecount, byte, path);
|
|
}
|
|
|
|
/*
|
|
* Send an IPC command with ballast to an already-running server daemon.
|
|
*/
|
|
static int client__sendbytes(void)
|
|
{
|
|
struct ipc_client_connect_options options
|
|
= IPC_CLIENT_CONNECT_OPTIONS_INIT;
|
|
|
|
options.wait_if_busy = 1;
|
|
options.wait_if_not_found = 0;
|
|
options.uds_disallow_chdir = 0;
|
|
|
|
return do_sendbytes(cl_args.bytecount, cl_args.bytevalue, cl_args.path,
|
|
&options);
|
|
}
|
|
|
|
struct multiple_thread_data {
|
|
pthread_t pthread_id;
|
|
struct multiple_thread_data *next;
|
|
const char *path;
|
|
int bytecount;
|
|
int batchsize;
|
|
int sum_errors;
|
|
int sum_good;
|
|
char letter;
|
|
};
|
|
|
|
static void *multiple_thread_proc(void *_multiple_thread_data)
|
|
{
|
|
struct multiple_thread_data *d = _multiple_thread_data;
|
|
int k;
|
|
struct ipc_client_connect_options options
|
|
= IPC_CLIENT_CONNECT_OPTIONS_INIT;
|
|
|
|
options.wait_if_busy = 1;
|
|
options.wait_if_not_found = 0;
|
|
/*
|
|
* A multi-threaded client should not be randomly calling chdir().
|
|
* The test will pass without this restriction because the test is
|
|
* not otherwise accessing the filesystem, but it makes us honest.
|
|
*/
|
|
options.uds_disallow_chdir = 1;
|
|
|
|
trace2_thread_start("multiple");
|
|
|
|
for (k = 0; k < d->batchsize; k++) {
|
|
if (do_sendbytes(d->bytecount + k, d->letter, d->path, &options))
|
|
d->sum_errors++;
|
|
else
|
|
d->sum_good++;
|
|
}
|
|
|
|
trace2_thread_exit();
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Start a client-side thread pool. Each thread sends a series of
|
|
* IPC requests. Each request is on a new connection to the server.
|
|
*/
|
|
static int client__multiple(void)
|
|
{
|
|
struct multiple_thread_data *list = NULL;
|
|
int k;
|
|
int sum_join_errors = 0;
|
|
int sum_thread_errors = 0;
|
|
int sum_good = 0;
|
|
|
|
for (k = 0; k < cl_args.nr_threads; k++) {
|
|
struct multiple_thread_data *d = xcalloc(1, sizeof(*d));
|
|
d->next = list;
|
|
d->path = cl_args.path;
|
|
d->bytecount = cl_args.bytecount + cl_args.batchsize*(k/26);
|
|
d->batchsize = cl_args.batchsize;
|
|
d->sum_errors = 0;
|
|
d->sum_good = 0;
|
|
d->letter = 'A' + (k % 26);
|
|
|
|
if (pthread_create(&d->pthread_id, NULL, multiple_thread_proc, d)) {
|
|
warning("failed to create thread[%d] skipping remainder", k);
|
|
free(d);
|
|
break;
|
|
}
|
|
|
|
list = d;
|
|
}
|
|
|
|
while (list) {
|
|
struct multiple_thread_data *d = list;
|
|
|
|
if (pthread_join(d->pthread_id, NULL))
|
|
sum_join_errors++;
|
|
|
|
sum_thread_errors += d->sum_errors;
|
|
sum_good += d->sum_good;
|
|
|
|
list = d->next;
|
|
free(d);
|
|
}
|
|
|
|
printf("client (good %d) (join %d), (errors %d)\n",
|
|
sum_good, sum_join_errors, sum_thread_errors);
|
|
|
|
return (sum_join_errors + sum_thread_errors) ? 1 : 0;
|
|
}
|
|
|
|
int cmd__simple_ipc(int argc, const char **argv)
|
|
{
|
|
const char * const simple_ipc_usage[] = {
|
|
N_("test-helper simple-ipc is-active [<name>] [<options>]"),
|
|
N_("test-helper simple-ipc run-daemon [<name>] [<threads>]"),
|
|
N_("test-helper simple-ipc start-daemon [<name>] [<threads>] [<max-wait>]"),
|
|
N_("test-helper simple-ipc stop-daemon [<name>] [<max-wait>]"),
|
|
N_("test-helper simple-ipc send [<name>] [<token>]"),
|
|
N_("test-helper simple-ipc sendbytes [<name>] [<bytecount>] [<byte>]"),
|
|
N_("test-helper simple-ipc multiple [<name>] [<threads>] [<bytecount>] [<batchsize>]"),
|
|
NULL
|
|
};
|
|
|
|
const char *bytevalue = NULL;
|
|
|
|
struct option options[] = {
|
|
#ifndef GIT_WINDOWS_NATIVE
|
|
OPT_STRING(0, "name", &cl_args.path, N_("name"), N_("name or pathname of unix domain socket")),
|
|
#else
|
|
OPT_STRING(0, "name", &cl_args.path, N_("name"), N_("named-pipe name")),
|
|
#endif
|
|
OPT_INTEGER(0, "threads", &cl_args.nr_threads, N_("number of threads in server thread pool")),
|
|
OPT_INTEGER(0, "max-wait", &cl_args.max_wait_sec, N_("seconds to wait for daemon to start or stop")),
|
|
|
|
OPT_INTEGER(0, "bytecount", &cl_args.bytecount, N_("number of bytes")),
|
|
OPT_INTEGER(0, "batchsize", &cl_args.batchsize, N_("number of requests per thread")),
|
|
|
|
OPT_STRING(0, "byte", &bytevalue, N_("byte"), N_("ballast character")),
|
|
OPT_STRING(0, "token", &cl_args.token, N_("token"), N_("command token to send to the server")),
|
|
|
|
OPT_END()
|
|
};
|
|
|
|
if (argc < 2)
|
|
usage_with_options(simple_ipc_usage, options);
|
|
|
|
if (argc == 2 && !strcmp(argv[1], "-h"))
|
|
usage_with_options(simple_ipc_usage, options);
|
|
|
|
if (argc == 2 && !strcmp(argv[1], "SUPPORTS_SIMPLE_IPC"))
|
|
return 0;
|
|
|
|
cl_args.subcommand = argv[1];
|
|
|
|
argc--;
|
|
argv++;
|
|
|
|
argc = parse_options(argc, argv, NULL, options, simple_ipc_usage, 0);
|
|
|
|
if (cl_args.nr_threads < 1)
|
|
cl_args.nr_threads = 1;
|
|
if (cl_args.max_wait_sec < 0)
|
|
cl_args.max_wait_sec = 0;
|
|
if (cl_args.bytecount < 1)
|
|
cl_args.bytecount = 1;
|
|
if (cl_args.batchsize < 1)
|
|
cl_args.batchsize = 1;
|
|
|
|
if (bytevalue && *bytevalue)
|
|
cl_args.bytevalue = bytevalue[0];
|
|
|
|
/*
|
|
* Use '!!' on all dispatch functions to map from `error()` style
|
|
* (returns -1) style to `test_must_fail` style (expects 1). This
|
|
* makes shell error messages less confusing.
|
|
*/
|
|
|
|
if (!strcmp(cl_args.subcommand, "is-active"))
|
|
return !!client__probe_server();
|
|
|
|
if (!strcmp(cl_args.subcommand, "run-daemon"))
|
|
return !!daemon__run_server();
|
|
|
|
if (!strcmp(cl_args.subcommand, "start-daemon"))
|
|
return !!daemon__start_server();
|
|
|
|
/*
|
|
* Client commands follow. Ensure a server is running before
|
|
* sending any data. This might be overkill, but then again
|
|
* this is a test harness.
|
|
*/
|
|
|
|
if (!strcmp(cl_args.subcommand, "stop-daemon")) {
|
|
if (client__probe_server())
|
|
return 1;
|
|
return !!client__stop_server();
|
|
}
|
|
|
|
if (!strcmp(cl_args.subcommand, "send")) {
|
|
if (client__probe_server())
|
|
return 1;
|
|
return !!client__send_ipc();
|
|
}
|
|
|
|
if (!strcmp(cl_args.subcommand, "sendbytes")) {
|
|
if (client__probe_server())
|
|
return 1;
|
|
return !!client__sendbytes();
|
|
}
|
|
|
|
if (!strcmp(cl_args.subcommand, "multiple")) {
|
|
if (client__probe_server())
|
|
return 1;
|
|
return !!client__multiple();
|
|
}
|
|
|
|
die("Unhandled subcommand: '%s'", cl_args.subcommand);
|
|
}
|
|
#endif
|