1d4afb5ceSopenharmony_ci/* 2d4afb5ceSopenharmony_ci * lws-minimal-ws-server-threads-foreign-smp 3d4afb5ceSopenharmony_ci * 4d4afb5ceSopenharmony_ci * Written in 2010-2020 by Andy Green <andy@warmcat.com> 5d4afb5ceSopenharmony_ci * 6d4afb5ceSopenharmony_ci * This file is made available under the Creative Commons CC0 1.0 7d4afb5ceSopenharmony_ci * Universal Public Domain Dedication. 8d4afb5ceSopenharmony_ci * 9d4afb5ceSopenharmony_ci * This demonstrates a minimal ws server that can cooperate with 10d4afb5ceSopenharmony_ci * other threads cleanly. Two other threads are started, which fill 11d4afb5ceSopenharmony_ci * a ringbuffer with strings at 10Hz. 12d4afb5ceSopenharmony_ci * 13d4afb5ceSopenharmony_ci * The actual work and thread spawning etc are done in the protocol 14d4afb5ceSopenharmony_ci * implementation in protocol_lws_minimal.c. 15d4afb5ceSopenharmony_ci * 16d4afb5ceSopenharmony_ci * To keep it simple, it serves stuff in the subdirectory "./mount-origin" of 17d4afb5ceSopenharmony_ci * the directory it was started in. 18d4afb5ceSopenharmony_ci * You can change that by changing mount.origin. 19d4afb5ceSopenharmony_ci */ 20d4afb5ceSopenharmony_ci 21d4afb5ceSopenharmony_ci#include <libwebsockets.h> 22d4afb5ceSopenharmony_ci#include <string.h> 23d4afb5ceSopenharmony_ci#include <signal.h> 24d4afb5ceSopenharmony_ci#if defined(WIN32) 25d4afb5ceSopenharmony_ci#define HAVE_STRUCT_TIMESPEC 26d4afb5ceSopenharmony_ci#if defined(pid_t) 27d4afb5ceSopenharmony_ci#undef pid_t 28d4afb5ceSopenharmony_ci#endif 29d4afb5ceSopenharmony_ci#endif 30d4afb5ceSopenharmony_ci#include <pthread.h> 31d4afb5ceSopenharmony_ci#include <uv.h> 32d4afb5ceSopenharmony_ci 33d4afb5ceSopenharmony_ci#define COUNT_THREADS 5 34d4afb5ceSopenharmony_ci 35d4afb5ceSopenharmony_ci#define LWS_PLUGIN_STATIC 36d4afb5ceSopenharmony_ci#include "protocol_lws_minimal.c" 37d4afb5ceSopenharmony_ci 38d4afb5ceSopenharmony_cistatic struct lws_protocols protocols[] = { 39d4afb5ceSopenharmony_ci { "http", lws_callback_http_dummy, 0, 0, 0, NULL, 0 }, 40d4afb5ceSopenharmony_ci LWS_PLUGIN_PROTOCOL_MINIMAL, 41d4afb5ceSopenharmony_ci LWS_PROTOCOL_LIST_TERM 42d4afb5ceSopenharmony_ci}; 43d4afb5ceSopenharmony_ci 44d4afb5ceSopenharmony_cistatic struct lws_context *context; 45d4afb5ceSopenharmony_cistatic int interrupted; 46d4afb5ceSopenharmony_cistatic uv_loop_t loop[COUNT_THREADS]; 47d4afb5ceSopenharmony_cistatic uv_signal_t *s, signal_outer[COUNT_THREADS]; 48d4afb5ceSopenharmony_ci 49d4afb5ceSopenharmony_cistatic const struct lws_http_mount mount = { 50d4afb5ceSopenharmony_ci /* .mount_next */ NULL, /* linked-list "next" */ 51d4afb5ceSopenharmony_ci /* .mountpoint */ "/", /* mountpoint URL */ 52d4afb5ceSopenharmony_ci /* .origin */ "./mount-origin", /* serve from dir */ 53d4afb5ceSopenharmony_ci /* .def */ "index.html", /* default filename */ 54d4afb5ceSopenharmony_ci /* .protocol */ NULL, 55d4afb5ceSopenharmony_ci /* .cgienv */ NULL, 56d4afb5ceSopenharmony_ci /* .extra_mimetypes */ NULL, 57d4afb5ceSopenharmony_ci /* .interpret */ NULL, 58d4afb5ceSopenharmony_ci /* .cgi_timeout */ 0, 59d4afb5ceSopenharmony_ci /* .cache_max_age */ 0, 60d4afb5ceSopenharmony_ci /* .auth_mask */ 0, 61d4afb5ceSopenharmony_ci /* .cache_reusable */ 0, 62d4afb5ceSopenharmony_ci /* .cache_revalidate */ 0, 63d4afb5ceSopenharmony_ci /* .cache_intermediaries */ 0, 64d4afb5ceSopenharmony_ci /* .origin_protocol */ LWSMPRO_FILE, /* files in a dir */ 65d4afb5ceSopenharmony_ci /* .mountpoint_len */ 1, /* char count */ 66d4afb5ceSopenharmony_ci /* .basic_auth_login_file */ NULL, 67d4afb5ceSopenharmony_ci}; 68d4afb5ceSopenharmony_ci 69d4afb5ceSopenharmony_ci/* 70d4afb5ceSopenharmony_ci * This demonstrates how to pass a pointer into a specific protocol handler 71d4afb5ceSopenharmony_ci * running on a specific vhost. In this case, it's our default vhost and 72d4afb5ceSopenharmony_ci * we pass the pvo named "config" with the value a const char * "myconfig". 73d4afb5ceSopenharmony_ci * 74d4afb5ceSopenharmony_ci * This is the preferred way to pass configuration into a specific vhost + 75d4afb5ceSopenharmony_ci * protocol instance. 76d4afb5ceSopenharmony_ci */ 77d4afb5ceSopenharmony_ci 78d4afb5ceSopenharmony_cistatic const struct lws_protocol_vhost_options pvo_ops = { 79d4afb5ceSopenharmony_ci NULL, 80d4afb5ceSopenharmony_ci NULL, 81d4afb5ceSopenharmony_ci "config", /* pvo name */ 82d4afb5ceSopenharmony_ci (void *)"myconfig" /* pvo value */ 83d4afb5ceSopenharmony_ci}; 84d4afb5ceSopenharmony_ci 85d4afb5ceSopenharmony_cistatic const struct lws_protocol_vhost_options pvo = { 86d4afb5ceSopenharmony_ci NULL, /* "next" pvo linked-list */ 87d4afb5ceSopenharmony_ci &pvo_ops, /* "child" pvo linked-list */ 88d4afb5ceSopenharmony_ci "lws-minimal", /* protocol name we belong to on this vhost */ 89d4afb5ceSopenharmony_ci "" /* ignored */ 90d4afb5ceSopenharmony_ci}; 91d4afb5ceSopenharmony_ci 92d4afb5ceSopenharmony_civoid *thread_service(void *threadid) 93d4afb5ceSopenharmony_ci{ 94d4afb5ceSopenharmony_ci /* 95d4afb5ceSopenharmony_ci * This is a foreign thread context for each event loop... lws doesn't 96d4afb5ceSopenharmony_ci * know about it, except that it's getting called into from the event 97d4afb5ceSopenharmony_ci * lib bound to each of these. 98d4afb5ceSopenharmony_ci * 99d4afb5ceSopenharmony_ci * When closing, at the point we have detached everything related to 100d4afb5ceSopenharmony_ci * lws from the loop and destroyed the context we can as the "foreign 101d4afb5ceSopenharmony_ci * app" take care of stopping the foreign loop and cloing this thread. 102d4afb5ceSopenharmony_ci * 103d4afb5ceSopenharmony_ci * The call to lws_service_tsi just starts the related event loop 104d4afb5ceSopenharmony_ci */ 105d4afb5ceSopenharmony_ci while (lws_service_tsi(context, 0, 106d4afb5ceSopenharmony_ci (int)(lws_intptr_t)threadid) >= 0 && 107d4afb5ceSopenharmony_ci !interrupted) 108d4afb5ceSopenharmony_ci lwsl_notice("%s\n", __func__); 109d4afb5ceSopenharmony_ci 110d4afb5ceSopenharmony_ci lwsl_info("%s: thr %d: exiting\n", __func__, (int)(lws_intptr_t)threadid); 111d4afb5ceSopenharmony_ci 112d4afb5ceSopenharmony_ci pthread_exit(NULL); 113d4afb5ceSopenharmony_ci 114d4afb5ceSopenharmony_ci return NULL; 115d4afb5ceSopenharmony_ci} 116d4afb5ceSopenharmony_ci 117d4afb5ceSopenharmony_cistatic void 118d4afb5ceSopenharmony_cisignal_cb(uv_signal_t *watcher, int signum) 119d4afb5ceSopenharmony_ci{ 120d4afb5ceSopenharmony_ci int n; 121d4afb5ceSopenharmony_ci 122d4afb5ceSopenharmony_ci n = (int)(watcher - signal_outer); 123d4afb5ceSopenharmony_ci 124d4afb5ceSopenharmony_ci lwsl_notice("%s: thr %d: signal %d caught\n", __func__, n, 125d4afb5ceSopenharmony_ci watcher->signum); 126d4afb5ceSopenharmony_ci 127d4afb5ceSopenharmony_ci uv_signal_stop(watcher); 128d4afb5ceSopenharmony_ci uv_close((uv_handle_t *)&signal_outer[n], NULL); 129d4afb5ceSopenharmony_ci if (!interrupted) { 130d4afb5ceSopenharmony_ci interrupted = 1; 131d4afb5ceSopenharmony_ci lws_context_destroy(context); 132d4afb5ceSopenharmony_ci } 133d4afb5ceSopenharmony_ci} 134d4afb5ceSopenharmony_ci 135d4afb5ceSopenharmony_ciint main(int argc, const char **argv) 136d4afb5ceSopenharmony_ci{ 137d4afb5ceSopenharmony_ci int n, logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE; 138d4afb5ceSopenharmony_ci pthread_t pthread_service[COUNT_THREADS]; 139d4afb5ceSopenharmony_ci struct lws_context_creation_info info; 140d4afb5ceSopenharmony_ci void *foreign_loops[COUNT_THREADS]; 141d4afb5ceSopenharmony_ci int actual_threads; 142d4afb5ceSopenharmony_ci const char *p; 143d4afb5ceSopenharmony_ci void *retval; 144d4afb5ceSopenharmony_ci 145d4afb5ceSopenharmony_ci if ((p = lws_cmdline_option(argc, argv, "-d"))) 146d4afb5ceSopenharmony_ci logs = atoi(p); 147d4afb5ceSopenharmony_ci 148d4afb5ceSopenharmony_ci lws_set_log_level(logs, NULL); 149d4afb5ceSopenharmony_ci lwsl_user("LWS minimal ws server + threads + smp | visit http://localhost:7681\n"); 150d4afb5ceSopenharmony_ci 151d4afb5ceSopenharmony_ci for (n = 0; n < COUNT_THREADS; n++) { 152d4afb5ceSopenharmony_ci uv_loop_init(&loop[n]); 153d4afb5ceSopenharmony_ci 154d4afb5ceSopenharmony_ci s = &signal_outer[n]; 155d4afb5ceSopenharmony_ci uv_signal_init(&loop[n], s); 156d4afb5ceSopenharmony_ci uv_signal_start(s, signal_cb, SIGINT); 157d4afb5ceSopenharmony_ci 158d4afb5ceSopenharmony_ci foreign_loops[n] = &loop[n]; 159d4afb5ceSopenharmony_ci } 160d4afb5ceSopenharmony_ci 161d4afb5ceSopenharmony_ci memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */ 162d4afb5ceSopenharmony_ci info.port = 7681; 163d4afb5ceSopenharmony_ci info.mounts = &mount; 164d4afb5ceSopenharmony_ci info.pcontext = &context; 165d4afb5ceSopenharmony_ci info.protocols = protocols; 166d4afb5ceSopenharmony_ci info.pvo = &pvo; /* per-vhost options */ 167d4afb5ceSopenharmony_ci info.foreign_loops = foreign_loops; 168d4afb5ceSopenharmony_ci info.count_threads = COUNT_THREADS; 169d4afb5ceSopenharmony_ci info.options = LWS_SERVER_OPTION_LIBUV | 170d4afb5ceSopenharmony_ci LWS_SERVER_OPTION_HTTP_HEADERS_SECURITY_BEST_PRACTICES_ENFORCE; 171d4afb5ceSopenharmony_ci 172d4afb5ceSopenharmony_ci context = lws_create_context(&info); 173d4afb5ceSopenharmony_ci if (!context) { 174d4afb5ceSopenharmony_ci lwsl_err("lws init failed\n"); 175d4afb5ceSopenharmony_ci return 1; 176d4afb5ceSopenharmony_ci } 177d4afb5ceSopenharmony_ci 178d4afb5ceSopenharmony_ci actual_threads = lws_get_count_threads(context); 179d4afb5ceSopenharmony_ci lwsl_notice(" Service threads: %d\n", actual_threads); 180d4afb5ceSopenharmony_ci 181d4afb5ceSopenharmony_ci /* start all the service threads */ 182d4afb5ceSopenharmony_ci 183d4afb5ceSopenharmony_ci for (n = 0; n < actual_threads; n++) 184d4afb5ceSopenharmony_ci if (pthread_create(&pthread_service[n], NULL, thread_service, 185d4afb5ceSopenharmony_ci (void *)(lws_intptr_t)n)) 186d4afb5ceSopenharmony_ci lwsl_err("Failed to start service thread\n"); 187d4afb5ceSopenharmony_ci 188d4afb5ceSopenharmony_ci /* wait for all the service threads to exit */ 189d4afb5ceSopenharmony_ci 190d4afb5ceSopenharmony_ci while ((--n) >= 0) 191d4afb5ceSopenharmony_ci pthread_join(pthread_service[n], &retval); 192d4afb5ceSopenharmony_ci 193d4afb5ceSopenharmony_ci lws_context_destroy(context); 194d4afb5ceSopenharmony_ci 195d4afb5ceSopenharmony_ci for (n = 0; n < COUNT_THREADS; n++) { 196d4afb5ceSopenharmony_ci int m; 197d4afb5ceSopenharmony_ci 198d4afb5ceSopenharmony_ci m = uv_loop_close(&loop[n]); 199d4afb5ceSopenharmony_ci if (m) 200d4afb5ceSopenharmony_ci lwsl_notice("%s: uv_close_loop %d: %d\n", __func__, n, m); 201d4afb5ceSopenharmony_ci } 202d4afb5ceSopenharmony_ci 203d4afb5ceSopenharmony_ci return 0; 204d4afb5ceSopenharmony_ci} 205