mirror of
https://github.com/warmcat/libwebsockets.git
synced 2025-03-16 00:00:07 +01:00

This is a huge patch that should be a global NOP. For unix type platforms it enables -Wconversion to issue warnings (-> error) for all automatic casts that seem less than ideal but are normally concealed by the toolchain. This is things like passing an int to a size_t argument. Once enabled, I went through all args on my default build (which build most things) and tried to make the removed default cast explicit. With that approach it neither change nor bloat the code, since it compiles to whatever it was doing before, just with the casts made explicit... in a few cases I changed some length args from int to size_t but largely left the causes alone. From now on, new code that is relying on less than ideal casting will complain and nudge me to improve it by warnings.
221 lines
5.1 KiB
C
221 lines
5.1 KiB
C
#include <libwebsockets.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#if defined(WIN32)
|
|
#define HAVE_STRUCT_TIMESPEC
|
|
#if defined(pid_t)
|
|
#undef pid_t
|
|
#endif
|
|
#endif
|
|
|
|
static int nclients = 11;
|
|
unsigned char msg[LWS_PRE+128];
|
|
static int message_delay = 500000; // microseconds
|
|
static int connection_delay = 100000; // microseconds
|
|
static struct lws_context *context;
|
|
static const char *server_address = "localhost", *pro = "lws-minimal";
|
|
static int interrupted = 0, port = 7681, ssl_connection = 0;
|
|
|
|
static int connect_client()
|
|
{
|
|
struct lws_client_connect_info i;
|
|
|
|
memset(&i, 0, sizeof(i));
|
|
|
|
i.context = context;
|
|
i.port = port;
|
|
i.address = server_address;
|
|
i.path = "/";
|
|
i.host = i.address;
|
|
i.origin = i.address;
|
|
i.ssl_connection = ssl_connection;
|
|
i.protocol = pro;
|
|
i.local_protocol_name = pro;
|
|
|
|
//usleep(connection_delay);
|
|
lwsl_notice("%s: connection %s:%d\n", __func__, i.address, i.port);
|
|
if (!lws_client_connect_via_info(&i)) return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
callback(struct lws *wsi, enum lws_callback_reasons reason,
|
|
void *user, void *in, size_t len)
|
|
{
|
|
int m= 0, n = 0;
|
|
short r;
|
|
#if defined(_DEBUG)
|
|
size_t remain;
|
|
int first = 0, final = 0;
|
|
#endif
|
|
|
|
//lwsl_notice("callback called with reason %d\n", reason);
|
|
switch (reason) {
|
|
|
|
case LWS_CALLBACK_PROTOCOL_INIT:
|
|
for (n = 0; n < nclients; n++)
|
|
connect_client();
|
|
break;
|
|
|
|
case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
|
|
lwsl_err("CLIENT_CONNECTION_ERROR: %s\n", in ? (char *)in :
|
|
"(null)");
|
|
if(--nclients == 0) interrupted = 1;
|
|
break;
|
|
|
|
/* --- client callbacks --- */
|
|
|
|
case LWS_CALLBACK_CLIENT_ESTABLISHED:
|
|
lws_callback_on_writable(wsi);
|
|
lwsl_user("%s: established connection, wsi = %p\n",
|
|
__func__, wsi);
|
|
break;
|
|
|
|
case LWS_CALLBACK_CLIENT_CLOSED:
|
|
lwsl_user("%s: CLOSED\n", __func__);
|
|
if(--nclients == 0) interrupted = 1;
|
|
break;
|
|
|
|
case LWS_CALLBACK_CLIENT_WRITEABLE:
|
|
|
|
m = lws_write(wsi, msg + LWS_PRE, 128, LWS_WRITE_TEXT);
|
|
if (m < 128) {
|
|
lwsl_err("sending message failed: %d < %d\n", m, n);
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
* Schedule the timer after minimum message delay plus the
|
|
* random number of centiseconds.
|
|
*/
|
|
if (lws_get_random(lws_get_context(wsi), &r, 2) == 2) {
|
|
n = message_delay + 10000*(r % 100);
|
|
lwsl_debug("set timer on %d usecs\n", n);
|
|
lws_set_timer_usecs(wsi, n);
|
|
}
|
|
break;
|
|
|
|
case LWS_CALLBACK_TIMER:
|
|
// Let the main loop know we want to send another message to the
|
|
// server
|
|
lws_callback_on_writable(wsi);
|
|
break;
|
|
|
|
case LWS_CALLBACK_CLIENT_RECEIVE:
|
|
#if defined(_DEBUG)
|
|
first = lws_is_first_fragment(wsi);
|
|
final = lws_is_final_fragment(wsi);
|
|
remain = lws_remaining_packet_payload(wsi);
|
|
lwsl_debug("LWS_CALLBACK_RECEIVE: len = %lu, first = %d, "
|
|
"final = %d, remains = %lu\n",
|
|
(unsigned long)len, first, final,
|
|
(unsigned long)remain);
|
|
#endif
|
|
break;
|
|
|
|
case LWS_CALLBACK_WS_PEER_INITIATED_CLOSE:
|
|
lwsl_notice("server initiated connection close: len = %lu, "
|
|
"in = %s\n", (unsigned long)len, (char*)in);
|
|
return 0;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return lws_callback_http_dummy(wsi, reason, user, in, len);
|
|
}
|
|
|
|
static const struct lws_protocols protocols[] = {
|
|
{ "spam-rx-tx", callback, 4096, 4096, 0, NULL, 0 },
|
|
{ NULL, NULL, 0, 0 }
|
|
};
|
|
|
|
static void
|
|
sigint_handler(int sig)
|
|
{
|
|
interrupted = 1;
|
|
}
|
|
|
|
int main(int argc, const char **argv)
|
|
{
|
|
struct lws_context_creation_info info;
|
|
const char *p;
|
|
int n = 0, logs =
|
|
LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
|
|
#ifndef WIN32
|
|
srandom((unsigned int)time(0));
|
|
#endif
|
|
|
|
memset(msg, 'x', sizeof(msg));
|
|
|
|
signal(SIGINT, sigint_handler);
|
|
|
|
if (lws_cmdline_option(argc, argv, "-d"))
|
|
logs |= LLL_INFO | LLL_DEBUG;
|
|
|
|
lws_set_log_level(logs, NULL);
|
|
|
|
memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
|
|
info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;
|
|
info.port = CONTEXT_PORT_NO_LISTEN; /* we do not run any server */
|
|
info.protocols = protocols;
|
|
#if defined(LWS_WITH_MBEDTLS) || defined(USE_WOLFSSL)
|
|
/*
|
|
* OpenSSL uses the system trust store. mbedTLS has to be told which
|
|
* CA to trust explicitly.
|
|
*/
|
|
info.client_ssl_ca_filepath = "./libwebsockets.org.cer";
|
|
#endif
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-h"))) {
|
|
server_address = p;
|
|
}
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-s"))) {
|
|
ssl_connection |=
|
|
LCCSCF_USE_SSL |
|
|
LCCSCF_ALLOW_SELFSIGNED |
|
|
LCCSCF_SKIP_SERVER_CERT_HOSTNAME_CHECK;
|
|
}
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-p")))
|
|
port = atoi(p);
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-n"))) {
|
|
n = atoi(p);
|
|
if (n < 1)
|
|
n = 1;
|
|
if (n < nclients)
|
|
nclients = n;
|
|
lwsl_notice("Start test clients: %d\n", nclients);
|
|
}
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-c"))) {
|
|
connection_delay = atoi(p);
|
|
lwsl_notice("Connection delay: %d\n", connection_delay);
|
|
}
|
|
|
|
if ((p = lws_cmdline_option(argc, argv, "-m"))) {
|
|
message_delay = atoi(p);
|
|
lwsl_notice("Message delay: %d\n", connection_delay);
|
|
}
|
|
|
|
info.fd_limit_per_thread = (unsigned int)(1 + nclients + 1);
|
|
|
|
context = lws_create_context(&info);
|
|
if (!context) {
|
|
lwsl_err("lws init failed\n");
|
|
return 1;
|
|
}
|
|
|
|
while (n >= 0 && !interrupted)
|
|
n = lws_service(context, 0);
|
|
|
|
lwsl_notice("%s: exiting service loop. n = %d, interrupted = %d\n",
|
|
__func__, n, interrupted);
|
|
|
|
lws_context_destroy(context);
|
|
|
|
return 0;
|
|
}
|