test-server-pthreads.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. /*
  2. * libwebsockets-test-server - libwebsockets test implementation
  3. *
  4. * Copyright (C) 2010-2016 Andy Green <andy@warmcat.com>
  5. *
  6. * This file is made available under the Creative Commons CC0 1.0
  7. * Universal Public Domain Dedication.
  8. *
  9. * The person who associated a work with this deed has dedicated
  10. * the work to the public domain by waiving all of his or her rights
  11. * to the work worldwide under copyright law, including all related
  12. * and neighboring rights, to the extent allowed by law. You can copy,
  13. * modify, distribute and perform the work, even for commercial purposes,
  14. * all without asking permission.
  15. *
  16. * The test apps are intended to be adapted for use in your code, which
  17. * may be proprietary. So unlike the library itself, they are licensed
  18. * Public Domain.
  19. */
  20. #include "test-server.h"
  21. #include <pthread.h>
  22. int close_testing;
  23. int max_poll_elements;
  24. int debug_level = 7;
  25. #ifdef EXTERNAL_POLL
  26. struct lws_pollfd *pollfds;
  27. int *fd_lookup;
  28. int count_pollfds;
  29. #endif
  30. volatile int force_exit = 0;
  31. struct lws_context *context;
  32. #if defined(LWS_USE_POLARSSL)
  33. #else
  34. #if defined(LWS_USE_MBEDTLS)
  35. #else
  36. #if defined(LWS_OPENSSL_SUPPORT) && defined(LWS_HAVE_SSL_CTX_set1_param)
  37. char crl_path[1024] = "";
  38. #endif
  39. #endif
  40. #endif
  41. /*
  42. * This mutex lock protects code that changes or relies on wsi list outside of
  43. * the service thread. The service thread will acquire it when changing the
  44. * wsi list and other threads should acquire it while dereferencing wsis or
  45. * calling apis like lws_callback_on_writable_all_protocol() which
  46. * use the wsi list and wsis from a different thread context.
  47. */
  48. pthread_mutex_t lock_established_conns;
  49. /* http server gets files from this path */
  50. #define LOCAL_RESOURCE_PATH INSTALL_DATADIR"/libwebsockets-test-server"
  51. char *resource_path = LOCAL_RESOURCE_PATH;
  52. /*
  53. * multithreaded version - protect wsi lifecycle changes in the library
  54. * these are called from protocol 0 callbacks
  55. */
  56. void test_server_lock(int care)
  57. {
  58. if (care)
  59. pthread_mutex_lock(&lock_established_conns);
  60. }
  61. void test_server_unlock(int care)
  62. {
  63. if (care)
  64. pthread_mutex_unlock(&lock_established_conns);
  65. }
  66. /*
  67. * This demo server shows how to use libwebsockets for one or more
  68. * websocket protocols in the same server
  69. *
  70. * It defines the following websocket protocols:
  71. *
  72. * dumb-increment-protocol: once the socket is opened, an incrementing
  73. * ascii string is sent down it every 50ms.
  74. * If you send "reset\n" on the websocket, then
  75. * the incrementing number is reset to 0.
  76. *
  77. * lws-mirror-protocol: copies any received packet to every connection also
  78. * using this protocol, including the sender
  79. */
  80. enum demo_protocols {
  81. /* always first */
  82. PROTOCOL_HTTP = 0,
  83. PROTOCOL_DUMB_INCREMENT,
  84. PROTOCOL_LWS_MIRROR,
  85. /* always last */
  86. DEMO_PROTOCOL_COUNT
  87. };
  88. /* list of supported protocols and callbacks */
  89. static struct lws_protocols protocols[] = {
  90. /* first protocol must always be HTTP handler */
  91. {
  92. "http-only", /* name */
  93. callback_http, /* callback */
  94. sizeof (struct per_session_data__http), /* per_session_data_size */
  95. 0, /* max frame size / rx buffer */
  96. },
  97. {
  98. "dumb-increment-protocol",
  99. callback_dumb_increment,
  100. sizeof(struct per_session_data__dumb_increment),
  101. 10,
  102. },
  103. {
  104. "lws-mirror-protocol",
  105. callback_lws_mirror,
  106. sizeof(struct per_session_data__lws_mirror),
  107. 128,
  108. },
  109. { NULL, NULL, 0, 0 } /* terminator */
  110. };
  111. void *thread_dumb_increment(void *threadid)
  112. {
  113. while (!force_exit) {
  114. /*
  115. * this lock means wsi in the active list cannot
  116. * disappear underneath us, because the code to add and remove
  117. * them is protected by the same lock
  118. */
  119. pthread_mutex_lock(&lock_established_conns);
  120. lws_callback_on_writable_all_protocol(context,
  121. &protocols[PROTOCOL_DUMB_INCREMENT]);
  122. pthread_mutex_unlock(&lock_established_conns);
  123. usleep(100000);
  124. }
  125. pthread_exit(NULL);
  126. }
  127. void *thread_service(void *threadid)
  128. {
  129. while (lws_service_tsi(context, 50, (int)(long)threadid) >= 0 && !force_exit)
  130. ;
  131. pthread_exit(NULL);
  132. }
  133. void sighandler(int sig)
  134. {
  135. force_exit = 1;
  136. lws_cancel_service(context);
  137. }
  138. static const struct lws_extension exts[] = {
  139. {
  140. "permessage-deflate",
  141. lws_extension_callback_pm_deflate,
  142. "permessage-deflate; client_no_context_takeover; client_max_window_bits"
  143. },
  144. {
  145. "deflate-frame",
  146. lws_extension_callback_pm_deflate,
  147. "deflate_frame"
  148. },
  149. { NULL, NULL, NULL /* terminator */ }
  150. };
  151. static struct option options[] = {
  152. { "help", no_argument, NULL, 'h' },
  153. { "debug", required_argument, NULL, 'd' },
  154. { "port", required_argument, NULL, 'p' },
  155. { "ssl", no_argument, NULL, 's' },
  156. { "allow-non-ssl", no_argument, NULL, 'a' },
  157. { "interface", required_argument, NULL, 'i' },
  158. { "closetest", no_argument, NULL, 'c' },
  159. { "libev", no_argument, NULL, 'e' },
  160. { "threads", required_argument, NULL, 'j' },
  161. #ifndef LWS_NO_DAEMONIZE
  162. { "daemonize", no_argument, NULL, 'D' },
  163. #endif
  164. { "resource_path", required_argument, NULL, 'r' },
  165. { NULL, 0, 0, 0 }
  166. };
  167. int main(int argc, char **argv)
  168. {
  169. struct lws_context_creation_info info;
  170. char interface_name[128] = "";
  171. const char *iface = NULL;
  172. pthread_t pthread_dumb, pthread_service[32];
  173. char cert_path[1024];
  174. char key_path[1024];
  175. int threads = 1;
  176. int use_ssl = 0;
  177. void *retval;
  178. int opts = 0;
  179. int n = 0;
  180. #ifndef _WIN32
  181. /* LOG_PERROR is not POSIX standard, and may not be portable */
  182. #ifdef __sun
  183. int syslog_options = LOG_PID;
  184. #else
  185. int syslog_options = LOG_PID | LOG_PERROR;
  186. #endif
  187. #endif
  188. #ifndef LWS_NO_DAEMONIZE
  189. int daemonize = 0;
  190. #endif
  191. /*
  192. * take care to zero down the info struct, he contains random garbaage
  193. * from the stack otherwise
  194. */
  195. memset(&info, 0, sizeof info);
  196. info.port = 7681;
  197. pthread_mutex_init(&lock_established_conns, NULL);
  198. while (n >= 0) {
  199. n = getopt_long(argc, argv, "eci:hsap:d:Dr:j:", options, NULL);
  200. if (n < 0)
  201. continue;
  202. switch (n) {
  203. case 'j':
  204. threads = atoi(optarg);
  205. if (threads > ARRAY_SIZE(pthread_service)) {
  206. lwsl_err("Max threads %d\n",
  207. ARRAY_SIZE(pthread_service));
  208. return 1;
  209. }
  210. break;
  211. case 'e':
  212. opts |= LWS_SERVER_OPTION_LIBEV;
  213. break;
  214. #ifndef LWS_NO_DAEMONIZE
  215. case 'D':
  216. daemonize = 1;
  217. #if !defined(_WIN32) && !defined(__sun)
  218. syslog_options &= ~LOG_PERROR;
  219. #endif
  220. break;
  221. #endif
  222. case 'd':
  223. debug_level = atoi(optarg);
  224. break;
  225. case 's':
  226. use_ssl = 1;
  227. break;
  228. case 'a':
  229. opts |= LWS_SERVER_OPTION_ALLOW_NON_SSL_ON_SSL_PORT;
  230. break;
  231. case 'p':
  232. info.port = atoi(optarg);
  233. break;
  234. case 'i':
  235. strncpy(interface_name, optarg, sizeof interface_name);
  236. interface_name[(sizeof interface_name) - 1] = '\0';
  237. iface = interface_name;
  238. break;
  239. case 'c':
  240. close_testing = 1;
  241. fprintf(stderr, " Close testing mode -- closes on "
  242. "client after 50 dumb increments"
  243. "and suppresses lws_mirror spam\n");
  244. break;
  245. case 'r':
  246. resource_path = optarg;
  247. printf("Setting resource path to \"%s\"\n", resource_path);
  248. break;
  249. case 'h':
  250. fprintf(stderr, "Usage: test-server "
  251. "[--port=<p>] [--ssl] "
  252. "[-d <log bitfield>] "
  253. "[--resource_path <path>]\n");
  254. exit(1);
  255. }
  256. }
  257. #if !defined(LWS_NO_DAEMONIZE) && !defined(WIN32)
  258. /*
  259. * normally lock path would be /var/lock/lwsts or similar, to
  260. * simplify getting started without having to take care about
  261. * permissions or running as root, set to /tmp/.lwsts-lock
  262. */
  263. if (daemonize && lws_daemonize("/tmp/.lwsts-lock")) {
  264. fprintf(stderr, "Failed to daemonize\n");
  265. return 1;
  266. }
  267. #endif
  268. signal(SIGINT, sighandler);
  269. #ifndef _WIN32
  270. /* we will only try to log things according to our debug_level */
  271. setlogmask(LOG_UPTO (LOG_DEBUG));
  272. openlog("lwsts", syslog_options, LOG_DAEMON);
  273. #endif
  274. /* tell the library what debug level to emit and to send it to syslog */
  275. lws_set_log_level(debug_level, lwsl_emit_syslog);
  276. lwsl_notice("libwebsockets test server pthreads - license LGPL2.1+SLE\n");
  277. lwsl_notice("(C) Copyright 2010-2016 Andy Green <andy@warmcat.com>\n");
  278. printf("Using resource path \"%s\"\n", resource_path);
  279. #ifdef EXTERNAL_POLL
  280. max_poll_elements = getdtablesize();
  281. pollfds = malloc(max_poll_elements * sizeof (struct lws_pollfd));
  282. fd_lookup = malloc(max_poll_elements * sizeof (int));
  283. if (pollfds == NULL || fd_lookup == NULL) {
  284. lwsl_err("Out of memory pollfds=%d\n", max_poll_elements);
  285. return -1;
  286. }
  287. #endif
  288. info.iface = iface;
  289. info.protocols = protocols;
  290. info.extensions = exts;
  291. info.ssl_cert_filepath = NULL;
  292. info.ssl_private_key_filepath = NULL;
  293. if (use_ssl) {
  294. if (strlen(resource_path) > sizeof(cert_path) - 32) {
  295. lwsl_err("resource path too long\n");
  296. return -1;
  297. }
  298. sprintf(cert_path, "%s/libwebsockets-test-server.pem",
  299. resource_path);
  300. if (strlen(resource_path) > sizeof(key_path) - 32) {
  301. lwsl_err("resource path too long\n");
  302. return -1;
  303. }
  304. sprintf(key_path, "%s/libwebsockets-test-server.key.pem",
  305. resource_path);
  306. info.ssl_cert_filepath = cert_path;
  307. info.ssl_private_key_filepath = key_path;
  308. }
  309. info.gid = -1;
  310. info.uid = -1;
  311. info.options = opts;
  312. info.count_threads = threads;
  313. info.extensions = exts;
  314. info.max_http_header_pool = 4;
  315. context = lws_create_context(&info);
  316. if (context == NULL) {
  317. lwsl_err("libwebsocket init failed\n");
  318. return -1;
  319. }
  320. /* start the dumb increment thread */
  321. n = pthread_create(&pthread_dumb, NULL, thread_dumb_increment, 0);
  322. if (n) {
  323. lwsl_err("Unable to create dumb thread\n");
  324. goto done;
  325. }
  326. /*
  327. * notice the actual number of threads may be capped by the library,
  328. * so use lws_get_count_threads() to get the actual amount of threads
  329. * initialized.
  330. */
  331. for (n = 0; n < lws_get_count_threads(context); n++)
  332. if (pthread_create(&pthread_service[n], NULL, thread_service,
  333. (void *)(long)n))
  334. lwsl_err("Failed to start service thread\n");
  335. /* wait for all the service threads to exit */
  336. while ((--n) >= 0)
  337. pthread_join(pthread_service[n], &retval);
  338. /* wait for pthread_dumb to exit */
  339. pthread_join(pthread_dumb, &retval);
  340. done:
  341. lws_context_destroy(context);
  342. pthread_mutex_destroy(&lock_established_conns);
  343. lwsl_notice("libwebsockets-test-server exited cleanly\n");
  344. #ifndef _WIN32
  345. closelog();
  346. #endif
  347. return 0;
  348. }