context.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945
  1. /*
  2. * libwebsockets - small server side websockets and web server implementation
  3. *
  4. * Copyright (C) 2010-2015 Andy Green <andy@warmcat.com>
  5. *
  6. * This library is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation:
  9. * version 2.1 of the License.
  10. *
  11. * This library is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public
  17. * License along with this library; if not, write to the Free Software
  18. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
  19. * MA 02110-1301 USA
  20. */
  21. #include "private-libwebsockets.h"
  22. #ifndef LWS_BUILD_HASH
  23. #define LWS_BUILD_HASH "unknown-build-hash"
  24. #endif
  25. static const char *library_version = LWS_LIBRARY_VERSION " " LWS_BUILD_HASH;
  26. /**
  27. * lws_get_library_version: get version and git hash library built from
  28. *
  29. * returns a const char * to a string like "1.1 178d78c"
  30. * representing the library version followed by the git head hash it
  31. * was built from
  32. */
  33. LWS_VISIBLE const char *
  34. lws_get_library_version(void)
  35. {
  36. return library_version;
  37. }
  38. static const char * const mount_protocols[] = {
  39. "http://",
  40. "https://",
  41. "file://",
  42. "cgi://",
  43. ">http://",
  44. ">https://",
  45. "callback://"
  46. };
  47. LWS_VISIBLE void *
  48. lws_protocol_vh_priv_zalloc(struct lws_vhost *vhost, const struct lws_protocols *prot,
  49. int size)
  50. {
  51. int n = 0;
  52. /* allocate the vh priv array only on demand */
  53. if (!vhost->protocol_vh_privs) {
  54. vhost->protocol_vh_privs = (void **)lws_zalloc(
  55. vhost->count_protocols * sizeof(void *));
  56. if (!vhost->protocol_vh_privs)
  57. return NULL;
  58. }
  59. while (n < vhost->count_protocols && &vhost->protocols[n] != prot)
  60. n++;
  61. if (n == vhost->count_protocols) {
  62. n = 0;
  63. while (n < vhost->count_protocols &&
  64. strcmp(vhost->protocols[n].name, prot->name))
  65. n++;
  66. if (n == vhost->count_protocols)
  67. return NULL;
  68. }
  69. vhost->protocol_vh_privs[n] = lws_zalloc(size);
  70. return vhost->protocol_vh_privs[n];
  71. }
  72. LWS_VISIBLE void *
  73. lws_protocol_vh_priv_get(struct lws_vhost *vhost, const struct lws_protocols *prot)
  74. {
  75. int n = 0;
  76. if (!vhost->protocol_vh_privs)
  77. return NULL;
  78. while (n < vhost->count_protocols && &vhost->protocols[n] != prot)
  79. n++;
  80. if (n == vhost->count_protocols) {
  81. n = 0;
  82. while (n < vhost->count_protocols &&
  83. strcmp(vhost->protocols[n].name, prot->name))
  84. n++;
  85. if (n == vhost->count_protocols) {
  86. lwsl_err("%s: unknown protocol %p\n", __func__, prot);
  87. return NULL;
  88. }
  89. }
  90. return vhost->protocol_vh_privs[n];
  91. }
  92. static const struct lws_protocol_vhost_options *
  93. lws_vhost_protocol_options(struct lws_vhost *vh, const char *name)
  94. {
  95. const struct lws_protocol_vhost_options *pvo = vh->pvo;
  96. while (pvo) {
  97. // lwsl_notice("%s: '%s' '%s'\n", __func__, pvo->name, name);
  98. if (!strcmp(pvo->name, name))
  99. return pvo;
  100. pvo = pvo->next;
  101. }
  102. return NULL;
  103. }
  104. int
  105. lws_protocol_init(struct lws_context *context)
  106. {
  107. struct lws_vhost *vh = context->vhost_list;
  108. const struct lws_protocol_vhost_options *pvo, *pvo1;
  109. struct lws wsi;
  110. int n;
  111. memset(&wsi, 0, sizeof(wsi));
  112. wsi.context = context;
  113. lwsl_notice("%s\n", __func__);
  114. while (vh) {
  115. wsi.vhost = vh;
  116. /* initialize supported protocols on this vhost */
  117. for (n = 0; n < vh->count_protocols; n++) {
  118. wsi.protocol = &vh->protocols[n];
  119. pvo = lws_vhost_protocol_options(vh,
  120. vh->protocols[n].name);
  121. if (pvo) {
  122. /*
  123. * linked list of options specific to
  124. * vh + protocol
  125. */
  126. pvo1 = pvo;
  127. pvo = pvo1->options;
  128. while (pvo) {
  129. lwsl_notice(" vh %s prot %s opt %s\n",
  130. vh->name,
  131. vh->protocols[n].name,
  132. pvo->name);
  133. if (!strcmp(pvo->name, "default")) {
  134. lwsl_notice("Setting default "
  135. "protocol for vh %s to %s\n",
  136. vh->name,
  137. vh->protocols[n].name);
  138. vh->default_protocol_index = n;
  139. }
  140. pvo = pvo->next;
  141. }
  142. pvo = pvo1->options;
  143. }
  144. /*
  145. * inform all the protocols that they are doing their one-time
  146. * initialization if they want to.
  147. *
  148. * NOTE the wsi is all zeros except for the context, vh and
  149. * protocol ptrs so lws_get_context(wsi) etc can work
  150. */
  151. if (vh->protocols[n].callback(&wsi,
  152. LWS_CALLBACK_PROTOCOL_INIT, NULL,
  153. (void *)pvo, 0))
  154. return 1;
  155. }
  156. vh = vh->vhost_next;
  157. }
  158. context->protocol_init_done = 1;
  159. lws_finalize_startup(context);
  160. return 0;
  161. }
  162. LWS_VISIBLE int
  163. lws_callback_http_dummy(struct lws *wsi, enum lws_callback_reasons reason,
  164. void *user, void *in, size_t len)
  165. {
  166. #ifdef LWS_WITH_CGI
  167. struct lws_cgi_args *args;
  168. char buf[128];
  169. int n;
  170. #endif
  171. switch (reason) {
  172. case LWS_CALLBACK_HTTP:
  173. #ifndef LWS_NO_SERVER
  174. if (lws_return_http_status(wsi, HTTP_STATUS_NOT_FOUND, NULL))
  175. return -1;
  176. if (lws_http_transaction_completed(wsi))
  177. #endif
  178. return -1;
  179. break;
  180. case LWS_CALLBACK_HTTP_WRITEABLE:
  181. #ifdef LWS_WITH_CGI
  182. if (wsi->reason_bf & 1) {
  183. if (lws_cgi_write_split_stdout_headers(wsi) < 0)
  184. return -1;
  185. wsi->reason_bf &= ~1;
  186. break;
  187. }
  188. #endif
  189. break;
  190. #ifdef LWS_WITH_CGI
  191. /* CGI IO events (POLLIN/OUT) appear here, our default policy is:
  192. *
  193. * - POST data goes on subprocess stdin
  194. * - subprocess stdout goes on http via writeable callback
  195. * - subprocess stderr goes to the logs
  196. */
  197. case LWS_CALLBACK_CGI:
  198. args = (struct lws_cgi_args *)in;
  199. switch (args->ch) { /* which of stdin/out/err ? */
  200. case LWS_STDIN:
  201. /* TBD stdin rx flow control */
  202. break;
  203. case LWS_STDOUT:
  204. wsi->reason_bf |= 1;
  205. /* when writing to MASTER would not block */
  206. lws_callback_on_writable(wsi);
  207. break;
  208. case LWS_STDERR:
  209. n = read(lws_get_socket_fd(args->stdwsi[LWS_STDERR]),
  210. buf, sizeof(buf) - 2);
  211. if (n > 0) {
  212. if (buf[n - 1] != '\n')
  213. buf[n++] = '\n';
  214. buf[n] = '\0';
  215. lwsl_notice("CGI-stderr: %s\n", buf);
  216. }
  217. break;
  218. }
  219. break;
  220. case LWS_CALLBACK_CGI_TERMINATED:
  221. return -1;
  222. case LWS_CALLBACK_CGI_STDIN_DATA: /* POST body for stdin */
  223. args = (struct lws_cgi_args *)in;
  224. args->data[args->len] = '\0';
  225. n = write(lws_get_socket_fd(args->stdwsi[LWS_STDIN]),
  226. args->data, args->len);
  227. if (n < args->len)
  228. lwsl_notice("LWS_CALLBACK_CGI_STDIN_DATA: "
  229. "sent %d only %d went", n, args->len);
  230. return n;
  231. #endif
  232. default:
  233. break;
  234. }
  235. return 0;
  236. }
  237. /* list of supported protocols and callbacks */
  238. static const struct lws_protocols protocols_dummy[] = {
  239. /* first protocol must always be HTTP handler */
  240. {
  241. "http-only", /* name */
  242. lws_callback_http_dummy, /* callback */
  243. 0, /* per_session_data_size */
  244. 0, /* max frame size / rx buffer */
  245. },
  246. /*
  247. * the other protocols are provided by lws plugins
  248. */
  249. { NULL, NULL, 0, 0 } /* terminator */
  250. };
  251. LWS_VISIBLE struct lws_vhost *
  252. lws_create_vhost(struct lws_context *context,
  253. struct lws_context_creation_info *info)
  254. {
  255. struct lws_vhost *vh = lws_zalloc(sizeof(*vh)),
  256. **vh1 = &context->vhost_list;
  257. const struct lws_http_mount *mounts;
  258. const struct lws_protocol_vhost_options *pvo;
  259. #ifdef LWS_WITH_PLUGINS
  260. struct lws_plugin *plugin = context->plugin_list;
  261. struct lws_protocols *lwsp;
  262. int m, f = !info->pvo;
  263. #endif
  264. #ifdef LWS_HAVE_GETENV
  265. char *p;
  266. #endif
  267. int n;
  268. if (!vh)
  269. return NULL;
  270. if (!info->protocols)
  271. info->protocols = &protocols_dummy[0];
  272. vh->context = context;
  273. if (!info->vhost_name)
  274. vh->name = "default";
  275. else
  276. vh->name = info->vhost_name;
  277. vh->iface = info->iface;
  278. for (vh->count_protocols = 0;
  279. info->protocols[vh->count_protocols].callback;
  280. vh->count_protocols++)
  281. ;
  282. vh->options = info->options;
  283. vh->pvo = info->pvo;
  284. vh->headers = info->headers;
  285. if (info->keepalive_timeout)
  286. vh->keepalive_timeout = info->keepalive_timeout;
  287. else
  288. vh->keepalive_timeout = 5;
  289. #ifdef LWS_WITH_PLUGINS
  290. if (plugin) {
  291. /*
  292. * give the vhost a unified list of protocols including the
  293. * ones that came from plugins
  294. */
  295. lwsp = lws_zalloc(sizeof(struct lws_protocols) *
  296. (vh->count_protocols +
  297. context->plugin_protocol_count + 1));
  298. if (!lwsp)
  299. return NULL;
  300. m = vh->count_protocols;
  301. memcpy(lwsp, info->protocols,
  302. sizeof(struct lws_protocols) * m);
  303. /* for compatibility, all protocols enabled on vhost if only
  304. * the default vhost exists. Otherwise only vhosts who ask
  305. * for a protocol get it enabled.
  306. */
  307. if (info->options & LWS_SERVER_OPTION_EXPLICIT_VHOSTS)
  308. f = 0;
  309. while (plugin) {
  310. for (n = 0; n < plugin->caps.count_protocols; n++) {
  311. /*
  312. * for compatibility's sake, no pvo implies
  313. * allow all protocols
  314. */
  315. if (f || lws_vhost_protocol_options(vh,
  316. plugin->caps.protocols[n].name)) {
  317. memcpy(&lwsp[m],
  318. &plugin->caps.protocols[n],
  319. sizeof(struct lws_protocols));
  320. m++;
  321. vh->count_protocols++;
  322. }
  323. }
  324. plugin = plugin->list;
  325. }
  326. vh->protocols = lwsp;
  327. } else
  328. #endif
  329. vh->protocols = info->protocols;
  330. vh->same_vh_protocol_list = (struct lws **)
  331. lws_zalloc(sizeof(struct lws *) * vh->count_protocols);
  332. vh->mount_list = info->mounts;
  333. #ifdef LWS_USE_UNIX_SOCK
  334. if (LWS_UNIX_SOCK_ENABLED(context)) {
  335. lwsl_notice("Creating Vhost '%s' path \"%s\", %d protocols\n",
  336. vh->name, info->iface, vh->count_protocols);
  337. } else
  338. #endif
  339. lwsl_notice("Creating Vhost '%s' port %d, %d protocols, IPv6 %s\n",
  340. vh->name, info->port, vh->count_protocols, LWS_IPV6_ENABLED(vh) ? "on" : "off");
  341. mounts = info->mounts;
  342. while (mounts) {
  343. lwsl_notice(" mounting %s%s to %s\n",
  344. mount_protocols[mounts->origin_protocol],
  345. mounts->origin, mounts->mountpoint);
  346. /* convert interpreter protocol names to pointers */
  347. pvo = mounts->interpret;
  348. while (pvo) {
  349. for (n = 0; n < vh->count_protocols; n++)
  350. if (!strcmp(pvo->value, vh->protocols[n].name)) {
  351. ((struct lws_protocol_vhost_options *)pvo)->value =
  352. (const char *)(long)n;
  353. break;
  354. }
  355. if (n == vh->count_protocols)
  356. lwsl_err("ignoring unknown interpret protocol %s\n", pvo->value);
  357. pvo = pvo->next;
  358. }
  359. mounts = mounts->mount_next;
  360. }
  361. #ifndef LWS_NO_EXTENSIONS
  362. #ifdef LWS_WITH_PLUGINS
  363. if (context->plugin_extension_count) {
  364. m = 0;
  365. while (info->extensions && info->extensions[m].callback)
  366. m++;
  367. /*
  368. * give the vhost a unified list of extensions including the
  369. * ones that came from plugins
  370. */
  371. vh->extensions = lws_zalloc(sizeof(struct lws_extension) *
  372. (m +
  373. context->plugin_extension_count + 1));
  374. if (!vh->extensions)
  375. return NULL;
  376. memcpy((struct lws_extension *)vh->extensions, info->extensions,
  377. sizeof(struct lws_extension) * m);
  378. plugin = context->plugin_list;
  379. while (plugin) {
  380. memcpy((struct lws_extension *)&vh->extensions[m],
  381. plugin->caps.extensions,
  382. sizeof(struct lws_extension) *
  383. plugin->caps.count_extensions);
  384. m += plugin->caps.count_extensions;
  385. plugin = plugin->list;
  386. }
  387. } else
  388. #endif
  389. vh->extensions = info->extensions;
  390. #endif
  391. vh->listen_port = info->port;
  392. #if !defined(LWS_WITH_ESP8266)
  393. vh->http_proxy_port = 0;
  394. vh->http_proxy_address[0] = '\0';
  395. /* either use proxy from info, or try get it from env var */
  396. if (info->http_proxy_address) {
  397. /* override for backwards compatibility */
  398. if (info->http_proxy_port)
  399. vh->http_proxy_port = info->http_proxy_port;
  400. lws_set_proxy(vh, info->http_proxy_address);
  401. } else {
  402. #ifdef LWS_HAVE_GETENV
  403. p = getenv("http_proxy");
  404. if (p)
  405. lws_set_proxy(vh, p);
  406. #endif
  407. }
  408. #endif
  409. vh->ka_time = info->ka_time;
  410. vh->ka_interval = info->ka_interval;
  411. vh->ka_probes = info->ka_probes;
  412. if (vh->options & LWS_SERVER_OPTION_STS)
  413. lwsl_notice(" STS enabled\n");
  414. #ifdef LWS_WITH_ACCESS_LOG
  415. if (info->log_filepath) {
  416. vh->log_fd = open(info->log_filepath, O_CREAT | O_APPEND | O_RDWR, 0600);
  417. if (vh->log_fd == (int)LWS_INVALID_FILE) {
  418. lwsl_err("unable to open log filepath %s\n",
  419. info->log_filepath);
  420. goto bail;
  421. }
  422. #ifndef WIN32
  423. if (context->uid != -1)
  424. if (chown(info->log_filepath, context->uid,
  425. context->gid) == -1)
  426. lwsl_err("unable to chown log file %s\n",
  427. info->log_filepath);
  428. #endif
  429. } else
  430. vh->log_fd = (int)LWS_INVALID_FILE;
  431. #endif
  432. if (lws_context_init_server_ssl(info, vh))
  433. goto bail;
  434. if (lws_context_init_client_ssl(info, vh))
  435. goto bail;
  436. if (lws_context_init_server(info, vh))
  437. goto bail;
  438. while (1) {
  439. if (!(*vh1)) {
  440. *vh1 = vh;
  441. break;
  442. }
  443. vh1 = &(*vh1)->vhost_next;
  444. };
  445. return vh;
  446. bail:
  447. lws_free(vh);
  448. return NULL;
  449. }
  450. LWS_VISIBLE int
  451. lws_init_vhost_client_ssl(const struct lws_context_creation_info *info,
  452. struct lws_vhost *vhost)
  453. {
  454. struct lws_context_creation_info i;
  455. memcpy(&i, info, sizeof(i));
  456. i.port = CONTEXT_PORT_NO_LISTEN;
  457. return lws_context_init_client_ssl(&i, vhost);
  458. }
  459. struct lws wsi;
  460. LWS_VISIBLE struct lws_context *
  461. lws_create_context(struct lws_context_creation_info *info)
  462. {
  463. struct lws_context *context = NULL;
  464. #ifndef LWS_NO_DAEMONIZE
  465. int pid_daemon = get_daemonize_pid();
  466. #endif
  467. int n, m;
  468. #if defined(__ANDROID__)
  469. struct rlimit rt;
  470. #endif
  471. lwsl_notice("Initial logging level %d\n", log_level);
  472. lwsl_notice("Libwebsockets version: %s\n", library_version);
  473. #if LWS_POSIX
  474. #ifdef LWS_USE_IPV6
  475. if (!lws_check_opt(info->options, LWS_SERVER_OPTION_DISABLE_IPV6))
  476. lwsl_notice("IPV6 compiled in and enabled\n");
  477. else
  478. lwsl_notice("IPV6 compiled in but disabled\n");
  479. #else
  480. lwsl_notice("IPV6 not compiled in\n");
  481. #endif
  482. lws_feature_status_libev(info);
  483. lws_feature_status_libuv(info);
  484. #endif
  485. lwsl_info(" LWS_DEF_HEADER_LEN : %u\n", LWS_DEF_HEADER_LEN);
  486. lwsl_info(" LWS_MAX_PROTOCOLS : %u\n", LWS_MAX_PROTOCOLS);
  487. lwsl_info(" LWS_MAX_SMP : %u\n", LWS_MAX_SMP);
  488. lwsl_info(" SPEC_LATEST_SUPPORTED : %u\n", SPEC_LATEST_SUPPORTED);
  489. lwsl_info(" sizeof (*info) : %u\n", sizeof(*info));
  490. #if LWS_POSIX
  491. lwsl_info(" SYSTEM_RANDOM_FILEPATH: '%s'\n", SYSTEM_RANDOM_FILEPATH);
  492. #endif
  493. if (lws_plat_context_early_init())
  494. return NULL;
  495. context = lws_zalloc(sizeof(struct lws_context));
  496. if (!context) {
  497. lwsl_err("No memory for websocket context\n");
  498. return NULL;
  499. }
  500. if (info->pt_serv_buf_size)
  501. context->pt_serv_buf_size = info->pt_serv_buf_size;
  502. else
  503. context->pt_serv_buf_size = 4096;
  504. context->time_up = time(NULL);
  505. #ifndef LWS_NO_DAEMONIZE
  506. if (pid_daemon) {
  507. context->started_with_parent = pid_daemon;
  508. lwsl_notice(" Started with daemon pid %d\n", pid_daemon);
  509. }
  510. #endif
  511. #if defined(__ANDROID__)
  512. n = getrlimit ( RLIMIT_NOFILE,&rt);
  513. if (-1 == n) {
  514. lwsl_err("Get RLIMIT_NOFILE failed!\n");
  515. return NULL;
  516. }
  517. context->max_fds = rt.rlim_cur;
  518. #else
  519. context->max_fds = getdtablesize();
  520. #endif
  521. if (info->count_threads)
  522. context->count_threads = info->count_threads;
  523. else
  524. context->count_threads = 1;
  525. if (context->count_threads > LWS_MAX_SMP)
  526. context->count_threads = LWS_MAX_SMP;
  527. context->token_limits = info->token_limits;
  528. context->options = info->options;
  529. if (info->timeout_secs)
  530. context->timeout_secs = info->timeout_secs;
  531. else
  532. context->timeout_secs = AWAITING_TIMEOUT;
  533. context->ws_ping_pong_interval = info->ws_ping_pong_interval;
  534. lwsl_info(" default timeout (secs): %u\n", context->timeout_secs);
  535. if (info->max_http_header_data)
  536. context->max_http_header_data = info->max_http_header_data;
  537. else
  538. if (info->max_http_header_data2)
  539. context->max_http_header_data =
  540. info->max_http_header_data2;
  541. else
  542. context->max_http_header_data = LWS_DEF_HEADER_LEN;
  543. if (info->max_http_header_pool)
  544. context->max_http_header_pool = info->max_http_header_pool;
  545. else
  546. context->max_http_header_pool = LWS_DEF_HEADER_POOL;
  547. /*
  548. * Allocate the per-thread storage for scratchpad buffers,
  549. * and header data pool
  550. */
  551. for (n = 0; n < context->count_threads; n++) {
  552. context->pt[n].serv_buf = lws_zalloc(context->pt_serv_buf_size);
  553. if (!context->pt[n].serv_buf) {
  554. lwsl_err("OOM\n");
  555. return NULL;
  556. }
  557. #ifdef LWS_USE_LIBUV
  558. context->pt[n].context = context;
  559. #endif
  560. context->pt[n].tid = n;
  561. context->pt[n].http_header_data = lws_malloc(context->max_http_header_data *
  562. context->max_http_header_pool);
  563. if (!context->pt[n].http_header_data)
  564. goto bail;
  565. context->pt[n].ah_pool = lws_zalloc(sizeof(struct allocated_headers) *
  566. context->max_http_header_pool);
  567. for (m = 0; m < context->max_http_header_pool; m++)
  568. context->pt[n].ah_pool[m].data =
  569. (char *)context->pt[n].http_header_data +
  570. (m * context->max_http_header_data);
  571. if (!context->pt[n].ah_pool)
  572. goto bail;
  573. lws_pt_mutex_init(&context->pt[n]);
  574. }
  575. if (info->fd_limit_per_thread)
  576. context->fd_limit_per_thread = info->fd_limit_per_thread;
  577. else
  578. context->fd_limit_per_thread = context->max_fds /
  579. context->count_threads;
  580. lwsl_notice(" Threads: %d each %d fds\n", context->count_threads,
  581. context->fd_limit_per_thread);
  582. memset(&wsi, 0, sizeof(wsi));
  583. wsi.context = context;
  584. if (!info->ka_interval && info->ka_time > 0) {
  585. lwsl_err("info->ka_interval can't be 0 if ka_time used\n");
  586. return NULL;
  587. }
  588. #ifdef LWS_USE_LIBEV
  589. /* (Issue #264) In order to *avoid breaking backwards compatibility*, we
  590. * enable libev mediated SIGINT handling with a default handler of
  591. * lws_sigint_cb. The handler can be overridden or disabled
  592. * by invoking lws_sigint_cfg after creating the context, but
  593. * before invoking lws_initloop:
  594. */
  595. context->use_ev_sigint = 1;
  596. context->lws_ev_sigint_cb = &lws_ev_sigint_cb;
  597. #endif /* LWS_USE_LIBEV */
  598. #ifdef LWS_USE_LIBUV
  599. /* (Issue #264) In order to *avoid breaking backwards compatibility*, we
  600. * enable libev mediated SIGINT handling with a default handler of
  601. * lws_sigint_cb. The handler can be overridden or disabled
  602. * by invoking lws_sigint_cfg after creating the context, but
  603. * before invoking lws_initloop:
  604. */
  605. context->use_ev_sigint = 1;
  606. context->lws_uv_sigint_cb = &lws_uv_sigint_cb;
  607. #endif
  608. lwsl_info(" mem: context: %5u bytes (%d ctx + (%d thr x %d))\n",
  609. sizeof(struct lws_context) +
  610. (context->count_threads * context->pt_serv_buf_size),
  611. sizeof(struct lws_context),
  612. context->count_threads,
  613. context->pt_serv_buf_size);
  614. lwsl_info(" mem: http hdr rsvd: %5u bytes (%u thr x (%u + %u) x %u))\n",
  615. (context->max_http_header_data +
  616. sizeof(struct allocated_headers)) *
  617. context->max_http_header_pool * context->count_threads,
  618. context->count_threads,
  619. context->max_http_header_data,
  620. sizeof(struct allocated_headers),
  621. context->max_http_header_pool);
  622. n = sizeof(struct lws_pollfd) * context->count_threads *
  623. context->fd_limit_per_thread;
  624. context->pt[0].fds = lws_zalloc(n);
  625. if (context->pt[0].fds == NULL) {
  626. lwsl_err("OOM allocating %d fds\n", context->max_fds);
  627. goto bail;
  628. }
  629. lwsl_info(" mem: pollfd map: %5u\n", n);
  630. if (info->server_string) {
  631. context->server_string = info->server_string;
  632. context->server_string_len = (short)
  633. strlen(context->server_string);
  634. } else {
  635. context->server_string = "libwebsockets";
  636. context->server_string_len = 13;
  637. }
  638. #if LWS_MAX_SMP > 1
  639. /* each thread serves his own chunk of fds */
  640. for (n = 1; n < (int)info->count_threads; n++)
  641. context->pt[n].fds = context->pt[n - 1].fds +
  642. context->fd_limit_per_thread;
  643. #endif
  644. if (lws_plat_init(context, info))
  645. goto bail;
  646. lws_context_init_ssl_library(info);
  647. context->user_space = info->user;
  648. /*
  649. * if he's not saying he'll make his own vhosts later then act
  650. * compatibly and make a default vhost using the data in the info
  651. */
  652. if (!lws_check_opt(info->options, LWS_SERVER_OPTION_EXPLICIT_VHOSTS))
  653. if (!lws_create_vhost(context, info)) {
  654. lwsl_err("Failed to create default vhost\n");
  655. return NULL;
  656. }
  657. lws_context_init_extensions(info, context);
  658. lwsl_notice(" mem: per-conn: %5u bytes + protocol rx buf\n",
  659. sizeof(struct lws));
  660. strcpy(context->canonical_hostname, "unknown");
  661. lws_server_get_canonical_hostname(context, info);
  662. context->uid = info->uid;
  663. context->gid = info->gid;
  664. /*
  665. * drop any root privs for this process
  666. * to listen on port < 1023 we would have needed root, but now we are
  667. * listening, we don't want the power for anything else
  668. */
  669. if (!lws_check_opt(info->options, LWS_SERVER_OPTION_EXPLICIT_VHOSTS))
  670. lws_plat_drop_app_privileges(info);
  671. /*
  672. * give all extensions a chance to create any per-context
  673. * allocations they need
  674. */
  675. if (info->port != CONTEXT_PORT_NO_LISTEN) {
  676. if (lws_ext_cb_all_exts(context, NULL,
  677. LWS_EXT_CB_SERVER_CONTEXT_CONSTRUCT, NULL, 0) < 0)
  678. goto bail;
  679. } else
  680. if (lws_ext_cb_all_exts(context, NULL,
  681. LWS_EXT_CB_CLIENT_CONTEXT_CONSTRUCT, NULL, 0) < 0)
  682. goto bail;
  683. return context;
  684. bail:
  685. lws_context_destroy(context);
  686. return NULL;
  687. }
  688. LWS_VISIBLE void
  689. lws_context_destroy(struct lws_context *context)
  690. {
  691. const struct lws_protocols *protocol = NULL;
  692. struct lws_context_per_thread *pt;
  693. struct lws_vhost *vh = NULL, *vh1;
  694. struct lws wsi;
  695. int n, m;
  696. lwsl_notice("%s\n", __func__);
  697. if (!context)
  698. return;
  699. m = context->count_threads;
  700. context->being_destroyed = 1;
  701. memset(&wsi, 0, sizeof(wsi));
  702. wsi.context = context;
  703. #ifdef LWS_LATENCY
  704. if (context->worst_latency_info[0])
  705. lwsl_notice("Worst latency: %s\n", context->worst_latency_info);
  706. #endif
  707. while (m--) {
  708. pt = &context->pt[m];
  709. for (n = 0; (unsigned int)n < context->pt[m].fds_count; n++) {
  710. struct lws *wsi = wsi_from_fd(context, pt->fds[n].fd);
  711. if (!wsi)
  712. continue;
  713. lws_close_free_wsi(wsi,
  714. LWS_CLOSE_STATUS_NOSTATUS_CONTEXT_DESTROY
  715. /* no protocol close */);
  716. n--;
  717. }
  718. lws_pt_mutex_destroy(pt);
  719. }
  720. /*
  721. * give all extensions a chance to clean up any per-context
  722. * allocations they might have made
  723. */
  724. n = lws_ext_cb_all_exts(context, NULL,
  725. LWS_EXT_CB_SERVER_CONTEXT_DESTRUCT, NULL, 0);
  726. n = lws_ext_cb_all_exts(context, NULL,
  727. LWS_EXT_CB_CLIENT_CONTEXT_DESTRUCT, NULL, 0);
  728. /*
  729. * inform all the protocols that they are done and will have no more
  730. * callbacks.
  731. *
  732. * We can't free things until after the event loop shuts down.
  733. */
  734. if (context->protocol_init_done)
  735. vh = context->vhost_list;
  736. while (vh) {
  737. wsi.vhost = vh;
  738. protocol = vh->protocols;
  739. if (protocol) {
  740. n = 0;
  741. while (n < vh->count_protocols) {
  742. wsi.protocol = protocol;
  743. protocol->callback(&wsi, LWS_CALLBACK_PROTOCOL_DESTROY,
  744. NULL, NULL, 0);
  745. protocol++;
  746. n++;
  747. }
  748. }
  749. vh = vh->vhost_next;
  750. }
  751. for (n = 0; n < context->count_threads; n++) {
  752. pt = &context->pt[n];
  753. lws_libev_destroyloop(context, n);
  754. lws_libuv_destroyloop(context, n);
  755. lws_free_set_NULL(context->pt[n].serv_buf);
  756. if (pt->ah_pool)
  757. lws_free(pt->ah_pool);
  758. if (pt->http_header_data)
  759. lws_free(pt->http_header_data);
  760. }
  761. lws_plat_context_early_destroy(context);
  762. lws_ssl_context_destroy(context);
  763. if (context->pt[0].fds)
  764. lws_free_set_NULL(context->pt[0].fds);
  765. /* free all the vhost allocations */
  766. vh = context->vhost_list;
  767. while (vh) {
  768. protocol = vh->protocols;
  769. if (protocol) {
  770. n = 0;
  771. while (n < vh->count_protocols) {
  772. if (vh->protocol_vh_privs &&
  773. vh->protocol_vh_privs[n]) {
  774. lws_free(vh->protocol_vh_privs[n]);
  775. vh->protocol_vh_privs[n] = NULL;
  776. }
  777. protocol++;
  778. n++;
  779. }
  780. }
  781. if (vh->protocol_vh_privs)
  782. lws_free(vh->protocol_vh_privs);
  783. lws_ssl_SSL_CTX_destroy(vh);
  784. lws_free(vh->same_vh_protocol_list);
  785. #ifdef LWS_WITH_PLUGINS
  786. if (context->plugin_list)
  787. lws_free((void *)vh->protocols);
  788. #ifndef LWS_NO_EXTENSIONS
  789. if (context->plugin_extension_count)
  790. lws_free((void *)vh->extensions);
  791. #endif
  792. #endif
  793. #ifdef LWS_WITH_ACCESS_LOG
  794. if (vh->log_fd != (int)LWS_INVALID_FILE)
  795. close(vh->log_fd);
  796. #endif
  797. vh1 = vh->vhost_next;
  798. lws_free(vh);
  799. vh = vh1;
  800. }
  801. lws_plat_context_late_destroy(context);
  802. lws_free(context);
  803. }