Project homepage Mailing List  Warmcat.com  API Docs  Github Mirror 
{"schema":"libjg2-1", "vpath":"/git/", "avatar":"/git/avatar/", "alang":"", "gen_ut":1711642003, "reponame":"libwebsockets", "desc":"libwebsockets lightweight C networking library", "owner": { "name": "Andy Green", "email": "andy@warmcat.com", "md5": "c50933ca2aa61e0fe2c43d46bb6b59cb" },"url":"https://libwebsockets.org/repo/libwebsockets", "f":3, "items": [ {"schema":"libjg2-1", "cid":"5d9fd48a3f5d5f61ffd682cd2960b178", "commit": {"type":"commit", "time": 1525260453, "time_ofs": 480, "oid_tree": { "oid": "a38a861b57c62ac90b0bb8aee24f9d6d5ffb8155", "alias": []}, "oid":{ "oid": "a01ad0dd2076cf597c58cbb945ba6dbae3c9878c", "alias": []}, "msg": "hrtimer: add support for libevent and libev", "sig_commit": { "git_time": { "time": 1525260453, "offset": 480 }, "name": "Andy Green", "email": "andy@warmcat.com", "md5": "c50933ca2aa61e0fe2c43d46bb6b59cb" }, "sig_author": { "git_time": { "time": 1525257358, "offset": 480 }, "name": "Andy Green", "email": "andy@warmcat.com", "md5": "c50933ca2aa61e0fe2c43d46bb6b59cb" }}, "body": "hrtimer: add support for libevent and libev" , "diff": "diff --git a/.gitignore b/.gitignore\nindex d2b3e44..74c0e3c 100644\n--- a/.gitignore\n+++ b/.gitignore\n@@ -47,3 +47,5 @@ libwebsockets.pc\n build/\n *.swp\n doc\n+/build2/\n+/build3/\ndiff --git a/lib/context.c b/lib/context.c\nindex c85a481..2ccfbe9 100644\n--- a/lib/context.c\n+++ b/lib/context.c\n@@ -1163,6 +1163,7 @@ lws_create_context(const struct lws_context_creation_info *info)\n \tlwsl_info(\u0022Using event loop: %s\u005cn\u0022, context-\u003eevent_loop_ops-\u003ename);\n \n #if defined(LWS_WITH_TLS)\n+\ttime(\u0026context-\u003etls.last_cert_check_s);\n \tif (info-\u003ealpn)\n \t\tcontext-\u003etls.alpn_default \u003d info-\u003ealpn;\n \telse {\n@@ -1365,8 +1366,6 @@ lws_create_context(const struct lws_context_creation_info *info)\n \tif (!lws_check_opt(info-\u003eoptions, LWS_SERVER_OPTION_EXPLICIT_VHOSTS))\n \t\tlws_plat_drop_app_privileges(info);\n \n-\ttime(\u0026context-\u003elast_cert_check_s);\n-\n \t/* expedite post-context init (eg, protocols) */\n \tlws_cancel_service(context);\n \n@@ -1729,6 +1728,22 @@ static void\n lws_context_destroy3(struct lws_context *context)\n {\n \tstruct lws_context **pcontext_finalize \u003d context-\u003epcontext_finalize;\n+\tstruct lws_context_per_thread *pt;\n+\tint n;\n+\n+\tfor (n \u003d 0; n \u003c context-\u003ecount_threads; n++) {\n+\t\tpt \u003d \u0026context-\u003ept[n];\n+\n+\t\tif (context-\u003eevent_loop_ops-\u003edestroy_pt)\n+\t\t\tcontext-\u003eevent_loop_ops-\u003edestroy_pt(context, n);\n+\n+\t\tlws_free_set_NULL(context-\u003ept[n].serv_buf);\n+\n+#if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)\n+\t\twhile (pt-\u003ehttp.ah_list)\n+\t\t\t_lws_destroy_ah(pt, pt-\u003ehttp.ah_list);\n+#endif\n+\t}\n \n \tlws_free(context);\n \tlwsl_info(\u0022%s: ctx %p freed\u005cn\u0022, __func__, context);\n@@ -1746,8 +1761,9 @@ lws_context_destroy2(struct lws_context *context)\n {\n \tstruct lws_vhost *vh \u003d NULL, *vh1;\n #if defined(LWS_WITH_PEER_LIMITS)\n-\tuint32_t n;\n+\tuint32_t nu;\n #endif\n+\tint n;\n \n \tlwsl_info(\u0022%s: ctx %p\u005cn\u0022, __func__, context);\n \n@@ -1780,9 +1796,9 @@ lws_context_destroy2(struct lws_context *context)\n \tlws_plat_context_late_destroy(context);\n \n #if defined(LWS_WITH_PEER_LIMITS)\n-\tfor (n \u003d 0; n \u003c context-\u003epl_hash_elements; n++)\t{\n+\tfor (nu \u003d 0; nu \u003c context-\u003epl_hash_elements; nu++)\t{\n \t\tlws_start_foreach_llp(struct lws_peer **, peer,\n-\t\t\t\t context-\u003epl_hash_table[n]) {\n+\t\t\t\t context-\u003epl_hash_table[nu]) {\n \t\t\tstruct lws_peer *df \u003d *peer;\n \t\t\t*peer \u003d df-\u003enext;\n \t\t\tlws_free(df);\n@@ -1807,6 +1823,11 @@ lws_context_destroy2(struct lws_context *context)\n \t\t\treturn;\n \t\t}\n \n+\tif (!context-\u003ept[0].event_loop_foreign)\n+\t\tfor (n \u003d 0; n \u003c context-\u003ecount_threads; n++)\n+\t\t\tif (context-\u003ept[n].inside_service)\n+\t\t\t\treturn;\n+\n \tlws_context_destroy3(context);\n }\n \n@@ -1844,6 +1865,8 @@ lws_context_destroy(struct lws_context *context)\n \t\t}\n \t\tlwsl_info(\u0022%s: ctx %p: already being destroyed\u005cn\u0022,\n \t\t\t __func__, context);\n+\n+\t\tlws_context_destroy3(context);\n \t\treturn;\n \t}\n \n@@ -1891,20 +1914,6 @@ lws_context_destroy(struct lws_context *context)\n \t\tlws_pt_mutex_destroy(pt);\n \t}\n \n-\tfor (n \u003d 0; n \u003c context-\u003ecount_threads; n++) {\n-\t\tpt \u003d \u0026context-\u003ept[n];\n-\n-\t\tif (context-\u003eevent_loop_ops-\u003edestroy_pt)\n-\t\t\tcontext-\u003eevent_loop_ops-\u003edestroy_pt(context, n);\n-\n-\t\tlws_free_set_NULL(context-\u003ept[n].serv_buf);\n-\n-#if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)\n-\t\twhile (pt-\u003ehttp.ah_list)\n-\t\t\t_lws_destroy_ah(pt, pt-\u003ehttp.ah_list);\n-#endif\n-\t}\n-\n \t/*\n \t * inform all the protocols that they are done and will have no more\n \t * callbacks.\n@@ -1944,10 +1953,5 @@ lws_context_destroy(struct lws_context *context)\n \t\treturn;\n \t}\n \n-\tif (!context-\u003ept[0].event_loop_foreign)\n-\t\tfor (n \u003d 0; n \u003c context-\u003ecount_threads; n++)\n-\t\t\tif (context-\u003ept[n].inside_service)\n-\t\t\t\treturn;\n-\n \tlws_context_destroy2(context);\n }\ndiff --git a/lib/event-libs/libev/libev.c b/lib/event-libs/libev/libev.c\nindex 746d571..d60072c 100644\n--- a/lib/event-libs/libev/libev.c\n+++ b/lib/event-libs/libev/libev.c\n@@ -1,7 +1,7 @@\n /*\n * libwebsockets - small server side websockets and web server implementation\n *\n- * Copyright (C) 2010-2017 Andy Green \u003candy@warmcat.com\u003e\n+ * Copyright (C) 2010-2018 Andy Green \u003candy@warmcat.com\u003e\n *\n * This library is free software; you can redistribute it and/or\n * modify it under the terms of the GNU Lesser General Public\n@@ -22,12 +22,65 @@\n #include \u0022private-libwebsockets.h\u0022\n \n static void\n+lws_ev_hrtimer_cb(struct ev_loop *loop, struct ev_timer *watcher, int revents)\n+{\n+\tstruct lws_context_per_thread *pt \u003d\n+\t\t\t(struct lws_context_per_thread *)watcher-\u003edata;\n+\tlws_usec_t us;\n+\n+\tlws_pt_lock(pt, __func__);\n+\tus \u003d __lws_hrtimer_service(pt);\n+\tif (us !\u003d LWS_HRTIMER_NOWAIT) {\n+\t\tev_timer_set(\u0026pt-\u003eev.hrtimer, ((float)us) / 1000000.0, 0);\n+\t\tev_timer_start(pt-\u003eev.io_loop, \u0026pt-\u003eev.hrtimer);\n+\t}\n+\tlws_pt_unlock(pt);\n+}\n+\n+static void\n+lws_ev_idle_cb(struct ev_loop *loop, struct ev_idle *handle, int revents)\n+{\n+\tstruct lws_context_per_thread *pt \u003d lws_container_of(handle,\n+\t\t\t\t\tstruct lws_context_per_thread, ev.idle);\n+\tlws_usec_t us;\n+\n+\tlws_service_do_ripe_rxflow(pt);\n+\n+\t/*\n+\t * is there anybody with pending stuff that needs service forcing?\n+\t */\n+\tif (!lws_service_adjust_timeout(pt-\u003econtext, 1, pt-\u003etid)) {\n+\t\t/* -1 timeout means just do forced service */\n+\t\t_lws_plat_service_tsi(pt-\u003econtext, -1, pt-\u003etid);\n+\t\t/* still somebody left who wants forced service? */\n+\t\tif (!lws_service_adjust_timeout(pt-\u003econtext, 1, pt-\u003etid))\n+\t\t\t/* yes... come back again later */\n+\t\treturn;\n+\t}\n+\n+\t/* account for hrtimer */\n+\n+\tlws_pt_lock(pt, __func__);\n+\tus \u003d __lws_hrtimer_service(pt);\n+\tif (us !\u003d LWS_HRTIMER_NOWAIT) {\n+\t\tev_timer_set(\u0026pt-\u003eev.hrtimer, ((float)us) / 1000000.0, 0);\n+\t\tev_timer_start(pt-\u003eev.io_loop, \u0026pt-\u003eev.hrtimer);\n+\t}\n+\tlws_pt_unlock(pt);\n+\n+\t/* there is nobody who needs service forcing, shut down idle */\n+\tev_idle_stop(loop, handle);\n+}\n+\n+static void\n lws_accept_cb(struct ev_loop *loop, struct ev_io *watcher, int revents)\n {\n+\tstruct lws_context_per_thread *pt;\n \tstruct lws_io_watcher *lws_io \u003d lws_container_of(watcher,\n \t\t\t\t\tstruct lws_io_watcher, ev.watcher);\n \tstruct lws_context *context \u003d lws_io-\u003econtext;\n \tstruct lws_pollfd eventfd;\n+\tstruct lws *wsi;\n \n \tif (revents \u0026 EV_ERROR)\n \t\treturn;\n@@ -45,7 +98,12 @@ lws_accept_cb(struct ev_loop *loop, struct ev_io *watcher, int revents)\n \t\teventfd.revents |\u003d LWS_POLLOUT;\n \t}\n \n-\tlws_service_fd(context, \u0026eventfd);\n+\twsi \u003d wsi_from_fd(context, watcher-\u003efd);\n+\tpt \u003d \u0026context-\u003ept[(int)wsi-\u003etsi];\n+\n+\tlws_service_fd_tsi(context, \u0026eventfd, (int)wsi-\u003etsi);\n+\n+\tev_idle_start(pt-\u003eev.io_loop, \u0026pt-\u003eev.idle);\n }\n \n LWS_VISIBLE void\n@@ -64,6 +122,7 @@ lws_ev_sigint_cb(struct ev_loop *loop, struct ev_signal *watcher, int revents)\n static int\n elops_init_pt_ev(struct lws_context *context, void *_loop, int tsi)\n {\n+\tstruct lws_context_per_thread *pt \u003d \u0026context-\u003ept[tsi];\n \tstruct ev_signal *w_sigint \u003d \u0026context-\u003ept[tsi].w_sigint.ev.watcher;\n \tstruct lws_vhost *vh \u003d context-\u003evhost_list;\n \tconst char *backend_name;\n@@ -84,7 +143,7 @@ elops_init_pt_ev(struct lws_context *context, void *_loop, int tsi)\n \t\treturn -1;\n \t}\n \n-\tcontext-\u003ept[tsi].ev.io_loop \u003d loop;\n+\tpt-\u003eev.io_loop \u003d loop;\n \n \t/*\n \t * Initialize the accept w_accept with all the listening sockets\n@@ -138,6 +197,11 @@ elops_init_pt_ev(struct lws_context *context, void *_loop, int tsi)\n \tlwsl_info(\u0022 libev backend: %s\u005cn\u0022, backend_name);\n \t(void)backend_name;\n \n+\tev_timer_init(\u0026pt-\u003eev.hrtimer, lws_ev_hrtimer_cb, 0, 0);\n+\tpt-\u003eev.hrtimer.data \u003d pt;\n+\n+\tev_idle_init(\u0026pt-\u003eev.idle, lws_ev_idle_cb);\n+\n \treturn status;\n }\n \n@@ -147,17 +211,22 @@ elops_destroy_pt_ev(struct lws_context *context, int tsi)\n \tstruct lws_context_per_thread *pt \u003d \u0026context-\u003ept[tsi];\n \tstruct lws_vhost *vh \u003d context-\u003evhost_list;\n \n-\tif (!pt-\u003eev.io_loop)\n-\t\treturn;\n-\n \twhile (vh) {\n \t\tif (vh-\u003elserv_wsi)\n \t\t\tev_io_stop(pt-\u003eev.io_loop, \u0026vh-\u003ew_accept.ev.watcher);\n \t\tvh \u003d vh-\u003evhost_next;\n \t}\n-\tif (!pt-\u003eevent_loop_foreign)\n+\n+\t/* static assets */\n+\n+\tev_timer_stop(pt-\u003eev.io_loop, \u0026pt-\u003eev.hrtimer);\n+\tev_idle_stop(pt-\u003eev.io_loop, \u0026pt-\u003eev.idle);\n+\n+\tif (!pt-\u003eevent_loop_foreign) {\n \t\tev_signal_stop(pt-\u003eev.io_loop, \u0026pt-\u003ew_sigint.ev.watcher);\n \n+\t\tev_loop_destroy(pt-\u003eev.io_loop);\n+\t}\n }\n \n static int\n@@ -226,7 +295,7 @@ static int\n elops_destroy_context2_ev(struct lws_context *context)\n {\n \tstruct lws_context_per_thread *pt;\n-\tint n, m, internal \u003d 0;\n+\tint n, m;\n \n \tlwsl_debug(\u0022%s\u005cn\u0022, __func__);\n \n@@ -240,7 +309,6 @@ elops_destroy_context2_ev(struct lws_context *context)\n \t\tif (pt-\u003eevent_loop_foreign || !pt-\u003eev.io_loop)\n \t\t\tcontinue;\n \n-\t\tinternal \u003d 1;\n \t\tif (!context-\u003efinalize_destroy_after_internal_loops_stopped) {\n \t\t\tev_break(pt-\u003eev.io_loop, EVBREAK_ONE);\n \t\t\tcontinue;\n@@ -252,7 +320,7 @@ elops_destroy_context2_ev(struct lws_context *context)\n \t\tev_loop_destroy(pt-\u003eev.io_loop);\n \t}\n \n-\treturn internal;\n+\treturn 0;\n }\n \n static int\n@@ -281,6 +349,15 @@ elops_init_vhost_listen_wsi_ev(struct lws *wsi)\n \treturn 0;\n }\n \n+static void\n+elops_destroy_wsi_ev(struct lws *wsi)\n+{\n+\tstruct lws_context_per_thread *pt \u003d \u0026wsi-\u003econtext-\u003ept[(int)wsi-\u003etsi];\n+\n+\tev_io_stop(pt-\u003eev.io_loop, \u0026wsi-\u003ew_read.ev.watcher);\n+\tev_io_stop(pt-\u003eev.io_loop, \u0026wsi-\u003ew_write.ev.watcher);\n+}\n+\n struct lws_event_loop_ops event_loop_ops_ev \u003d {\n \t/* name */\t\t\t\u0022libev\u0022,\n \t/* init_context */\t\telops_init_context_ev,\n@@ -295,7 +372,7 @@ struct lws_event_loop_ops event_loop_ops_ev \u003d {\n \t/* io */\t\t\telops_io_ev,\n \t/* run_pt */\t\t\telops_run_pt_ev,\n \t/* destroy_pt */\t\telops_destroy_pt_ev,\n-\t/* destroy wsi */\t\tNULL,\n+\t/* destroy wsi */\t\telops_destroy_wsi_ev,\n \n \t/* periodic_events_available */\t0,\n };\ndiff --git a/lib/event-libs/libev/private.h b/lib/event-libs/libev/private.h\nindex 54e1b12..52de727 100644\n--- a/lib/event-libs/libev/private.h\n+++ b/lib/event-libs/libev/private.h\n@@ -23,9 +23,19 @@\n \n #include \u003cev.h\u003e\n \n+#define LWS_EV_REFCOUNT_STATIC_HANDLE_NEW(_x, _ctx) \u005c\n+\t\t{ (_x)-\u003edata \u003d _ctx; \u005c\n+\t\t_ctx-\u003ecount_event_loop_static_asset_handles++; }\n+#define LWS_EV_REFCOUNT_STATIC_HANDLE_TO_CONTEXT(_x) \u005c\n+\t\t\t((struct lws_context *)(_x)-\u003edata)))\n+#define LWS_EV_REFCOUNT_STATIC_HANDLE_DESTROYED(_x) \u005c\n+\t\t(--(LWS_UV_REFCOUNT_STATIC_HANDLE_TO_CONTEXT(_x)-\u003e \u005c\n+\t\t\t\tcount_event_loop_static_asset_handles))\n+\n struct lws_pt_eventlibs_libev {\n \tstruct ev_loop *io_loop;\n \tstruct ev_timer hrtimer;\n+\tstruct ev_idle idle;\n };\n \n struct lws_io_watcher_libev {\ndiff --git a/lib/event-libs/libevent/libevent.c b/lib/event-libs/libevent/libevent.c\nindex 43dda05..5cb4726 100644\n--- a/lib/event-libs/libevent/libevent.c\n+++ b/lib/event-libs/libevent/libevent.c\n@@ -22,23 +22,82 @@\n #include \u0022private-libwebsockets.h\u0022\n \n static void\n+lws_event_hrtimer_cb(int fd, short event, void *p)\n+{\n+\tstruct lws_context_per_thread *pt \u003d (struct lws_context_per_thread *)p;\n+\tstruct timeval tv;\n+\tlws_usec_t us;\n+\n+\tlws_pt_lock(pt, __func__);\n+\tus \u003d __lws_hrtimer_service(pt);\n+\tif (us !\u003d LWS_HRTIMER_NOWAIT) {\n+\t\ttv.tv_sec \u003d us / 1000000;\n+\t\ttv.tv_usec \u003d us - (tv.tv_sec * 1000000);\n+\t\tevtimer_add(pt-\u003eevent.hrtimer, \u0026tv);\n+\t}\n+\tlws_pt_unlock(pt);\n+}\n+\n+static void\n+lws_event_idle_timer_cb(int fd, short event, void *p)\n+{\n+\tstruct lws_context_per_thread *pt \u003d (struct lws_context_per_thread *)p;\n+\tstruct timeval tv;\n+\tlws_usec_t us;\n+\n+\tlws_service_do_ripe_rxflow(pt);\n+\n+\t/*\n+\t * is there anybody with pending stuff that needs service forcing?\n+\t */\n+\tif (!lws_service_adjust_timeout(pt-\u003econtext, 1, pt-\u003etid)) {\n+\t\t/* -1 timeout means just do forced service */\n+\t\t_lws_plat_service_tsi(pt-\u003econtext, -1, pt-\u003etid);\n+\t\t/* still somebody left who wants forced service? */\n+\t\tif (!lws_service_adjust_timeout(pt-\u003econtext, 1, pt-\u003etid)) {\n+\t\t\t/* yes... come back again later */\n+\n+\t\t\ttv.tv_sec \u003d 0;\n+\t\t\ttv.tv_usec \u003d 1000;\n+\t\t\tevtimer_add(pt-\u003eevent.idle_timer, \u0026tv);\n+\n+\t\t\treturn;\n+\t\t}\n+\t}\n+\n+\t/* account for hrtimer */\n+\n+\tlws_pt_lock(pt, __func__);\n+\tus \u003d __lws_hrtimer_service(pt);\n+\tif (us !\u003d LWS_HRTIMER_NOWAIT) {\n+\t\ttv.tv_sec \u003d us / 1000000;\n+\t\ttv.tv_usec \u003d us - (tv.tv_sec * 1000000);\n+\t\tevtimer_add(pt-\u003eevent.hrtimer, \u0026tv);\n+\t}\n+\tlws_pt_unlock(pt);\n+}\n+\n+static void\n lws_event_cb(evutil_socket_t sock_fd, short revents, void *ctx)\n {\n \tstruct lws_io_watcher *lws_io \u003d (struct lws_io_watcher *)ctx;\n \tstruct lws_context *context \u003d lws_io-\u003econtext;\n+\tstruct lws_context_per_thread *pt;\n \tstruct lws_pollfd eventfd;\n+\tstruct timeval tv;\n+\tstruct lws *wsi;\n \n \tif (revents \u0026 EV_TIMEOUT)\n \t\treturn;\n \n \t/* !!! EV_CLOSED doesn't exist in libevent2 */\n-\t#if LIBEVENT_VERSION_NUMBER \u003c 0x02000000\n+#if LIBEVENT_VERSION_NUMBER \u003c 0x02000000\n \tif (revents \u0026 EV_CLOSED) {\n \t\tevent_del(lws_io-\u003eevent.watcher);\n \t\tevent_free(lws_io-\u003eevent.watcher);\n \t\treturn;\n \t}\n-\t#endif\n+#endif\n \n \teventfd.fd \u003d sock_fd;\n \teventfd.events \u003d 0;\n@@ -52,7 +111,16 @@ lws_event_cb(evutil_socket_t sock_fd, short revents, void *ctx)\n \t\teventfd.revents |\u003d LWS_POLLOUT;\n \t}\n \n-\tlws_service_fd(context, \u0026eventfd);\n+\twsi \u003d wsi_from_fd(context, sock_fd);\n+\tpt \u003d \u0026context-\u003ept[(int)wsi-\u003etsi];\n+\n+\tlws_service_fd_tsi(context, \u0026eventfd, wsi-\u003etsi);\n+\n+\t/* set the idle timer for 1ms ahead */\n+\n+\ttv.tv_sec \u003d 0;\n+\ttv.tv_usec \u003d 1000;\n+\tevtimer_add(pt-\u003eevent.idle_timer, \u0026tv);\n }\n \n LWS_VISIBLE void\n@@ -77,6 +145,7 @@ elops_init_pt_event(struct lws_context *context, void *_loop, int tsi)\n {\n \tstruct lws_vhost *vh \u003d context-\u003evhost_list;\n \tstruct event_base *loop \u003d (struct event_base *)_loop;\n+\tstruct lws_context_per_thread *pt \u003d \u0026context-\u003ept[tsi];\n \n \tlwsl_info(\u0022%s: loop %p\u005cn\u0022, __func__, _loop);\n \n@@ -91,7 +160,7 @@ elops_init_pt_event(struct lws_context *context, void *_loop, int tsi)\n \t\treturn -1;\n \t}\n \n-\tcontext-\u003ept[tsi].event.io_loop \u003d loop;\n+\tpt-\u003eevent.io_loop \u003d loop;\n \n \t/*\n \t* Initialize all events with the listening sockets\n@@ -110,13 +179,22 @@ elops_init_pt_event(struct lws_context *context, void *_loop, int tsi)\n \t\tvh \u003d vh-\u003evhost_next;\n \t}\n \n+\t/* static event loop objects */\n+\n+\tpt-\u003eevent.hrtimer \u003d event_new(loop, -1, EV_PERSIST,\n+\t\t\t\t lws_event_hrtimer_cb, pt);\n+\n+\tpt-\u003eevent.idle_timer \u003d event_new(loop, -1, EV_PERSIST,\n+\t\t\t\t lws_event_idle_timer_cb, pt);\n+\n \t/* Register the signal watcher unless it's a foreign loop */\n-\tif (context-\u003ept[tsi].event_loop_foreign)\n+\n+\tif (pt-\u003eevent_loop_foreign)\n \t\treturn 0;\n \n-\tcontext-\u003ept[tsi].w_sigint.event.watcher \u003d evsignal_new(loop, SIGINT,\n-\t\t\tlws_event_sigint_cb, \u0026context-\u003ept[tsi]);\n-\tevent_add(context-\u003ept[tsi].w_sigint.event.watcher, NULL);\n+\tpt-\u003ew_sigint.event.watcher \u003d evsignal_new(loop, SIGINT,\n+\t\t\t\t\t\t lws_event_sigint_cb, pt);\n+\tevent_add(pt-\u003ew_sigint.event.watcher, NULL);\n \n \treturn 0;\n }\n@@ -217,8 +295,15 @@ elops_destroy_pt_event(struct lws_context *context, int tsi)\n \t\tvh \u003d vh-\u003evhost_next;\n \t}\n \n-\tif (!pt-\u003eevent_loop_foreign)\n+\tevent_free(pt-\u003eevent.hrtimer);\n+\tevent_free(pt-\u003eevent.idle_timer);\n+\n+\tif (!pt-\u003eevent_loop_foreign) {\n+\t\tevent_del(pt-\u003ew_sigint.event.watcher);\n \t\tevent_free(pt-\u003ew_sigint.event.watcher);\n+\n+\t\tevent_base_free(pt-\u003eevent.io_loop);\n+\t}\n }\n \n static void\n@@ -227,10 +312,10 @@ elops_destroy_wsi_event(struct lws *wsi)\n \tif (!wsi)\n \t\treturn;\n \n-\tif(wsi-\u003ew_read.event.watcher)\n+\tif (wsi-\u003ew_read.event.watcher)\n \t\tevent_free(wsi-\u003ew_read.event.watcher);\n \n-\tif(wsi-\u003ew_write.event.watcher)\n+\tif (wsi-\u003ew_write.event.watcher)\n \t\tevent_free(wsi-\u003ew_write.event.watcher);\n }\n \n@@ -271,7 +356,7 @@ static int\n elops_destroy_context2_event(struct lws_context *context)\n {\n \tstruct lws_context_per_thread *pt;\n-\tint n, m, internal \u003d 0;\n+\tint n, m;\n \n \tlwsl_debug(\u0022%s\u005cn\u0022, __func__);\n \n@@ -285,7 +370,6 @@ elops_destroy_context2_event(struct lws_context *context)\n \t\tif (pt-\u003eevent_loop_foreign || !pt-\u003eevent.io_loop)\n \t\t\tcontinue;\n \n-\t\tinternal \u003d 1;\n \t\tif (!context-\u003efinalize_destroy_after_internal_loops_stopped) {\n \t\t\tevent_base_loopexit(pt-\u003eevent.io_loop, NULL);\n \t\t\tcontinue;\n@@ -305,7 +389,7 @@ elops_destroy_context2_event(struct lws_context *context)\n \n \t}\n \n-\treturn internal;\n+\treturn 0;\n }\n \n struct lws_event_loop_ops event_loop_ops_event \u003d {\ndiff --git a/lib/event-libs/libevent/private.h b/lib/event-libs/libevent/private.h\nindex 347c7fe..1c2d360 100644\n--- a/lib/event-libs/libevent/private.h\n+++ b/lib/event-libs/libevent/private.h\n@@ -26,6 +26,7 @@\n struct lws_pt_eventlibs_libevent {\n \tstruct event_base *io_loop;\n \tstruct event *hrtimer;\n+\tstruct event *idle_timer;\n };\n \n struct lws_io_watcher_libevent {\ndiff --git a/lib/event-libs/libuv/libuv.c b/lib/event-libs/libuv/libuv.c\nindex ecc7e2d..ee948b6 100644\n--- a/lib/event-libs/libuv/libuv.c\n+++ b/lib/event-libs/libuv/libuv.c\n@@ -1,7 +1,7 @@\n /*\n * libwebsockets - small server side websockets and web server implementation\n *\n- * Copyright (C) 2010-2017 Andy Green \u003candy@warmcat.com\u003e\n+ * Copyright (C) 2010-2018 Andy Green \u003candy@warmcat.com\u003e\n *\n * This library is free software; you can redistribute it and/or\n * modify it under the terms of the GNU Lesser General Public\n@@ -116,11 +116,7 @@ lws_io_cb(uv_poll_t *watcher, int status, int revents)\n \t\t\teventfd.revents |\u003d LWS_POLLOUT;\n \t\t}\n \t}\n-\tlws_service_fd(context, \u0026eventfd);\n-\n-\tlws_pt_lock(pt, __func__);\n-\t__lws_hrtimer_service(pt);\n-\tlws_pt_unlock(pt);\n+\tlws_service_fd_tsi(context, \u0026eventfd, wsi-\u003etsi);\n \n \tuv_idle_start(\u0026pt-\u003euv.idle, lws_uv_idle);\n }\n@@ -526,7 +522,8 @@ elops_destroy_context1_uv(struct lws_context *context)\n \t\t\t\t\t\t UV_RUN_NOWAIT)))\n \t\t\t\t\t;\n \t\t\tif (m)\n-\t\t\t\tlwsl_err(\u0022%s: tsi %d: failed to close everything\u005cn\u0022, __func__, n);\n+\t\t\t\tlwsl_err(\u0022%s: tsi %d: not all closed\u005cn\u0022,\n+\t\t\t\t\t __func__, n);\n \n \t\t}\n \t}\n@@ -891,8 +888,9 @@ lws_libuv_closewsi(uv_handle_t* handle)\n \t\t\tvh \u003d vh-\u003evhost_next;\n \t\t}\n \n-\t\tif (context-\u003ept[0].event_loop_foreign) {\n-\t\t\tlwsl_info(\u0022%s: calling lws_context_destroy2\u005cn\u0022, __func__);\n+\t\tif (!context-\u003ecount_event_loop_static_asset_handles \u0026\u0026\n+\t\t context-\u003ept[0].event_loop_foreign) {\n+\t\t\tlwsl_info(\u0022%s: call lws_context_destroy2\u005cn\u0022, __func__);\n \t\t\tlws_context_destroy2(context);\n \t\t}\n \t}\ndiff --git a/lib/event-libs/libuv/private.h b/lib/event-libs/libuv/private.h\nindex 173340e..cf8f710 100644\n--- a/lib/event-libs/libuv/private.h\n+++ b/lib/event-libs/libuv/private.h\n@@ -24,6 +24,11 @@\n #include \u003cuv.h\u003e\n \n /*\n+ * libuv's async destroy cb means that asking to close something doesn't mean\n+ * you can destroy it or parent things until after the close completes.\n+ *\n+ * So we must reference-count creation and close completions with libuv.\n+ *\n * All \u0022static\u0022 (per-pt or per-context) uv handles must\n *\n * - have their .data set to point to the context\ndiff --git a/lib/private-libwebsockets.h b/lib/private-libwebsockets.h\nindex 4ff5878..9f60bd6 100644\n--- a/lib/private-libwebsockets.h\n+++ b/lib/private-libwebsockets.h\n@@ -757,7 +757,6 @@ struct lws_peer {\n struct lws_context {\n \ttime_t last_timeout_check_s;\n \ttime_t last_ws_ping_pong_check_s;\n-\ttime_t last_cert_check_s;\n \ttime_t time_up;\n \ttime_t time_discontiguity;\n \ttime_t time_fixup;\ndiff --git a/lib/service.c b/lib/service.c\nindex 8cf8c93..a08506f 100644\n--- a/lib/service.c\n+++ b/lib/service.c\n@@ -554,9 +554,6 @@ lws_service_periodic_checks(struct lws_context *context,\n \tstruct lws *wsi;\n \tint timed_out \u003d 0;\n \ttime_t now;\n-#if defined(LWS_WITH_TLS)\n-\tint n \u003d 0;\n-#endif\n #if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)\n \tstruct allocated_headers *ah;\n \tint m;\n@@ -810,12 +807,10 @@ lws_service_periodic_checks(struct lws_context *context,\n \t/*\n \t * Phase 6: check the remaining cert lifetime daily\n \t */\n-#if defined(LWS_WITH_TLS)\n-\tn \u003d lws_compare_time_t(context, now, context-\u003elast_cert_check_s);\n-\tif ((!context-\u003elast_cert_check_s || n \u003e (24 * 60 * 60)) \u0026\u0026\n-\t !lws_tls_check_all_cert_lifetimes(context))\n-\t\tcontext-\u003elast_cert_check_s \u003d now;\n-#endif\n+\n+\tif (context-\u003etls_ops \u0026\u0026\n+\t context-\u003etls_ops-\u003eperiodic_housekeeping)\n+\t\tcontext-\u003etls_ops-\u003eperiodic_housekeeping(context, now);\n \n \treturn timed_out;\n }\n@@ -869,7 +864,8 @@ lws_service_fd_tsi(struct lws_context *context, struct lws_pollfd *pollfd,\n \t}\n \n #if defined(LWS_WITH_TLS)\n-\tif (lwsi_state(wsi) \u003d\u003d LRS_SHUTDOWN \u0026\u0026 lws_is_ssl(wsi) \u0026\u0026 wsi-\u003etls.ssl) {\n+\tif (lwsi_state(wsi) \u003d\u003d LRS_SHUTDOWN \u0026\u0026\n+\t lws_is_ssl(wsi) \u0026\u0026 wsi-\u003etls.ssl) {\n \t\tswitch (__lws_tls_shutdown(wsi)) {\n \t\tcase LWS_SSL_CAPABLE_DONE:\n \t\tcase LWS_SSL_CAPABLE_ERROR:\n@@ -927,6 +923,10 @@ handled:\n #endif\n \tpollfd-\u003erevents \u003d 0;\n \n+\tlws_pt_lock(pt, __func__);\n+\t__lws_hrtimer_service(pt);\n+\tlws_pt_unlock(pt);\n+\n \treturn 0;\n }\n \ndiff --git a/lib/tls/mbedtls/ssl.c b/lib/tls/mbedtls/ssl.c\nindex 44eb061..b024dfa 100644\n--- a/lib/tls/mbedtls/ssl.c\n+++ b/lib/tls/mbedtls/ssl.c\n@@ -501,7 +501,20 @@ tops_fake_POLLIN_for_buffered_mbedtls(struct lws_context_per_thread *pt)\n \treturn lws_tls_fake_POLLIN_for_buffered(pt);\n }\n \n+static int\n+tops_periodic_housekeeping_mbedtls(struct lws_context *context, time_t now)\n+{\n+\tint n;\n+\n+\tn \u003d lws_compare_time_t(context, now, context-\u003etls.last_cert_check_s);\n+\tif ((!context-\u003etls.last_cert_check_s || n \u003e (24 * 60 * 60)) \u0026\u0026\n+\t !lws_tls_check_all_cert_lifetimes(context))\n+\t\tcontext-\u003etls.last_cert_check_s \u003d now;\n+\n+\treturn 0;\n+}\n+\n const struct lws_tls_ops tls_ops_mbedtls \u003d {\n \t/* fake_POLLIN_for_buffered */\ttops_fake_POLLIN_for_buffered_mbedtls,\n-\n+\t/* periodic_housekeeping */\ttops_periodic_housekeeping_mbedtls,\n };\ndiff --git a/lib/tls/openssl/ssl.c b/lib/tls/openssl/ssl.c\nindex d92ea78..ec43770 100644\n--- a/lib/tls/openssl/ssl.c\n+++ b/lib/tls/openssl/ssl.c\n@@ -689,7 +689,21 @@ tops_fake_POLLIN_for_buffered_openssl(struct lws_context_per_thread *pt)\n \treturn lws_tls_fake_POLLIN_for_buffered(pt);\n }\n \n+static int\n+tops_periodic_housekeeping_openssl(struct lws_context *context, time_t now)\n+{\n+\tint n;\n+\n+\tn \u003d lws_compare_time_t(context, now, context-\u003etls.last_cert_check_s);\n+\tif ((!context-\u003etls.last_cert_check_s || n \u003e (24 * 60 * 60)) \u0026\u0026\n+\t !lws_tls_check_all_cert_lifetimes(context))\n+\t\tcontext-\u003etls.last_cert_check_s \u003d now;\n+\n+\treturn 0;\n+}\n+\n const struct lws_tls_ops tls_ops_openssl \u003d {\n \t/* fake_POLLIN_for_buffered */\ttops_fake_POLLIN_for_buffered_openssl,\n+\t/* periodic_housekeeping */\ttops_periodic_housekeeping_openssl,\n \n };\ndiff --git a/lib/tls/private.h b/lib/tls/private.h\nindex 20f9bd3..5c6beeb 100644\n--- a/lib/tls/private.h\n+++ b/lib/tls/private.h\n@@ -93,6 +93,7 @@ struct lws_context_per_thread;\n \n struct lws_tls_ops {\n \tint (*fake_POLLIN_for_buffered)(struct lws_context_per_thread *pt);\n+\tint (*periodic_housekeeping)(struct lws_context *context, time_t now);\n };\n \n #if defined(LWS_WITH_TLS)\n@@ -110,6 +111,7 @@ extern const struct lws_tls_ops tls_ops_openssl, tls_ops_mbedtls;\n struct lws_context_tls {\n \tchar alpn_discovered[32];\n \tconst char *alpn_default;\n+\ttime_t last_cert_check_s;\n };\n \n struct lws_pt_tls {\ndiff --git a/lib/tls/tls-server.c b/lib/tls/tls-server.c\nindex 01e4ca6..19549a5 100644\n--- a/lib/tls/tls-server.c\n+++ b/lib/tls/tls-server.c\n@@ -214,7 +214,7 @@ lws_server_socket_service_ssl(struct lws *wsi, lws_sockfd_type accept_fd)\n \t\t\tlws_gate_accepts(context, 0);\n \n #if defined(LWS_WITH_STATS)\n-\tcontext-\u003eupdated \u003d 1;\n+\t\tcontext-\u003eupdated \u003d 1;\n #endif\n \t\t/*\n \t\t * we are not accepted yet, but we need to enter ourselves\ndiff --git a/minimal-examples/http-server/README.md b/minimal-examples/http-server/README.md\nindex d9ea25f..1adb694 100644\n--- a/minimal-examples/http-server/README.md\n+++ b/minimal-examples/http-server/README.md\n@@ -3,6 +3,7 @@\n minimal-http-server-basicauth|Shows how to protect a mount using a password file and basic auth\n minimal-http-server-dynamic|Serves both static and dynamically generated http content\n minimal-http-server-eventlib-foreign|Demonstrates integrating lws with a foreign event library\n+minimal-http-server-eventlib-demos|Using the demo plugins with event libraries\n minimal-http-server-eventlib|Same as minimal-http-server but works with a supported event library\n minimal-http-server-form-get|Process a GET form\n minimal-http-server-form-post-file|Process a multipart POST form with file transfer\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/CMakeLists.txt b/minimal-examples/http-server/minimal-http-server-eventlib-demos/CMakeLists.txt\nnew file mode 100644\nindex 0000000..593d687\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/CMakeLists.txt\n@@ -0,0 +1,79 @@\n+cmake_minimum_required(VERSION 2.8)\n+include(CheckCSourceCompiles)\n+\n+set(SAMP lws-minimal-http-server-eventlib-demos)\n+set(SRCS minimal-http-server-eventlib-demos.c)\n+\n+# If we are being built as part of lws, confirm current build config supports\n+# reqconfig, else skip building ourselves.\n+#\n+# If we are being built externally, confirm installed lws was configured to\n+# support reqconfig, else error out with a helpful message about the problem.\n+#\n+MACRO(require_lws_config reqconfig _val result)\n+\n+\tif (DEFINED ${reqconfig})\n+\tif (${reqconfig})\n+\t\tset (rq 1)\n+\telse()\n+\t\tset (rq 0)\n+\tendif()\n+\telse()\n+\t\tset(rq 0)\n+\tendif()\n+\n+\tif (${_val} EQUAL ${rq})\n+\t\tset(SAME 1)\n+\telse()\n+\t\tset(SAME 0)\n+\tendif()\n+\n+\tif (LWS_WITH_MINIMAL_EXAMPLES AND NOT ${SAME})\n+\t\tif (${_val})\n+\t\t\tmessage(\u0022${SAMP}: skipping as lws being built without ${reqconfig}\u0022)\n+\t\telse()\n+\t\t\tmessage(\u0022${SAMP}: skipping as lws built with ${reqconfig}\u0022)\n+\t\tendif()\n+\t\tset(${result} 0)\n+\telse()\n+\t\tif (LWS_WITH_MINIMAL_EXAMPLES)\n+\t\t\tset(MET ${SAME})\n+\t\telse()\n+\t\t\tCHECK_C_SOURCE_COMPILES(\u0022#include \u003clibwebsockets.h\u003e\u005cnint main(void) {\u005cn#if defined(${reqconfig})\u005cn return 0;\u005cn#else\u005cn fail;\u005cn#endif\u005cn return 0;\u005cn}\u005cn\u0022 HAS_${reqconfig})\n+\t\t\tif (NOT DEFINED HAS_${reqconfig} OR NOT HAS_${reqconfig})\n+\t\t\t\tset(HAS_${reqconfig} 0)\n+\t\t\telse()\n+\t\t\t\tset(HAS_${reqconfig} 1)\n+\t\t\tendif()\n+\t\t\tif ((HAS_${reqconfig} AND ${_val}) OR (NOT HAS_${reqconfig} AND NOT ${_val}))\n+\t\t\t\tset(MET 1)\n+\t\t\telse()\n+\t\t\t\tset(MET 0)\n+\t\t\tendif()\n+\t\tendif()\n+\t\tif (NOT MET)\n+\t\t\tif (${_val})\n+\t\t\t\tmessage(FATAL_ERROR \u0022This project requires lws must have been configured with ${reqconfig}\u0022)\n+\t\t\telse()\n+\t\t\t\tmessage(FATAL_ERROR \u0022Lws configuration of ${reqconfig} is incompatible with this project\u0022)\n+\t\t\tendif()\n+\t\tendif()\n+\t\n+\tendif()\n+ENDMACRO()\n+\n+set(requirements 1)\n+require_lws_config(LWS_ROLE_H1 1 requirements)\n+require_lws_config(LWS_ROLE_WS 1 requirements)\n+require_lws_config(LWS_WITHOUT_SERVER 0 requirements)\n+\n+if (requirements)\n+\tadd_executable(${SAMP} ${SRCS})\n+\n+\tif (websockets_shared)\n+\t\ttarget_link_libraries(${SAMP} websockets_shared)\n+\t\tadd_dependencies(${SAMP} websockets_shared)\n+\telse()\n+\t\ttarget_link_libraries(${SAMP} websockets)\n+\tendif()\n+endif()\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/README.md b/minimal-examples/http-server/minimal-http-server-eventlib-demos/README.md\nnew file mode 100644\nindex 0000000..90720e4\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/README.md\n@@ -0,0 +1,30 @@\n+# lws minimal http server eventlib demos\n+\n+This demonstrates a slightly more complex demo that can use\n+any of the event loops (it defaults to poll)\n+\n+It uses statically included plugins to provide the lws test server functions\n+\n+Commandline option|Meaning\n+---|---\n+-d \u003cloglevel\u003e|Debug verbosity in decimal, eg, -d15\n+--uv|Use the libuv event library (lws must have been configured with `-DLWS_WITH_LIBUV\u003d1`)\n+--event|Use the libevent library (lws must have been configured with `-DLWS_WITH_LIBEVENT\u003d1`)\n+--ev|Use the libev event library (lws must have been configured with `-DLWS_WITH_LIBEV\u003d1`)\n+\n+## build\n+\n+```\n+ $ cmake . \u0026\u0026 make\n+```\n+\n+## usage\n+\n+```\n+ $ ./lws-minimal-http-server-eventlib-demos\n+[2018/03/04 09:30:02:7986] USER: LWS minimal http server-eventlib-demos | visit http://localhost:7681\n+[2018/03/04 09:30:02:7986] NOTICE: Creating Vhost 'default' port 7681, 1 protocols, IPv6 on\n+```\n+\n+Visit http://localhost:7681\n+\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.cert b/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.cert\nnew file mode 100644\nindex 0000000..6f372db\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.cert\n@@ -0,0 +1,34 @@\n+-----BEGIN CERTIFICATE-----\n+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\n+e2Hl/7M0c/lbZ4FvXgARlex2rkgS0Ka06HE\u003d\n+-----END CERTIFICATE-----\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.key b/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.key\nnew file mode 100644\nindex 0000000..148f859\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/localhost-100y.key\n@@ -0,0 +1,52 @@\n+-----BEGIN PRIVATE KEY-----\n+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\n+fBrpEY1IATtPq1taBZZogRqI3rOkkPk\u003d\n+-----END PRIVATE KEY-----\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/minimal-http-server-eventlib-demos.c b/minimal-examples/http-server/minimal-http-server-eventlib-demos/minimal-http-server-eventlib-demos.c\nnew file mode 100644\nindex 0000000..f624fa5\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/minimal-http-server-eventlib-demos.c\n@@ -0,0 +1,186 @@\n+/*\n+ * lws-minimal-http-server-eventlib\n+ *\n+ * Copyright (C) 2018 Andy Green \u003candy@warmcat.com\u003e\n+ *\n+ * This file is made available under the Creative Commons CC0 1.0\n+ * Universal Public Domain Dedication.\n+ *\n+ * This demonstrates a minimal http[s] server that can work with any of the\n+ * supported event loop backends, or the default poll() one.\n+ *\n+ * To keep it simple, it serves stuff from the subdirectory \n+ * \u0022./mount-origin\u0022 of the directory it was started in.\n+ * You can change that by changing mount.origin below.\n+ */\n+\n+#include \u003clibwebsockets.h\u003e\n+#include \u003cstring.h\u003e\n+#include \u003csignal.h\u003e\n+\n+#define LWS_PLUGIN_STATIC\n+#include \u0022../../../plugins/protocol_lws_mirror.c\u0022\n+#include \u0022../../../plugins/protocol_lws_status.c\u0022\n+#include \u0022../../../plugins/protocol_dumb_increment.c\u0022\n+#include \u0022../../../plugins/protocol_post_demo.c\u0022\n+\n+static struct lws_context *context;\n+\n+static struct lws_protocols protocols[] \u003d {\n+\t/* first protocol must always be HTTP handler */\n+\n+\t{ \u0022http-only\u0022, lws_callback_http_dummy, 0, 0, },\n+\tLWS_PLUGIN_PROTOCOL_DUMB_INCREMENT,\n+\tLWS_PLUGIN_PROTOCOL_MIRROR,\n+\tLWS_PLUGIN_PROTOCOL_LWS_STATUS,\n+\tLWS_PLUGIN_PROTOCOL_POST_DEMO,\n+\t{ NULL, NULL, 0, 0 } /* terminator */\n+};\n+\n+/*\n+ * mount handlers for sections of the URL space\n+ */\n+\n+static const struct lws_http_mount mount_ziptest \u003d {\n+\tNULL,\t\t\t/* linked-list pointer to next*/\n+\t\u0022/ziptest\u0022,\t\t/* mountpoint in URL namespace on this vhost */\n+\t\u0022candide.zip\u0022,\t/* handler */\n+\tNULL,\t/* default filename if none given */\n+\tNULL,\n+\tNULL,\n+\tNULL,\n+\tNULL,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\tLWSMPRO_FILE,\t/* origin points to a callback */\n+\t8,\t\t\t/* strlen(\u0022/ziptest\u0022), ie length of the mountpoint */\n+\tNULL,\n+\n+\t{ NULL, NULL } // sentinel\n+};\n+\n+static const struct lws_http_mount mount_post \u003d {\n+\t(struct lws_http_mount *)\u0026mount_ziptest, /* linked-list pointer to next*/\n+\t\u0022/formtest\u0022,\t\t/* mountpoint in URL namespace on this vhost */\n+\t\u0022protocol-post-demo\u0022,\t/* handler */\n+\tNULL,\t/* default filename if none given */\n+\tNULL,\n+\tNULL,\n+\tNULL,\n+\tNULL,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\t0,\n+\tLWSMPRO_CALLBACK,\t/* origin points to a callback */\n+\t9,\t\t\t/* strlen(\u0022/formtest\u0022), ie length of the mountpoint */\n+\tNULL,\n+\n+\t{ NULL, NULL } // sentinel\n+};\n+\n+\n+static const struct lws_http_mount mount \u003d {\n+\t/* .mount_next */\t\t\u0026mount_post,\t/* linked-list \u0022next\u0022 */\n+\t/* .mountpoint */\t\t\u0022/\u0022,\t\t/* mountpoint URL */\n+\t/* .origin */\t\t\t\u0022./mount-origin\u0022, /* serve from dir */\n+\t/* .def */\t\t\t\u0022test.html\u0022,\t/* default filename */\n+\t/* .protocol */\t\t\tNULL,\n+\t/* .cgienv */\t\t\tNULL,\n+\t/* .extra_mimetypes */\t\tNULL,\n+\t/* .interpret */\t\tNULL,\n+\t/* .cgi_timeout */\t\t0,\n+\t/* .cache_max_age */\t\t0,\n+\t/* .auth_mask */\t\t0,\n+\t/* .cache_reusable */\t\t0,\n+\t/* .cache_revalidate */\t\t0,\n+\t/* .cache_intermediaries */\t0,\n+\t/* .origin_protocol */\t\tLWSMPRO_FILE,\t/* files in a dir */\n+\t/* .mountpoint_len */\t\t1,\t\t/* char count */\n+\t/* .basic_auth_login_file */\tNULL,\n+};\n+\n+void signal_cb(void *handle, int signum)\n+{\n+\tlwsl_err(\u0022%s: signal %d\u005cn\u0022, __func__, signum);\n+\n+\tswitch (signum) {\n+\tcase SIGTERM:\n+\tcase SIGINT:\n+\t\tbreak;\n+\tdefault:\n+\n+\t\tbreak;\n+\t}\n+\tlws_context_destroy(context);\n+}\n+\n+void sigint_handler(int sig)\n+{\n+\tsignal_cb(NULL, sig);\n+}\n+\n+int main(int argc, const char **argv)\n+{\n+\tstruct lws_context_creation_info info;\n+\tconst char *p;\n+\tint logs \u003d LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE\n+\t\t\t/* for LLL_ verbosity above NOTICE to be built into lws,\n+\t\t\t * lws must have been configured and built with\n+\t\t\t * -DCMAKE_BUILD_TYPE\u003dDEBUG instead of \u003dRELEASE */\n+\t\t\t/* | LLL_INFO */ /* | LLL_PARSER */ /* | LLL_HEADER */\n+\t\t\t/* | LLL_EXT */ /* | LLL_CLIENT */ /* | LLL_LATENCY */\n+\t\t\t/* | LLL_DEBUG */;\n+\n+\tif ((p \u003d lws_cmdline_option(argc, argv, \u0022-d\u0022)))\n+\t\tlogs \u003d atoi(p);\n+\n+\tlws_set_log_level(logs, NULL);\n+\tlwsl_user(\u0022LWS minimal http server eventlib | visit http://localhost:7681\u005cn\u0022);\n+\tlwsl_user(\u0022 [-s (ssl)] [--uv (libuv)] [--ev (libev)] [--event (libevent)]\u005cn\u0022);\n+\n+\tmemset(\u0026info, 0, sizeof info); /* otherwise uninitialized garbage */\n+\tinfo.port \u003d 7681;\n+\tinfo.mounts \u003d \u0026mount;\n+\tinfo.error_document_404 \u003d \u0022/404.html\u0022;\n+\tinfo.pcontext \u003d \u0026context;\n+\tinfo.protocols \u003d protocols;\n+\tinfo.signal_cb \u003d signal_cb;\n+\n+\tif (lws_cmdline_option(argc, argv, \u0022-s\u0022)) {\n+\t\tinfo.options |\u003d LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT;\n+\t\tinfo.ssl_cert_filepath \u003d \u0022localhost-100y.cert\u0022;\n+\t\tinfo.ssl_private_key_filepath \u003d \u0022localhost-100y.key\u0022;\n+\t}\n+\n+\tif (lws_cmdline_option(argc, argv, \u0022--uv\u0022))\n+\t\tinfo.options |\u003d LWS_SERVER_OPTION_LIBUV;\n+\telse\n+\t\tif (lws_cmdline_option(argc, argv, \u0022--event\u0022))\n+\t\t\tinfo.options |\u003d LWS_SERVER_OPTION_LIBEVENT;\n+\t\telse\n+\t\t\tif (lws_cmdline_option(argc, argv, \u0022--ev\u0022))\n+\t\t\t\tinfo.options |\u003d LWS_SERVER_OPTION_LIBEV;\n+\t\t\telse\n+\t\t\t\tsignal(SIGINT, sigint_handler);\n+\n+\tcontext \u003d lws_create_context(\u0026info);\n+\tif (!context) {\n+\t\tlwsl_err(\u0022lws init failed\u005cn\u0022);\n+\t\treturn 1;\n+\t}\n+\n+\twhile (!lws_service(context, 0))\n+\t\t;\n+\n+\tlwsl_info(\u0022calling external context destroy\u005cn\u0022);\n+\tlws_context_destroy(context);\n+\n+\treturn 0;\n+}\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/404.html b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/404.html\nnew file mode 100644\nindex 0000000..1f7ae66\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/404.html\n@@ -0,0 +1,9 @@\n+\u003cmeta charset\u003d\u0022UTF-8\u0022\u003e \n+\u003chtml\u003e\n+\t\u003cbody\u003e\n+\t\t\u003cimg src\u003d\u0022libwebsockets.org-logo.png\u0022\u003e\u003cbr\u003e\n+\t\t\u003ch1\u003e404\u003c/h1\u003e\n+\t\tSorry, that file doesn't exist.\n+\t\u003c/body\u003e\n+\u003c/html\u003e\n+\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/candide.zip b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/candide.zip\nnew file mode 100644\nindex 0000000..82a6619\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/candide.zip differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/favicon.ico b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/favicon.ico\nnew file mode 100644\nindex 0000000..c0cc2e3\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/favicon.ico differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/http2.png b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/http2.png\nnew file mode 100644\nindex 0000000..b4129e7\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/http2.png differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/leaf.jpg b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/leaf.jpg\nnew file mode 100644\nindex 0000000..1a3f46b\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/leaf.jpg differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/libwebsockets.org-logo.png b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/libwebsockets.org-logo.png\nnew file mode 100644\nindex 0000000..2060a10\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/libwebsockets.org-logo.png differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/lws-common.js b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/lws-common.js\nnew file mode 100644\nindex 0000000..f0c5b92\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/lws-common.js\n@@ -0,0 +1,398 @@\n+/*\n+ * This section around grayOut came from here:\n+ * http://www.codingforums.com/archive/index.php/t-151720.html\n+ * Assumed public domain\n+ *\n+ * Init like this in your main html script, this also reapplies the gray\n+ *\n+ * lws_gray_out(true,{'zindex':'499'});\n+ *\n+ * To remove the gray\n+ *\n+ * lws_gray_out(false);\n+ *\n+ */\n+\n+function lws_gray_out(vis, options) {\n+\n+\tvar options \u003d options || {};\n+\tvar zindex \u003d options.zindex || 50;\n+\tvar opacity \u003d options.opacity || 70;\n+\tvar opaque \u003d (opacity / 100);\n+\tvar bgcolor \u003d options.bgcolor || '#000000';\n+\tvar dark \u003d document.getElementById('darkenScreenObject');\n+\n+\tif (!dark) {\n+\t\tvar tbody \u003d document.getElementsByTagName(\u0022body\u0022)[0];\n+\t\tvar tnode \u003d document.createElement('div');\n+\t\ttnode.style.position \u003d 'absolute';\n+\t\ttnode.style.top \u003d '0px';\n+\t\ttnode.style.left \u003d '0px';\n+\t\ttnode.style.overflow \u003d 'hidden';\n+\t\ttnode.style.display \u003d'none';\n+\t\ttnode.id \u003d 'darkenScreenObject';\n+\t\ttbody.appendChild(tnode);\n+\t\tdark \u003d document.getElementById('darkenScreenObject');\n+\t}\n+\tif (vis) {\n+\t\tdark.style.opacity \u003d opaque;\n+\t\tdark.style.MozOpacity \u003d opaque;\n+\t\t// dark.style.filter \u003d'alpha(opacity\u003d'+opacity+')';\n+\t\tdark.style.zIndex \u003d zindex;\n+\t\tdark.style.backgroundColor \u003d bgcolor;\n+\t\tdark.style.width \u003d gsize(1);\n+\t\tdark.style.height \u003d gsize(0);\n+\t\tdark.style.display \u003d'block';\n+\t\taddEvent(window, \u0022resize\u0022,\n+\t\t\tfunction() {\n+\t\t\t\tdark.style.height \u003d gsize(0);\n+\t\t\t\tdark.style.width \u003d gsize(1);\n+\t\t\t}\n+\t\t);\n+\t} else {\n+\t\tdark.style.display \u003d 'none';\n+\t\tremoveEvent(window, \u0022resize\u0022,\n+\t\t\tfunction() {\n+\t\t\t\tdark.style.height \u003d gsize(0);\n+\t\t\t\tdark.style.width \u003d gsize(1);\n+\t\t\t}\n+\t\t);\n+\t}\n+}\n+\n+function gsize(ptype)\n+{\n+\tvar h \u003d document.compatMode \u003d\u003d 'CSS1Compat' \u0026\u0026\n+\t\t!window.opera ?\n+\t\t\tdocument.documentElement.clientHeight :\n+\t\t\t\t\t\tdocument.body.clientHeight;\n+\tvar w \u003d document.compatMode \u003d\u003d 'CSS1Compat' \u0026\u0026\n+\t\t!window.opera ? \n+\t\t\tdocument.documentElement.clientWidth :\n+\t\t\t\t\t\tdocument.body.clientWidth;\n+\tif (document.body \u0026\u0026 \n+\t\t (document.body.scrollWidth || document.body.scrollHeight)) {\n+\t\tvar pageWidth \u003d (w \u003e (t \u003d document.body.scrollWidth)) ?\n+\t\t\t\t\t(\u0022\u0022 + w + \u0022px\u0022) : (\u0022\u0022 + (t) + \u0022px\u0022);\n+\t\tvar pageHeight \u003d (h \u003e (t \u003d document.body.scrollHeight)) ?\n+\t\t\t\t\t(\u0022\u0022 + h + \u0022px\u0022) : (\u0022\u0022 + (t) + \u0022px\u0022);\n+\t} else if (document.body.offsetWidth) {\n+\t\tvar pageWidth \u003d (w \u003e (t \u003d document.body.offsetWidth)) ?\n+\t\t\t\t\t(\u0022\u0022 + w + \u0022px\u0022) : (\u0022\u0022 + (t) + \u0022px\u0022);\n+\t\tvar pageHeight \u003d(h \u003e (t \u003d document.body.offsetHeight)) ?\n+\t\t\t\t\t(\u0022\u0022 + h + \u0022px\u0022) : (\u0022\u0022 + (t) + \u0022px\u0022);\n+\t} else {\n+\t\tvar pageWidth \u003d '100%';\n+\t\tvar pageHeight \u003d '100%';\n+\t}\n+\treturn (ptype \u003d\u003d 1) ? pageWidth : pageHeight;\n+}\n+\n+function addEvent( obj, type, fn ) {\n+\tif ( obj.attachEvent ) {\n+\t\tobj['e' + type + fn] \u003d fn;\n+\t\tobj[type+fn] \u003d function() { obj['e' + type+fn]( window.event );}\n+\t\tobj.attachEvent('on' + type, obj[type + fn]);\n+\t} else\n+\t\tobj.addEventListener(type, fn, false);\n+}\n+\n+function removeEvent( obj, type, fn ) {\n+\tif ( obj.detachEvent ) {\n+\t\tobj.detachEvent('on' + type, obj[type + fn]);\n+\t\tobj[type + fn] \u003d null;\n+\t} else\n+\t\tobj.removeEventListener(type, fn, false);\n+}\n+\n+/*\n+ * end of grayOut related stuff\n+ */\n+ \n+/*\n+ * lws-meta helpers\n+ */\n+\n+var lws_meta_cmd \u003d {\n+\tOPEN_SUBCHANNEL: 0x41,\n+\t/**\u003c Client requests to open new subchannel\n+\t */\n+\tOPEN_RESULT: 0x42,\n+\t/**\u003c Result of client request to open new subchannel */\n+\tCLOSE_NOT: 0x43,\n+\tCLOSE_RQ: 0x44,\n+\t/**\u003c client requests to close a subchannel */\n+\tWRITE: 0x45,\n+\t/**\u003c connection writes something to specific channel index */\n+\tRX: 0x46,\n+};\n+\n+function new_ws(urlpath, protocol)\n+{\n+\tif (typeof MozWebSocket !\u003d \u0022undefined\u0022)\n+\t\treturn new MozWebSocket(urlpath, protocol);\n+\n+\treturn new WebSocket(urlpath, protocol);\n+}\n+\n+function lws_meta_ws() {\n+\tvar real;\n+\t\n+\tvar channel_id_to_child;\n+\tvar pending_children;\n+\tvar active_children;\n+}\n+\n+function lws_meta_ws_child() {\n+\tvar onopen;\n+\tvar onmessage;\n+\tvar onclose;\n+\t\n+\tvar channel_id;\n+\t\n+\tvar subprotocol;\n+\tvar suburl;\n+\tvar cookie;\n+\t\n+\tvar extensions;\n+\t\n+\tvar parent;\n+}\n+\n+lws_meta_ws_child.prototype.send \u003d function(data)\n+{\n+\n+\tif (typeof data \u003d\u003d \u0022string\u0022) {\n+\t\tdata \u003d String.fromCharCode(lws_meta_cmd.WRITE) +\n+\t\t\tString.fromCharCode(this.channel_id) +\n+\t\t\tdata;\n+\t\t\n+\t\treturn this.parent.real.send(data);\n+\t}\n+\t\n+\t{\n+\n+\t\tvar ab \u003d new Uint8Array(data.length + 2);\n+\n+\t\tab[0] \u003d lws_meta_cmd.WRITE;\n+\t\tab[1] \u003d this.channel_id;\n+\t\tab.set(data, 2);\n+\t\n+\t\treturn this.parent.real.send(ab);\n+\t}\n+}\n+\n+lws_meta_ws_child.prototype.close \u003d function(close_code, close_string)\n+{\n+\tvar pkt \u003d new Uint8Array(129), m \u003d 0, pkt1;\n+\t\n+\tpkt[m++] \u003d lws_meta_cmd.CLOSE_RQ;\n+\tpkt[m++] \u003d this.channel_id;\n+\t\n+\tpkt[m++] \u003d close_string.length + 0x20;\n+\t\n+\tpkt[m++] \u003d close_code / 256;\n+\tpkt[m++] \u003d close_code % 256;\n+\t\n+\tfor (i \u003d 0; i \u003c close_string.length; i++)\n+\t\tpkt[m++] \u003d close_string.charCodeAt(i);\n+\t\n+\tpkt1 \u003d new Uint8Array(m);\n+\tfor (n \u003d 0; n \u003c m; n++)\n+\t\tpkt1[n] \u003d pkt[n];\n+\t\t\n+\tthis.parent.real.send(pkt1.buffer);\n+}\n+\n+/* make a real ws connection using lws_meta*/\n+lws_meta_ws.prototype.new_parent \u003d function(urlpath)\n+{\n+\tvar n, i, m \u003d 0, pkt1;\n+\t\n+\tthis.ordinal \u003d 1;\n+\tthis.pending_children \u003d [];\n+\tthis.active_children \u003d [];\n+\tthis.real \u003d new_ws(urlpath, \u0022lws-meta\u0022);\n+\t\n+\tthis.real.binaryType \u003d 'arraybuffer';\n+\tthis.real.myparent \u003d this;\n+\n+\tthis.real.onopen \u003d function() {\n+\t\tpkt \u003d new Uint8Array(1024);\n+\t\t\tvar n, i, m \u003d 0, pkt1;\n+\t\tconsole.log(\u0022real open - pending children \u0022 + this.myparent.pending_children.length);\n+\t\tfor (n \u003d 0; n \u003c this.myparent.pending_children.length; n++) {\n+\t\t\n+\t\t\tvar p \u003d this.myparent.pending_children[n];\n+\t\t\n+\t\t\tpkt[m++] \u003d lws_meta_cmd.OPEN_SUBCHANNEL;\n+\t\t\tfor (i \u003d 0; i \u003c p.subprotocol.length; i++)\n+\t\t\t\tpkt[m++] \u003d p.subprotocol.charCodeAt(i);\n+\t\t\tpkt[m++] \u003d 0;\n+\t\t\tfor (i \u003d 0; i \u003c p.suburl.length; i++)\n+\t\t\t\tpkt[m++] \u003d p.suburl.charCodeAt(i);\n+\t\t\tpkt[m++] \u003d 0;\n+\t\t\tfor (i \u003d 0; i \u003c p.cookie.length; i++)\n+\t\t\t\tpkt[m++] \u003d p.cookie.charCodeAt(i);\n+\t\t\tpkt[m++] \u003d 0;\n+\t\t}\n+\t\t\n+\t\tpkt1 \u003d new Uint8Array(m);\n+\t\tfor (n \u003d 0; n \u003c m; n++)\n+\t\t\tpkt1[n] \u003d pkt[n];\n+\t\t\n+\t\tconsole.log(this.myparent.pending_children[0].subprotocol);\n+\t\tconsole.log(pkt1);\n+\t\t\n+\t\tthis.send(pkt1.buffer);\n+\t}\n+\n+\n+\tthis.real.onmessage \u003d function(msg) {\n+\t\n+\t\tif (typeof msg.data !\u003d \u0022string\u0022) {\n+\t\t\tvar ba \u003d new Uint8Array(msg.data), n \u003d 0;\n+\t\t\t\n+\t\t\twhile (n \u003c ba.length) {\n+\n+\t\t\t\tswitch (ba[n++]) {\n+\t\t\t\tcase lws_meta_cmd.OPEN_RESULT:\n+\t\t\t\t{\n+\t\t\t\t\tvar m \u003d 0, cookie \u003d \u0022\u0022, protocol \u003d \u0022\u0022, ch \u003d 0;\n+\t\t\t\t\tvar ws \u003d this.myparent;\n+\t\t\t\t\t/* cookie NUL\n+\t\t\t\t\t * channel index + 0x20\n+\t\t\t\t\t * protocol NUL\n+\t\t\t\t\t */\n+\t\t\t\t\t while (ba[n])\n+\t\t\t\t\t \tcookie \u003d cookie + String.fromCharCode(ba[n++]);\n+\t\t\t\t\t n++;\n+\t\t\t\t\t ch \u003d ba[n++];\n+\t\t\t\t\t \n+\t\t\t\t\t while (ba[n])\n+\t\t\t\t\t \tprotocol \u003d protocol + String.fromCharCode(ba[n++]);\n+\t\t\t\t\t \t\n+\t\t\t\t\tconsole.log(\u0022open result \u0022 + cookie + \u0022 \u0022 + protocol + \u0022 \u0022 + ch + \u0022 pending len \u0022 + ws.pending_children.length);\n+\t\t\t\t\t\n+\t\t\t\t\tfor (m \u003d 0; m \u003c ws.pending_children.length; m++) {\n+\t\t\t\t\t\tif (ws.pending_children[m].cookie \u003d\u003d cookie) {\n+\t\t\t\t\t\t\tvar newchild \u003d ws.pending_children[m];\n+\t\t\t\n+\t\t\t\t\t\t\t/* found it */\n+\t\t\t\t\t\t\tws.pending_children[m].channel_id \u003d ch;\n+\t\t\t\t\t\t\t/* add to active children array */\n+\t\t\t\t\t\t\tws.active_children.push(ws.pending_children[m]);\n+\t\t\t\t\t\t\t/* remove from pending children array */\n+\t\t\t\t\t\t\tws.pending_children.splice(m, 1);\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\tnewchild.parent \u003d ws;\n+\t\t\t\t\t\t\tnewchild.extensions \u003d this.extensions;\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\tnewchild.onopen();\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\tconsole.log(\u0022made active \u0022 + cookie);\n+\t\t\t\t\t\t\tbreak;\n+\t\t\t\t\t\t}\n+\t\t\t\t\t}\n+\t\t\t\t\tbreak;\n+\t\t\t\t}\n+\t\n+\t\t\t\tcase lws_meta_cmd.CLOSE_NOT:\n+\t\t\t\t{\n+\t\t\t\t\tvar code \u003d 0, str \u003d \u0022\u0022, ch \u003d 0, m, le;\n+\t\t\t\t\tvar ba \u003d new Uint8Array(msg.data);\n+\t\t\t\t\t/*\n+\t\t\t\t\t * BYTE: channel\n+\t\t\t\t\t * BYTE: MSB status code\n+\t\t\t\t\t * BYTE: LSB status code\n+\t\t\t\t\t * BYTES: rest of message is close status string\n+\t\t\t\t\t */\n+\t\t\t\t\t \n+\t\t\t\t\t ch \u003d ba[n++];\n+\t\t\t\t\t le \u003d ba[n++] - 0x20;\n+\t\t\t\t\t code \u003d ba[n++] * 256;\n+\t\t\t\t\t code +\u003d ba[n++];\n+\t\t\t\t\t \n+\t\t\t\t\t while (le--)\n+\t\t\t\t\t \tstr +\u003d String.fromCharCode(ba[n++]);\n+\t\t\t\t\t \t\n+\t\t\t\t\tconsole.log(\u0022channel id \u0022 + ch + \u0022 code \u0022 + code + \u0022 str \u0022 + str + \u0022 len \u0022 + str.length);\n+\t\t\t\t\t \t\n+\t\t\t\t\tfor (m \u003d 0; m \u003c this.myparent.active_children.length; m++)\n+\t\t\t\t\t\tif (this.myparent.active_children[m].channel_id \u003d\u003d ch) {\n+\t\t\t\t\t\t\tvar child \u003d this.myparent.active_children[m];\n+\t\t\t\t\t\t\tvar ms \u003d new CloseEvent(\u0022close\u0022, { code:code, reason:str } );\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\t/* reply with close ack */\n+\t\t\t\t\t\t\tthis.send(msg.data);\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\tif (child.onclose)\n+\t\t\t\t\t\t\t\tchild.onclose(ms);\n+\t\t\t\t\t\t\t\n+\t\t\t\t\t\t\tthis.myparent.active_children.splice(m, 1);\n+\t\t\t\t\t\t\tbreak;\n+\t\t\t\t\t\t}\n+\n+\t\t\t\t}\n+\t\t\t\t} // switch\n+\t\t\t}\n+\t\t} else {\n+\t\t\tif (msg.data.charCodeAt(0) \u003d\u003d lws_meta_cmd.WRITE ) {\n+\t\t\t\tvar ch \u003d msg.data.charCodeAt(1), m, ms;\n+\t\t\t\tvar ws \u003d this.myparent, ms;\n+\t\t\t\t\t\t\t\t\n+\t\t\t\tfor (m \u003d 0; m \u003c ws.active_children.length; m++) {\n+\t\t\t\t\tif (ws.active_children[m].channel_id \u003d\u003d ch) {\n+\t\t\t\t\t\tms \u003d new MessageEvent(\u0022WebSocket\u0022, { data: msg.data.substr(2, msg.data.length - 2) } );\n+\t\t\t\t\t\tif (ws.active_children[m].onmessage)\n+\t\t\t\t\t\t\tws.active_children[m].onmessage(ms);\n+\t\t\t\t\t\tbreak;\n+\t\t\t\t\t}\n+\t\t\t\t}\n+\t\t\t}\n+\t\t}\n+\t}\n+\tthis.real.onclose \u003d function() {\n+\t\tvar ws \u003d this.myparent, m;\n+\t\tfor (m \u003d 0; m \u003c ws.active_children.length; m++) {\n+\t\t\tvar child \u003d ws.active_children[m];\n+\t\t\tvar ms \u003d new CloseEvent(\u0022close\u0022, { code:1000, reason:\u0022parent closed\u0022 } );\n+\t\t\t\n+\t\t\tif (child.onclose)\n+\t\t\t\tchild.onclose(ms);\n+\t\t}\n+\t}\n+\n+}\n+\n+\n+\n+/* make a child connection using existing lws_meta real ws connection */\n+lws_meta_ws.prototype.new_ws \u003d function(suburl, protocol)\n+{\n+\tvar ch \u003d new lws_meta_ws_child();\n+\t\n+\tch.suburl \u003d suburl;\n+\tch.subprotocol \u003d protocol;\n+\tch.cookie \u003d \u0022C\u0022 + this.ordinal++;\n+\t\n+\tthis.pending_children.push(ch);\n+\t\n+\tif (this.real.readyState \u003d\u003d 1)\n+\t\tthis.real.onopen();\n+\t\n+\treturn ch;\n+}\n+\n+\n+/*\n+ * end of lws-meta helpers\n+ */\n+ \n+function lws_san(s)\n+{\n+\tif (s.search(\u0022\u003c\u0022) !\u003d -1)\n+\t\treturn \u0022invalid string\u0022;\n+\t\n+\treturn s;\n+}\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/test.html b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/test.html\nnew file mode 100644\nindex 0000000..91c6dc2\n--- /dev/null\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/test.html\n@@ -0,0 +1,858 @@\n+\u003c!DOCTYPE html\u003e\n+\u003chtml lang\u003d\u0022en\u0022\u003e\n+\u003chead\u003e\n+ \u003cmeta charset\u003dutf-8 http-equiv\u003d\u0022Content-Language\u0022 content\u003d\u0022en\u0022/\u003e\n+ \u003cscript src\u003d\u0022/lws-common.js\u0022\u003e\u003c/script\u003e\n+ \u003ctitle\u003eMinimal Websocket test app\u003c/title\u003e\n+\u003cstyle type\u003d\u0022text/css\u0022\u003e\n+\tspan.title { font-size:18pt; font-family: Arial; font-weight:normal; text-align:center; color:#000000; }\n+\t.browser { font-size:12pt; font-family: Arial; font-weight:normal; text-align:center; color:#ffff00; vertical-align:middle; text-align:center; background:#d0b070; padding:12px; -webkit-border-radius:10px; border-radius:10px;}\n+\t.group2 { vertical-align:middle;\n+\t\ttext-align:center;\n+\t\tbackground:#f0f0e0; \n+\t\tpadding:12px; \n+\t\t-webkit-border-radius:10px; \n+\t\tborder-radius:10px; }\n+\t.explain { vertical-align:middle;\n+\t\ttext-align:center;\n+\t\tbackground:#f0f0c0; padding:12px;\n+\t\t-webkit-border-radius:10px;\n+\t\tborder-radius:10px;\n+\t\tcolor:#404000; }\n+\ttd.wsstatus { vertical-align:middle; width:200px; height:50px;\n+\t\ttext-align:center;\n+\t\tbackground:#f0f0c0; padding:6px;\n+\t\t-webkit-border-radius:8px;\n+\t\tborder-radius:8px;\n+\t\tcolor:#404000; }\n+\t.tdform { vertical-align:middle; width:200px; height:50px;\n+\t\ttext-align:center;\n+\t\tbackground:#f0f0d0; padding:6px;\n+\t\t-webkit-border-radius:8px;\n+\t\tmargin:10px;\n+\t\tborder-radius:8px;\n+\t\tborder: 1px solid black;\n+\t\tborder-collapse: collapse;font-size:18pt; font-family: Arial; font-weight:normal; text-align:center; color:#000000; \n+\t\tcolor:#404000; }\n+\t\t\n+\ttd.l { vertical-align:middle;\n+\t\ttext-align:center;\n+\t\tbackground:#d0d0b0; \n+\t\tpadding:3px; \n+\t\t-webkit-border-radius:3px;\n+\t\tborder-radius:3px; }\n+\t.content { vertical-align:top; text-align:center; background:#fffff0; padding:12px; -webkit-border-radius:10px; border-radius:10px; }\n+\t.canvas { vertical-align:top; text-align:center; background:#efefd0; padding:12px; -webkit-border-radius:10px; border-radius:10px; }\n+.tabs {\n+ position: relative; \n+ min-height: 750px; /* This part sucks */\n+ clear: both;\n+ margin: 25px 0;\n+}\n+.tab {\n+ float: left;\n+}\n+.tab label {\n+ background: #eee; \n+ padding: 10px; \n+ border: 1px solid #ccc; \n+ margin-left: -1px; \n+ position: relative;\n+ left: 1px; \n+}\n+.tab [type\u003dradio] {\n+ display: none; \n+}\n+.content {\n+ position: absolute;\n+ top: 28px;\n+ left: 0;\n+ background: white;\n+ right: 0;\n+ bottom: 0;\n+ padding: 20px;\n+ border: 1px solid #ccc; \n+}\n+[type\u003dradio]:checked ~ label {\n+ background: white;\n+ border-bottom: 1px solid white;\n+ z-index: 2;\n+}\n+[type\u003dradio]:checked ~ label ~ .content {\n+ z-index: 1;\n+}\n+\u003c/style\u003e\n+\u003c/head\u003e\n+\n+\u003cbody\u003e\n+\u003cheader\u003e\u003c/header\u003e\n+\u003carticle\u003e\n+\n+\u003ctable\u003e\u003ctr\u003e\u003ctd\u003e\n+\n+\u003ctable width\u003d600px\u003e\n+ \u003ctr\u003e\n+ \u003ctd valign\u003dmiddle align\u003dcenter\u003e\n+ \u003ca href\u003d\u0022https://libwebsockets.org\u0022\u003e\n+ \u003cimg src\u003d\u0022libwebsockets.org-logo.png\u0022\u003e\u003c/a\u003e\u003c/td\u003e\u003ctd\u003e\n+\t\u003csection class\u003d\u0022browser\u0022\u003e\n+\t\u003cdiv id\u003dbrow\u003e...\u003c/div\u003e\u003c/section\u003e\n+ \u003c/td\u003e\n+ \u003ctd width\u003d\u002264\u0022 height\u003d\u002264\u0022 id\u003d\u0022wstransport\u0022\u003e\u003c/td\u003e\n+ \u003ctd width\u003d\u002264\u0022 height\u003d\u002264\u0022 id\u003d\u0022transport\u0022\u003e\u003c/td\u003e\n+ \u003c/tr\u003e\n+\n+\u003c/table\u003e\n+\u003c/td\u003e\u003c/tr\u003e\n+\u003ctr\u003e\u003ctd colspan\u003d2 align\u003dcenter\u003e\n+Click \u003ca href\u003d\u0022leaf.jpg\u0022 target\u003d\u0022_blank\u0022\u003eHere\u003c/a\u003e to\n+have the test server send a big picture by http.\n+\u003c/td\u003e\u003c/tr\u003e\n+\u003ctr\u003e\u003ctd colspan\u003d2\u003e\n+\u003cdiv class\u003d\u0022tabs\u0022\u003e\n+\n+ \u003cdiv class\u003d\u0022tab\u0022\u003e\n+ \u003cinput type\u003d\u0022radio\u0022 id\u003d\u0022tab-1\u0022 name\u003d\u0022tab-group-1\u0022 checked\u003e\n+ \u003clabel for\u003d\u0022tab-1\u0022\u003eDumb Increment Demo\u003c/label\u003e\n+ \n+ \u003cdiv class\u003d\u0022content\u0022\u003e\n+ \u003cdiv id\u003d\u0022dumb\u0022 class\u003d\u0022group2\u0022\u003e\n+ \u003ctable\u003e\n+ \u003ctr\u003e\n+\t \u003ctd id\u003dwsdi_statustd align\u003dcenter class\u003d\u0022wsstatus\u0022\u003e\n+\t \u003cspan id\u003dwsdi_status\u003eWebsocket connection not initialized\u003c/span\u003e\u003c/td\u003e\n+ \u003ctd\u003e\u003cspan class\u003d\u0022title\u0022\u003edumb increment-protocol\u003c/span\u003e\u003c/td\u003e\n+\t \u003c/tr\u003e\n+\t \u003ctr\u003e\n+\t \u003ctd class\u003d\u0022explain\u0022 colspan\u003d2\u003e\n+The incrementing number is coming from the server at 20Hz and is individual for\n+each connection to the server... try opening a second browser window.\n+\u003cbr/\u003e\u003cbr/\u003e\n+The button sends a message over the websocket link to ask the server\n+to zero just this connection's number.\n+\t \u003c/td\u003e\n+\t \u003c/tr\u003e\n+\t \u003ctr\u003e\n+\t \u003ctd align\u003dcenter\u003e\u003cdiv id\u003dnumber style\u003d\u0022font-size:120%;\u0022\u003e \u003c/div\u003e\u003c/td\u003e\n+\t \u003ctd align\u003dcenter\u003e\n+\t \u003cinput type\u003dbutton id\u003doffset value\u003d\u0022Reset counter\u0022\u003e\n+\t \u003cinput type\u003dbutton id\u003djunk value\u003d\u0022Send junk\u0022\u003e\n+\t \u003c/td\u003e\n+\t \u003c/tr\u003e\n+\t \u003c/table\u003e\n+\t\u003c/div\u003e\n+ \u003c/div\u003e \n+ \u003c/div\u003e\n+\n+ \u003cdiv class\u003d\u0022tab\u0022\u003e\n+ \u003cinput type\u003d\u0022radio\u0022 id\u003d\u0022tab-2\u0022 name\u003d\u0022tab-group-1\u0022\u003e\n+ \u003clabel for\u003d\u0022tab-2\u0022\u003eMirror Demo\u003c/label\u003e\n+ \n+ \u003cdiv class\u003d\u0022content\u0022\u003e\n+ \u003cdiv id\u003d\u0022mirror\u0022 class\u003d\u0022group2\u0022\u003e\n+ \u003ctable\u003e\n+ \u003ctr\u003e\n+\t \u003ctd colspan\u003d1 id\u003dwslm_statustd align\u003dcenter class\u003d\u0022wsstatus\u0022\u003e\n+\t \u003cspan id\u003dwslm_status\u003eWebsocket connection not initialized\u003c/span\u003e\n+\t\u003c/td\u003e\n+ \u003ctd\u003e\n+ \u003cspan class\u003d\u0022title\u0022\u003elws-mirror-protocol\u003c/span\u003e\n+ \u003c/td\u003e\n+ \u003c/tr\u003e\n+ \u003ctr\u003e\n+ \u003ctd colspan\u003d2\u003e\n+ \u003cdiv class\u003d\u0022explain\u0022\u003e\n+Use the mouse to draw on the canvas below -- all other browser windows open\n+on this page see your drawing in realtime and you can see any of theirs as\n+well.\n+\u003cbr/\u003e\u003cbr/\u003e\n+The lws-mirror protocol doesn't interpret what is being sent to it, it just\n+re-sends it to every other websocket it has a connection with using that\n+protocol, including the guy who sent the packet.\n+\u003cbr/\u003e\u003cbr/\u003e\n+\u003cb\u003elibwebsockets-test-client\u003c/b\u003e joins in by spamming circles on to this shared canvas when\n+run.\n+ \u003c/div\u003e\n+ \u003c/td\u003e\n+ \u003c/tr\u003e\n+ \u003ctr\u003e\n+\t\u003ctd colspan\u003d2\u003eDrawing color:\n+\t \u003cselect id\u003d\u0022color\u0022\u003e\n+\t\t\u003coption value\u003d#000000\u003eBlack\u003c/option\u003e\n+\t\t\u003coption value\u003d#0000ff\u003eBlue\u003c/option\u003e\n+\t\t\u003coption value\u003d#20ff20\u003eGreen\u003c/option\u003e\n+\t\t\u003coption value\u003d#802020\u003eDark Red\u003c/option\u003e\n+\t \u003c/select\u003e\n+ \u003c/tr\u003e\n+ \u003ctr\u003e\n+\t \u003ctd colspan\u003d2 width\u003d500 height\u003d320\u003e\n+\t\t\u003cdiv id\u003d\u0022wslm_drawing\u0022 style\u003d\u0022background:white\u0022\u003e\u003c/div\u003e\n+\t\u003c/td\u003e\n+ \u003c/tr\u003e\n+ \u003c/table\u003e\n+ \u003c/div\u003e\n+ \u003c/div\u003e \n+ \u003c/div\u003e\n+ \n+ \u003cdiv class\u003d\u0022tab\u0022\u003e\n+ \u003cinput type\u003d\u0022radio\u0022 id\u003d\u0022tab-3\u0022 name\u003d\u0022tab-group-1\u0022\u003e\n+ \u003clabel for\u003d\u0022tab-3\u0022\u003eClose Testing\u003c/label\u003e\n+ \n+ \u003cdiv class\u003d\u0022content\u0022\u003e\n+\u003cdiv id\u003d\u0022ot\u0022 class\u003d\u0022group2\u0022\u003e\n+ \u003ctable\u003e\n+ \u003ctr\u003e\n+ \u003ctd\u003e\n+\n+\t\t\u003c/td\u003e\u003c/tr\u003e\n+\t\t\u003ctr\u003e\u003ctd id\u003dot_statustd align\u003dcenter class\u003d\u0022wsstatus\u0022\u003e\n+\t\t \u003cspan id\u003dot_status\u003eWebsocket connection not initialized\u003c/span\u003e\n+\t\t\u003c/td\u003e\n+\t\t\u003ctd colspan\u003d2\u003e\u003cspan class\u003d\u0022title\u0022\u003eOpen and close testing\u003c/span\u003e\u003c/td\u003e\n+\t\t\u003c/tr\u003e\n+\t\t\u003ctr\u003e\t\n+\u003ctd class\u003d\u0022explain\u0022 colspan\u003d3 style\u003d\u0022padding:3px\u0022\u003e\n+To help with open and close testing, you can open and close a connection by hand using\n+ the buttons.\u003cbr\u003e\n+ \u0022\u003cb\u003eClose\u003c/b\u003e\u0022 closes the connection from the browser with code 3000\n+ and reason 'Bye!\u0022.\u003cbr\u003e\n+ \u0022\u003cb\u003eRequest Server Close\u003c/b\u003e\u0022 sends a message asking the server to\n+initiate the close, which it does with code 1001 and reason \u0022Seeya\u0022.\n+\u003c/td\u003e\u003c/tr\u003e\n+\t\t\u003ctr\u003e\n+\t\t\u003ctd align\u003dcenter\u003e\u003cinput type\u003dbutton id\u003dot_open_btn value\u003d\u0022Open\u0022\u003e\u003c/td\u003e\n+\t\t\u003ctd align\u003dcenter\u003e\u003cinput type\u003dbutton id\u003dot_close_btn disabled value\u003d\u0022Close\u0022 \u003e\u003c/td\u003e\n+\t\t\u003ctd align\u003dcenter\u003e\u003cinput type\u003dbutton id\u003dot_req_close_btn disabled value\u003d\u0022Request Server Close\u0022 \u003e\u003c/td\u003e\n+\t\t\u003c/tr\u003e\n+\n+\u003c/table\u003e\n+\n+\u003c/div\u003e\n+ \u003c/div\u003e \n+ \u003c/div\u003e\n+ \n+ \u003cdiv class\u003d\u0022tab\u0022\u003e\n+ \u003cinput type\u003d\u0022radio\u0022 id\u003d\u0022tab-4\u0022 name\u003d\u0022tab-group-1\u0022\u003e\n+ \u003clabel for\u003d\u0022tab-4\u0022\u003eServer info\u003c/label\u003e\n+\n+ \u003cdiv class\u003d\u0022content\u0022\u003e\n+\u003cdiv id\u003d\u0022ot\u0022 class\u003d\u0022group2\u0022\u003e\n+ \u003ctable\u003e\n+ \u003ctr\u003e\n+\t\u003ctd id\u003ds_statustd align\u003dcenter class\u003d\u0022wsstatus\u0022\u003e\n+\t \u003cdiv id\u003ds_status\u003eWebsocket connection not initialized\u003c/div\u003e\n+\t\u003c/td\u003e\n+\t \u003ctd colspan\u003d1\u003e\n+\u003cspan class\u003d\u0022title\u0022\u003eServer Info\u003c/span\u003e\t \u003cinput type\u003dbutton id\u003dpmd value\u003d\u0022Test pmd\u0022\u003e\n+\n+\t\u003c/td\u003e\n+\t\u003c/tr\u003e\u003ctr\u003e\n+\u003ctd class\u003d\u0022explain\u0022 colspan\u003d2\u003e\n+This information is sent by the server over a ws[s] link and updated live\n+whenever the information changes server-side.\n+\u003c/td\u003e\u003c/tr\u003e\n+\t\u003ctr\u003e\n+\t\u003ctd align\u003dcenter colspan\u003d2\u003e\u003cdiv id\u003dservinfo\u003e\u003c/div\u003e\u003c/td\u003e\n+\t\u003c/tr\u003e\n+\t\u003ctr\u003e\n+\t\u003ctd align\u003dcenter colspan\u003d2\u003e\u003cdiv id\u003dconninfo style\u003d\u0022border: solid 2px #e0d040; padding: 4px; width: 500px; height:350px; overflow: auto;\u0022\u003e\u003c/div\u003e\u003c/td\u003e\n+\t\u003c/tr\u003e\n+\u003c/table\u003e\n+\u003c/div\u003e\n+ \u003c/div\u003e \n+ \u003c/div\u003e\n+\n+ \u003cdiv class\u003d\u0022tab\u0022\u003e\n+ \u003cinput type\u003d\u0022radio\u0022 id\u003d\u0022tab-5\u0022 name\u003d\u0022tab-group-1\u0022\u003e\n+ \u003clabel for\u003d\u0022tab-5\u0022\u003ePOST\u003c/label\u003e\n+\n+ \u003cdiv class\u003d\u0022content\u0022\u003e\n+\u003cdiv id\u003d\u0022ot\u0022 class\u003d\u0022group2\u0022\u003e\n+ \u003ctable width\u003d100%\u003e\n+ \u003ctr\u003e\n+\t \u003ctd colspan\u003d1\u003e\n+\u003cspan class\u003d\u0022title\u0022\u003ePOST Form testing\u003c/span\u003e\n+\t\u003c/td\u003e\n+\t\u003c/tr\u003e\u003ctr\u003e\n+\u003ctd class\u003d\u0022explain\u0022 colspan\u003d2\u003e\n+This tests POST handling in lws.\n+\u003c/td\u003e\u003c/tr\u003e\n+\t\u003ctr\u003e\n+\t\u003ctd align\u003dcenter colspan\u003d2 class\u003dtdform\u003e\u003cdiv id\u003dpostinfo style\u003dform\u003e\n+\tFORM 1: send with urlencoded POST body args\u003cbr\u003e\n+\t\u003cform action\u003d\u0022formtest\u0022 method\u003d\u0022post\u0022\u003e\n+ \u003cspan style\u003d\u0022font-size:12pt;\u0022\u003eSome text: \u003c/span\u003e\n+ \u003cinput type\u003d\u0022text\u0022 name\u003d\u0022text\u0022 value\u003d\u0022Give me some text\u0022\u003e\u003cbr\u003e\n+ \u003cinput type\u003d\u0022submit\u0022 name\u003d\u0022send\u0022 value\u003d\u0022Send the form\u0022\u003e\n+\t\u003c/form\u003e\n+\t\u003c/div\u003e\u003c/td\u003e\n+\t\u003c/tr\u003e\n+\n+\u003cscript nonce\u003d\u0022lwscaro\u0022\u003e\n+function check_file()\n+{\n+\tvar f \u003d document.getElementById('file').files[0];\n+\tvar max_len \u003d 100000;\n+\tvar dis \u003d 0;\n+\t\n+\tif (f) {\n+\t\tif (f.size \u003e\u003d max_len) {\n+\t\t\tdis \u003d 1;\n+\t\t\tdocument.getElementById('file_info').innerHTML \u003d\n+\t\t\t\t\u0022\u003cspan style\u003d\u005c\u0022color:red;font-weight:bold\u005c\u0022\u003eFile larger than \u0022+max_len+\u0022\u003c/span\u003e\u0022;\n+\t\t} else\n+\t\t\tdocument.getElementById('file_info').innerHTML \u003d\n+\t\t\t\t\u0022File length \u0022+f.size;\n+\t} else\n+\t\tdis \u003d 1;\n+\t\n+\tdocument.getElementById('upload').disabled \u003d dis;\n+}\n+\u003c/script\u003e\n+\n+\t\u003ctr\u003e\n+\t\u003ctd align\u003dcenter colspan\u003d2 class\u003dtdform\u003e\u003cdiv id\u003dpostinfo style\u003dform\u003e\n+\tFORM 2: send with multipart/form-data\u003cbr\u003e\n+\t(can handle file upload, test limited to 100KB)\u003cbr\u003e\n+\t\u003cform name\u003dmultipart action\u003d\u0022formtest\u0022 method\u003d\u0022post\u0022 enctype\u003d\u0022multipart/form-data\u0022\u003e\n+ \u003cspan style\u003d\u0022font-size:12pt;\u0022\u003eSome text: \u003c/span\u003e\n+ \u003cinput type\u003d\u0022text\u0022 name\u003d\u0022text\u0022 value\u003d\u0022Give me some text\u0022\u003e\n+\u003cbr\u003e\n+ \u003cinput type\u003d\u0022file\u0022 name\u003d\u0022file\u0022 id\u003d\u0022file\u0022 size\u003d\u002220\u0022\u003e\u0026nbsp;\u003cspan id\u003dfile_info style\u003d\u0022font-size:12pt;\u0022\u003e\u003c/span\u003e\u003cbr\u003e\n+ \u003cinput type\u003d\u0022submit\u0022 id\u003d\u0022upload\u0022 name\u003d\u0022upload\u0022 disabled\u003d1 value\u003d\u0022Upload\u0022\u003e\n+\t\u003c/form\u003e\n+\t\u003c/div\u003e\u003c/td\u003e\n+\t\u003c/tr\u003e\n+\t\n+\u003c/table\u003e\n+\u003c/div\u003e\n+ \u003c/div\u003e \n+ \u003c/div\u003e\n+\n+\u003c/div\u003e\n+\u003c/td\u003e\u003c/tr\u003e\u003c/table\u003e\n+\n+Looking for support? \u003ca href\u003d\u0022https://libwebsockets.org\u0022\u003ehttps://libwebsockets.org\u003c/a\u003e, \u003ca href\u003d\u0022https://github.com/warmcat/libwebsockets\u0022\u003ehttps://github.com/warmcat/libwebsockets\u003c/a\u003e\u003c/a\u003e\u003cbr/\u003e\n+Join the mailing list: \u003ca href\u003d\u0022https://libwebsockets.org/mailman/listinfo/libwebsockets\u0022\u003ehttps://libwebsockets.org/mailman/listinfo/libwebsockets\u003c/a\u003e\n+\n+\u003c/article\u003e\n+\n+\u003cscript nonce\u003d\u0022lwscaro\u0022\u003e\n+\n+document.getElementById('file').onchange \u003d check_file;\n+document.getElementById('offset').onclick \u003d reset;\n+document.getElementById('junk').onclick \u003d junk;\n+document.getElementById('color').onclick \u003d update_color;\n+document.getElementById('ot_open_btn').onclick \u003d ot_open;\n+document.getElementById('ot_close_btn').onclick \u003d ot_close;\n+document.getElementById('ot_req_close_btn').onclick \u003d ot_req_close;\n+document.getElementById('pmd').onclick \u003d on_pmd;\n+\n+/*\n+ * We display untrusted stuff in html context... reject anything\n+ * that has HTML stuff in it\n+ */\n+\n+function san(s)\n+{\n+\tif (s.search(\u0022\u003c\u0022) !\u003d -1)\n+\t\treturn \u0022invalid string\u0022;\n+\t\n+\treturn s;\n+}\n+\n+lws_gray_out(true,{'zindex':'499'});\n+\n+/* BrowserDetect came from http://www.quirksmode.org/js/detect.html */\n+\n+var BrowserDetect \u003d {\n+\tinit: function () {\n+\t\tthis.browser \u003d this.searchString(this.dataBrowser) || \u0022An unknown browser\u0022;\n+\t\tthis.version \u003d this.searchVersion(navigator.userAgent)\n+\t\t\t|| this.searchVersion(navigator.appVersion)\n+\t\t\t|| \u0022an unknown version\u0022;\n+\t\tthis.OS \u003d this.searchString(this.dataOS) || \u0022an unknown OS\u0022;\n+\t},\n+\tsearchString: function (data) {\n+\t\tfor (var i\u003d0;i\u003cdata.length;i++)\t{\n+\t\t\tvar dataString \u003d data[i].string;\n+\t\t\tvar dataProp \u003d data[i].prop;\n+\t\t\tthis.versionSearchString \u003d data[i].versionSearch || data[i].identity;\n+\t\t\tif (dataString) {\n+\t\t\t\tif (dataString.indexOf(data[i].subString) !\u003d -1)\n+\t\t\t\t\treturn data[i].identity;\n+\t\t\t}\n+\t\t\telse if (dataProp)\n+\t\t\t\treturn data[i].identity;\n+\t\t}\n+\t},\n+\tsearchVersion: function (dataString) {\n+\t\tvar index \u003d dataString.indexOf(this.versionSearchString);\n+\t\tif (index \u003d\u003d -1) return;\n+\t\treturn parseFloat(dataString.substring(index+this.versionSearchString.length+1));\n+\t},\n+\tdataBrowser: [\n+\t\t{\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022Chrome\u0022,\n+\t\t\tidentity: \u0022Chrome\u0022\n+\t\t},\n+\t\t{ \tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022OmniWeb\u0022,\n+\t\t\tversionSearch: \u0022OmniWeb/\u0022,\n+\t\t\tidentity: \u0022OmniWeb\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.vendor,\n+\t\t\tsubString: \u0022Apple\u0022,\n+\t\t\tidentity: \u0022Safari\u0022,\n+\t\t\tversionSearch: \u0022Version\u0022\n+\t\t},\n+\t\t{\n+\t\t\tprop: window.opera,\n+\t\t\tidentity: \u0022Opera\u0022,\n+\t\t\tversionSearch: \u0022Version\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.vendor,\n+\t\t\tsubString: \u0022iCab\u0022,\n+\t\t\tidentity: \u0022iCab\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.vendor,\n+\t\t\tsubString: \u0022KDE\u0022,\n+\t\t\tidentity: \u0022Konqueror\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022Firefox\u0022,\n+\t\t\tidentity: \u0022Firefox\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.vendor,\n+\t\t\tsubString: \u0022Camino\u0022,\n+\t\t\tidentity: \u0022Camino\u0022\n+\t\t},\n+\t\t{\t\t// for newer Netscapes (6+)\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022Netscape\u0022,\n+\t\t\tidentity: \u0022Netscape\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022MSIE\u0022,\n+\t\t\tidentity: \u0022Explorer\u0022,\n+\t\t\tversionSearch: \u0022MSIE\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022Gecko\u0022,\n+\t\t\tidentity: \u0022Mozilla\u0022,\n+\t\t\tversionSearch: \u0022rv\u0022\n+\t\t},\n+\t\t{ \t\t// for older Netscapes (4-)\n+\t\t\tstring: navigator.userAgent,\n+\t\t\tsubString: \u0022Mozilla\u0022,\n+\t\t\tidentity: \u0022Netscape\u0022,\n+\t\t\tversionSearch: \u0022Mozilla\u0022\n+\t\t}\n+\t],\n+\tdataOS : [\n+\t\t{\n+\t\t\tstring: navigator.platform,\n+\t\t\tsubString: \u0022Win\u0022,\n+\t\t\tidentity: \u0022Windows\u0022\n+\t\t},\n+\t\t{\n+\t\t\tstring: navigator.platform,\n+\t\t\tsubString: \u0022Mac\u0022,\n+\t\t\tidentity: \u0022Mac\u0022\n+\t\t},\n+\t\t{\n+\t\t\t string: navigator.userAgent,\n+\t\t\t subString: \u0022iPhone\u0022,\n+\t\t\t identity: \u0022iPhone/iPod\u0022\n+\t },\n+\t\t{\n+\t\t\tstring: navigator.platform,\n+\t\t\tsubString: \u0022Linux\u0022,\n+\t\t\tidentity: \u0022Linux\u0022\n+\t\t}\n+\t]\n+\n+};\n+BrowserDetect.init();\n+\n+document.getElementById(\u0022brow\u0022).textContent \u003d \u0022 \u0022 + BrowserDetect.browser + \u0022 \u0022\n+\t+ BrowserDetect.version +\u0022 \u0022 + BrowserDetect.OS +\u0022 \u0022;\n+\n+\tvar pos \u003d 0;\n+\n+function get_appropriate_ws_url(extra_url)\n+{\n+\tvar pcol;\n+\tvar u \u003d document.URL;\n+\n+\t/*\n+\t * We open the websocket encrypted if this page came on an\n+\t * https:// url itself, otherwise unencrypted\n+\t */\n+\n+\tif (u.substring(0, 5) \u003d\u003d \u0022https\u0022) {\n+\t\tpcol \u003d \u0022wss://\u0022;\n+\t\tu \u003d u.substr(8);\n+\t} else {\n+\t\tpcol \u003d \u0022ws://\u0022;\n+\t\tif (u.substring(0, 4) \u003d\u003d \u0022http\u0022)\n+\t\t\tu \u003d u.substr(7);\n+\t}\n+\n+\tu \u003d u.split('/');\n+\n+\t/* + \u0022/xxx\u0022 bit is for IE10 workaround */\n+\n+\treturn pcol + u[0] + \u0022/\u0022 + extra_url;\n+}\n+\n+var params \u003d {};\n+\n+if (location.search) {\n+ var parts \u003d location.search.substring(1).split('\u0026');\n+\n+ for (var i \u003d 0; i \u003c parts.length; i++) {\n+ var nv \u003d parts[i].split('\u003d');\n+ if (!nv[0]) continue;\n+ params[nv[0]] \u003d nv[1] || true;\n+ }\n+}\n+window.onload \u003d function() {\n+var transport_protocol \u003d \u0022\u0022;\n+\n+if ( performance \u0026\u0026 performance.timing.nextHopProtocol ) {\n+ transport_protocol \u003d performance.timing.nextHopProtocol;\n+} else if ( window.chrome \u0026\u0026 window.chrome.loadTimes ) {\n+ transport_protocol \u003d window.chrome.loadTimes().connectionInfo;\n+} else {\n+\n+ var p \u003d performance.getEntriesByType(\u0022resource\u0022);\n+ for (var i\u003d0; i \u003c p.length; i++) {\n+var value \u003d \u0022nextHopProtocol\u0022 in p[i];\n+ if (value)\n+ transport_protocol \u003d p[i].nextHopProtocol;\n+ }\n+ }\n+ \n+ console.log(\u0022transport protocol \u0022 + transport_protocol);\n+ \n+ if (transport_protocol \u003d\u003d \u0022h2\u0022)\n+ \tdocument.getElementById(\u0022transport\u0022).innerHTML \u003d \u0022\u003cimg src\u003d\u005c\u0022./http2.png\u005c\u0022\u003e\u0022;\n+}\n+\n+var mirror_name \u003d \u0022\u0022;\n+if (params.mirror)\n+\tmirror_name \u003d params.mirror;\n+\n+\tconsole.log(mirror_name);\n+\t\n+\n+/*\n+ * if using lws-meta to carry the other ws connections, declare the\n+ * parent connection object and start its connection to the server.\n+ *\n+ * These helpers are defined in lws-common.js\n+ */\n+\n+var use_lws_meta \u003d 0, lws_meta;\n+\n+if (use_lws_meta) {\n+\tlws_meta \u003d new lws_meta_ws();\n+\tlws_meta.new_parent(get_appropriate_ws_url(\u0022?mirror\u003d\u0022 + mirror_name));\n+}\n+\n+\n+document.getElementById(\u0022number\u0022).textContent \u003d get_appropriate_ws_url(mirror_name);\n+\n+/* dumb increment protocol */\n+\n+\t/*\n+\t * to connect via an lws-meta connection, start the connection using\n+\t * lws_meta.new_ws(). To connect by independent connection, start\n+\t * the connection using just new_ws()\n+\t */\n+\t\n+\tvar socket_di;\n+\t\n+\tif (use_lws_meta)\n+\t\tsocket_di \u003d lws_meta.new_ws(\u0022\u0022, \u0022dumb-increment-protocol\u0022);\n+\telse\n+\t\tsocket_di \u003d new_ws(get_appropriate_ws_url(\u0022\u0022), \u0022dumb-increment-protocol\u0022);\n+\n+\ttry {\n+\t\tsocket_di.onopen \u003d function() {\n+\t\t\tdocument.getElementById(\u0022wsdi_statustd\u0022).style.backgroundColor \u003d \u0022#40ff40\u0022;\n+\t\t\tdocument.getElementById(\u0022wsdi_status\u0022).innerHTML \u003d\n+\t\t\t\t\u0022 \u003cb\u003ewebsocket connection opened\u003c/b\u003e\u003cbr\u003e\u0022 +\n+\t\t\t\tsan(socket_di.extensions);\n+\t\t} \n+\n+\t\tsocket_di.onmessage \u003dfunction got_packet(msg) {\n+\t\t\tdocument.getElementById(\u0022number\u0022).textContent \u003d msg.data + \u0022\u005cn\u0022;\n+\t\t} \n+\n+\t\tsocket_di.onclose \u003d function(){\n+\t\t\tdocument.getElementById(\u0022wsdi_statustd\u0022).style.backgroundColor \u003d \u0022#ff4040\u0022;\n+\t\t\tdocument.getElementById(\u0022wsdi_status\u0022).textContent \u003d \u0022 websocket connection CLOSED \u0022;\n+\t\t}\n+\t} catch(exception) {\n+\t\talert('\u003cp\u003eError' + exception); \n+\t}\n+\t\n+\tvar socket_status, jso, s;\n+\n+\tif (use_lws_meta)\n+\t\tsocket_status \u003d lws_meta.new_ws(\u0022\u0022, \u0022lws-status\u0022);\n+\telse\n+\t\tsocket_status \u003d new_ws(get_appropriate_ws_url(\u0022\u0022), \u0022lws-status\u0022);\n+\n+\ttry {\n+\t\tsocket_status.onopen \u003d function() {\n+\t\t\tdocument.getElementById(\u0022s_statustd\u0022).style.backgroundColor \u003d \u0022#40ff40\u0022;\n+\t\t\tdocument.getElementById(\u0022s_status\u0022).innerHTML \u003d\n+\t\t\t\t\u0022 \u003cb\u003ewebsocket connection opened\u003c/b\u003e\u003cbr\u003e\u0022 +\n+\t\t\t\tsan(socket_status.extensions);\n+\t\t} \n+\n+\t\tsocket_status.onmessage \u003dfunction got_packet(msg) {\n+\t\t\tvar s;\n+\t\t\t\n+\t\t\tconsole.log(msg.data);\n+\t\t\t\n+\t\t\tjso \u003d JSON.parse(msg.data);\n+\t\t\t\n+\t\t\tif (jso.wss_over_h2 \u003d\u003d \u00221\u0022)\n+\t\t\t\tdocument.getElementById(\u0022wstransport\u0022).innerHTML \u003d \u0022\u003cimg src\u003d\u005c\u0022./wss-over-h2.png\u005c\u0022\u003e\u0022;\n+\t\t\t\n+\t\t\tdocument.getElementById(\u0022servinfo\u0022).innerHTML \u003d \n+\t\t\t\t\u0022\u003ctable\u003e\u003ctr\u003e\u003ctd class\u003dl\u003eBuild info\u003c/td\u003e\u003ctd\u003e\u0022+\n+\t\t\t\t\tsan(jso.version) + \u0022\u003c/td\u003e\u003c/tr\u003e\u0022 +\n+\t\t\t\t\t\u0022\u003ctr\u003e\u003ctd class\u003dl\u003eServer info\u003c/td\u003e\u003ctd\u003e\u0022 +\n+\t\t\t\t\tsan(jso.hostname) + \u0022\u003c/td\u003e\u003c/tr\u003e\u0022 +\n+\t\t\t\t\t\u0022\u003c/table\u003e\u0022;\n+\t\t\ts\u003d\u0022\u003ctable\u003e\u0022;\n+\t\t\tvar n;\n+\t\t\tfor (n \u003d 0; n \u003c jso.conns.length; n++) {\n+\t\t\t\tvar d \u003d new Date(parseInt(jso.conns[n].time) * 1000);\n+\t\t\t\t\n+\t\t\t\ts \u003d s + \u0022\u003ctr\u003e\u003ctd class\u003dl\u003eclient \u0022 + (n + 1) +\n+\t\t\t\t\u0022\u003c/td\u003e\u003ctd\u003e\u003cb\u003e\u0022 + san(jso.conns[n].peer) +\n+\t\t\t\t\u0022\u003c/b\u003e\u003cbr\u003e\u0022 + san(d.toString()) +\n+\t\t\t\t\u0022\u003cbr\u003e\u0022 + san(jso.conns[n].ua) +\n+\t\t\t\t\u0022\u003c/td\u003e\u003c/tr\u003e\u0022;\n+\t\t\t}\n+\t\t\ts \u003d s + \u0022\u003c/table\u003e\u0022;\n+\t\t\t\n+\t\t\tdocument.getElementById(\u0022conninfo\u0022).innerHTML \u003d s;\n+\t\t} \n+\n+\t\tsocket_status.onclose \u003d function(){\n+\t\t\tdocument.getElementById(\u0022s_statustd\u0022).style.backgroundColor \u003d \u0022#ff4040\u0022;\n+\t\t\tdocument.getElementById(\u0022s_status\u0022).textContent \u003d \u0022 websocket connection CLOSED \u0022;\n+\t\t}\n+\t} catch(exception) {\n+\t\talert('\u003cp\u003eError' + exception); \n+\t}\n+\n+function reset() {\n+\tsocket_di.send(\u0022reset\u005cn\u0022);\n+}\n+\n+\n+function junk() {\n+\tfor(var word \u003d ''; word.length \u003c 9000; word +\u003d 'a'){}\n+\tsocket_di.send(word);\n+}\n+\n+function on_pmd() {\n+\tsocket_status.send(\u0022{ \u005c\u0022RequestType\u005c\u0022:\u005c\u0022DDoS\u005c\u0022, \u005c\u0022blob\u005c\u0022:\u005c\u0022\u005c\u0022 }\u0022);\n+\tsocket_status.send(\u0022{ \u005c\u0022RequestType\u005c\u0022:\u005c\u0022SendImage\u005c\u0022, \u005c\u0022RequestID\u005c\u0022:\u005c\u0022283463389\u005c\u0022, \u005c\u0022toType\u005c\u0022:\u005c\u0022toUser\u005c\u0022, \u005c\u0022toID\u005c\u0022:\u005c\u00221036\u005c\u0022, \u005c\u0022fileType\u005c\u0022:\u005c\u0022image/jpeg\u005c\u0022, \u005c\u0022blob\u005c\u0022:\u005c\u0022\u005c\u0022}\u0022)\n+\tsocket_status.send(\u0022{ \u005c\u0022RequestType\u005c\u0022:\u005c\u0022SendImage\u005c\u0022, \u005c\u0022RequestID\u005c\u0022:\u005c\u0022788346414\u005c\u0022, \u005c\u0022toType\u005c\u0022:\u005c\u0022toUser\u005c\u0022, \u005c\u0022toID\u005c\u0022:\u005c\u00221036\u005c\u0022, \u005c\u0022fileType\u005c\u0022:\u005c\u0022image/jpeg\u005c\u0022, \u005c\u0022blob\u005c\u0022:\u005c\u0022\u003d\u005c\u0022}\u0022)\n+}\n+\n+var socket_ot;\n+\n+function ot_open() {\n+\tif (use_lws_meta)\n+\t\tsocket_ot \u003d lws_meta.new_ws(get_appropriate_ws_url(\u0022\u0022), \u0022dumb-increment-protocol\u0022);\n+\telse\n+\t\tsocket_ot \u003d new_ws(get_appropriate_ws_url(\u0022\u0022), \u0022dumb-increment-protocol\u0022);\n+\n+\tconsole.log(\u0022ot_open\u0022);\n+\n+\ttry {\n+\t\tsocket_ot.onopen \u003d function() {\n+\t\t\tdocument.getElementById(\u0022ot_statustd\u0022).style.backgroundColor \u003d \u0022#40ff40\u0022;\n+\t\t\tdocument.getElementById(\u0022ot_status\u0022).innerHTML \u003d \u0022 \u003cb\u003ewebsocket connection opened\u003c/b\u003e\u003cbr\u003e\u0022 + san(socket_di.extensions);\n+\t\t\tdocument.getElementById(\u0022ot_open_btn\u0022).disabled \u003d true;\n+\t\t\tdocument.getElementById(\u0022ot_close_btn\u0022).disabled \u003d false;\n+\t\t\tdocument.getElementById(\u0022ot_req_close_btn\u0022).disabled \u003d false;\n+\t\t\tconsole.log(\u0022ot_open.onopen\u0022);\n+\t\t} \n+\n+\t\tsocket_ot.onclose \u003d function(e){\n+\t\t\tdocument.getElementById(\u0022ot_statustd\u0022).style.backgroundColor \u003d \u0022#ff4040\u0022;\n+\t\t\tdocument.getElementById(\u0022ot_status\u0022).textContent \u003d \u0022 websocket connection CLOSED, code: \u0022 + e.code +\n+\t\t\t\u0022, reason: \u0022 + e.reason;\n+\t\t\tdocument.getElementById(\u0022ot_open_btn\u0022).disabled \u003d false;\n+\t\t\tdocument.getElementById(\u0022ot_close_btn\u0022).disabled \u003d true;\n+\t\t\tdocument.getElementById(\u0022ot_req_close_btn\u0022).disabled \u003d true;\n+\t\t}\n+\t} catch(exception) {\n+\t\talert('\u003cp\u003eError' + exception); \n+\t}\n+}\n+\n+/* browser will close the ws in a controlled way */\n+function ot_close() {\n+\tsocket_ot.close(3000, \u0022Bye!\u0022);\n+}\n+\n+/* we ask the server to close the ws in a controlled way */\n+function ot_req_close() {\n+\tsocket_ot.send(\u0022closeme\u005cn\u0022);\n+}\n+\n+/* lws-mirror protocol */\n+\n+\tvar down \u003d 0;\n+\tvar no_last \u003d 1;\n+\tvar last_x \u003d 0, last_y \u003d 0;\n+\tvar ctx;\n+\tvar socket_lm;\n+\tvar color \u003d \u0022#000000\u0022;\n+\tvar pending \u003d \u0022\u0022;\n+\tvar lm_timer;\n+\n+\tif (use_lws_meta)\n+\t\tsocket_lm \u003d lws_meta.new_ws(get_appropriate_ws_url(\u0022?mirror\u003d\u0022 + mirror_name),\n+\t\t\t\u0022lws-mirror-protocol\u0022);\n+\telse\n+\t\tsocket_lm \u003d new_ws(get_appropriate_ws_url(\u0022?mirror\u003d\u0022 + mirror_name),\n+\t\t\t\u0022lws-mirror-protocol\u0022);\n+\ttry {\n+\t\tsocket_lm.onopen \u003d function() {\n+\t\t\tdocument.getElementById(\u0022wslm_statustd\u0022).style.backgroundColor \u003d \u0022#40ff40\u0022;\n+\t\t\tdocument.getElementById(\u0022wslm_status\u0022).innerHTML \u003d\n+\t\t\t\t\u0022 \u003cb\u003ewebsocket connection opened\u003c/b\u003e\u003cbr\u003e\u0022 +\n+\t\t\t\tsan(socket_lm.extensions);\n+\t\t\tlws_gray_out(false);\n+\t\t} \n+\n+\t\tsocket_lm.onmessage \u003dfunction got_packet(msg) {\n+\t\t\tj \u003d msg.data.split(';');\n+\t\t\tf \u003d 0;\n+\t\t\twhile (f \u003c j.length - 1) {\n+\t\t\t\ti \u003d j[f].split(' ');\n+\t\t\t\tif (i[0] \u003d\u003d 'd') {\n+\t\t\t\t\tctx.strokeStyle \u003d i[1];\n+\t\t\t\t\tctx.beginPath();\n+\t\t\t\t\tctx.moveTo(+(i[2]), +(i[3]));\n+\t\t\t\t\tctx.lineTo(+(i[4]), +(i[5]));\n+\t\t\t\t\tctx.stroke();\n+\t\t\t\t}\n+\t\t\t\tif (i[0] \u003d\u003d 'c') {\n+\t\t\t\t\tctx.strokeStyle \u003d i[1];\n+\t\t\t\t\tctx.beginPath();\n+\t\t\t\t\tctx.arc(+(i[2]), +(i[3]), +(i[4]), 0, Math.PI*2, true); \n+\t\t\t\t\tctx.stroke();\n+\t\t\t\t}\n+\n+\t\t\t\tf++;\n+\t\t\t}\n+\t\t}\n+\n+\t\tsocket_lm.onclose \u003d function(){\n+\t\t\tdocument.getElementById(\u0022wslm_statustd\u0022).style.backgroundColor \u003d \u0022#ff4040\u0022;\n+\t\t\tdocument.getElementById(\u0022wslm_status\u0022).textContent \u003d \u0022 websocket connection CLOSED \u0022;\n+\t\t\tlws_gray_out(true,{'zindex':'499'});\n+\t\t}\n+\t} catch(exception) {\n+\t\talert('\u003cp\u003eError' + exception); \n+\t}\n+\n+\tvar canvas \u003d document.createElement('canvas');\n+\tcanvas.height \u003d 300;\n+\tcanvas.width \u003d 480;\n+\tctx \u003d canvas.getContext(\u00222d\u0022);\n+\n+\tdocument.getElementById('wslm_drawing').appendChild(canvas);\n+\n+\tcanvas.addEventListener('mousemove', ev_mousemove, false);\n+\tcanvas.addEventListener('mousedown', ev_mousedown, false);\n+\tcanvas.addEventListener('mouseup', ev_mouseup, false);\n+\n+\toffsetX \u003d offsetY \u003d 0;\n+\telement \u003d canvas;\n+ if (element.offsetParent) {\n+ do {\n+ offsetX +\u003d element.offsetLeft;\n+ offsetY +\u003d element.offsetTop;\n+ } while ((element \u003d element.offsetParent));\n+ }\n+ \n+function update_color() {\n+\tcolor \u003d document.getElementById(\u0022color\u0022).value;\n+}\n+\n+function ev_mousedown (ev) {\n+\tdown \u003d 1;\n+}\n+\n+function ev_mouseup(ev) {\n+\tdown \u003d 0;\n+\tno_last \u003d 1;\n+}\n+\n+function lm_timer_handler(ev) {\n+\tsocket_lm.send(pending);\n+\tpending\u003d\u0022\u0022;\n+}\n+\n+function ev_mousemove (ev) {\n+\tvar x, y;\n+\n+\tif (ev.offsetX) {\n+\t\tx \u003d ev.offsetX;\n+\t\ty \u003d ev.offsetY;\n+\t} else {\n+\t\tx \u003d ev.layerX - offsetX;\n+\t\ty \u003d ev.layerY - offsetY;\n+\n+\t}\n+\n+\tif (!down)\n+\t\treturn;\n+\tif (no_last) {\n+\t\tno_last \u003d 0;\n+\t\tlast_x \u003d x;\n+\t\tlast_y \u003d y;\n+\t\treturn;\n+\t}\n+\tpending \u003d pending + \u0022d \u0022 + color + \u0022 \u0022 + last_x + \u0022 \u0022 + last_y +\n+\t\t\t\u0022 \u0022 + x + ' ' + y + ';';\n+\t\t\t\n+\tif (pending.length \u003e 400) {\n+\t\tsocket_lm.send(pending);\n+\t\tclearTimeout(lm_timer);\n+\t\tpending \u003d \u0022\u0022;\n+\t} else\n+\t\tlm_timer \u003d setTimeout(lm_timer_handler, 1);\n+\n+\tlast_x \u003d x;\n+\tlast_y \u003d y;\n+}\n+\n+\n+\u003c/script\u003e\n+\n+\u003c/body\u003e\n+\u003c/html\u003e\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/wss-over-h2.png b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/wss-over-h2.png\nnew file mode 100644\nindex 0000000..1a62d83\nBinary files /dev/null and b/minimal-examples/http-server/minimal-http-server-eventlib-demos/mount-origin/wss-over-h2.png differ\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-foreign/README.md b/minimal-examples/http-server/minimal-http-server-eventlib-foreign/README.md\nindex a663d6e..4c21fa1 100644\n--- a/minimal-examples/http-server/minimal-http-server-eventlib-foreign/README.md\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-foreign/README.md\n@@ -41,8 +41,8 @@ exits itself.\n ## usage\n \n ```\n- $ ./lws-minimal-http-server-libuv-foreign\n-[2018/03/29 12:19:31:3480] USER: LWS minimal http server libuv + foreign loop | visit http://localhost:7681\n+ $ ./lws-minimal-http-server-eventlib-foreign\n+[2018/03/29 12:19:31:3480] USER: LWS minimal http server eventlib + foreign loop | visit http://localhost:7681\n [2018/03/29 12:19:31:3724] NOTICE: Creating Vhost 'default' port 7681, 1 protocols, IPv6 off\n [2018/03/29 12:19:31:3804] NOTICE: Using foreign event loop...\n [2018/03/29 12:19:31:3938] USER: Foreign 1Hz timer\ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib-foreign/minimal-http-server-eventlib-foreign.c b/minimal-examples/http-server/minimal-http-server-eventlib-foreign/minimal-http-server-eventlib-foreign.c\nindex 3e807d9..f31b7bd 100644\n--- a/minimal-examples/http-server/minimal-http-server-eventlib-foreign/minimal-http-server-eventlib-foreign.c\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib-foreign/minimal-http-server-eventlib-foreign.c\n@@ -337,7 +337,7 @@ int main(int argc, const char **argv)\n \t\tlogs \u003d atoi(p);\n \n \tlws_set_log_level(logs, NULL);\n-\tlwsl_user(\u0022LWS minimal http server libuv + foreign loop |\u0022\n+\tlwsl_user(\u0022LWS minimal http server eventlib + foreign loop |\u0022\n \t\t \u0022 visit http://localhost:7681\u005cn\u0022);\n \n \t/*\n@@ -383,8 +383,7 @@ int main(int argc, const char **argv)\n \tlwsl_user(\u0022\u005cn\u0022);\n \tlwsl_user(\u0022 Finally close only the timer and signalhandler and\u005cn\u0022);\n \tlwsl_user(\u0022 exit the loop cleanly\u005cn\u0022);\n-\n-\tlwsl_notice(\u0022%s\u005cn\u0022, info.ssl_cert_filepath);\n+\tlwsl_user(\u0022\u005cn\u0022);\n \n \t/* foreign loop specific startup and run */\n \ndiff --git a/minimal-examples/http-server/minimal-http-server-eventlib/README.md b/minimal-examples/http-server/minimal-http-server-eventlib/README.md\nindex 873e250..ecfb733 100644\n--- a/minimal-examples/http-server/minimal-http-server-eventlib/README.md\n+++ b/minimal-examples/http-server/minimal-http-server-eventlib/README.md\n@@ -1,4 +1,13 @@\n-# lws minimal http server libuv\n+# lws minimal http server eventlib\n+\n+This demonstrates a minimal http server that can use any of the event libraries\n+\n+Commandline option|Meaning\n+---|---\n+-d \u003cloglevel\u003e|Debug verbosity in decimal, eg, -d15\n+--uv|Use the libuv event library (lws must have been configured with `-DLWS_WITH_LIBUV\u003d1`)\n+--event|Use the libevent library (lws must have been configured with `-DLWS_WITH_LIBEVENT\u003d1`)\n+--ev|Use the libev event library (lws must have been configured with `-DLWS_WITH_LIBEV\u003d1`)\n \n ## build\n \n@@ -9,8 +18,8 @@\n ## usage\n \n ```\n- $ ./lws-minimal-http-server-libuv\n-[2018/03/04 09:30:02:7986] USER: LWS minimal http server-libuv | visit http://localhost:7681\n+ $ ./lws-minimal-http-server-eventlib\n+[2018/03/04 09:30:02:7986] USER: LWS minimal http server-eventlib | visit http://localhost:7681\n [2018/03/04 09:30:02:7986] NOTICE: Creating Vhost 'default' port 7681, 1 protocols, IPv6 on\n ```\n \n","s":{"c":1711642003,"u": 19872}} ],"g": 157919,"chitpc": 0,"ehitpc": 0,"indexed":0 , "ab": 0, "si": 0, "db":0, "di":0, "sat":0, "lfc": "0000"}