blob: bdfecdac456a8abb9182eea09a1c00ca4cdacd12 [file] [log] [blame]
Andy Green7c212cc2010-11-08 20:20:42 +00001/*
2 * libwebsockets - small server side websockets and web server implementation
Andy Greene77ddd82010-11-13 10:03:47 +00003 *
Andy Green8c1f6022016-01-26 20:56:56 +08004 * Copyright (C) 2010 - 2016 Andy Green <andy@warmcat.com>
Andy Green7c212cc2010-11-08 20:20:42 +00005 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation:
9 * version 2.1 of the License.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
19 * MA 02110-1301 USA
20 */
Joakim Soderberg91de9332013-02-06 15:30:33 +090021
Joakim Soderberg4c531232013-02-06 15:26:58 +090022#include "lws_config.h"
Roger A. Light7a474b42015-06-26 11:40:54 +020023#include "lws_config_private.h"
Andy Greene40aa9b2014-04-02 21:02:54 +080024
=?UTF-8?q?Joakim=20S=C3=B6derberg?=cefab312015-06-24 16:46:02 +020025#ifdef LWS_HAVE_SYS_TYPES_H
Andy Greene40aa9b2014-04-02 21:02:54 +080026#include <sys/types.h>
Joakim Soderberg91de9332013-02-06 15:30:33 +090027#endif
Joakim Soderberg4c531232013-02-06 15:26:58 +090028
Andy Green7c212cc2010-11-08 20:20:42 +000029#include <stdio.h>
30#include <stdlib.h>
31#include <string.h>
Patrick Ganstererb13eed42014-03-30 10:19:23 +020032#include <time.h>
Andy Green7c212cc2010-11-08 20:20:42 +000033#include <ctype.h>
Peter Young26757a72013-01-17 10:08:16 +080034#include <limits.h>
Peter Hinz56885f32011-03-02 22:03:47 +000035#include <stdarg.h>
Andy Green112f9802015-12-04 07:22:44 +080036#include <assert.h>
Andy Green8c1f6022016-01-26 20:56:56 +080037#if LWS_MAX_SMP > 1
38#include <pthread.h>
39#endif
Peter Hinz56885f32011-03-02 22:03:47 +000040
=?UTF-8?q?Joakim=20S=C3=B6derberg?=cefab312015-06-24 16:46:02 +020041#ifdef LWS_HAVE_SYS_STAT_H
Peter Hinz56885f32011-03-02 22:03:47 +000042#include <sys/stat.h>
Patrick Gansterere5720a32014-02-28 00:57:19 +010043#endif
Peter Hinz56885f32011-03-02 22:03:47 +000044
Andreas Pakulat68bd4bd2013-10-28 15:18:04 +010045#if defined(WIN32) || defined(_WIN32)
Stephan Eberleb820e2c2015-10-23 08:10:55 +020046#if (WINVER < 0x0501)
47#undef WINVER
48#undef _WIN32_WINNT
49#define WINVER 0x0501
50#define _WIN32_WINNT WINVER
51#endif
Joakim Soderberg4c531232013-02-06 15:26:58 +090052#define LWS_NO_DAEMONIZE
Patrick Gansterer2dbd8372014-02-28 12:37:52 +010053#define LWS_ERRNO WSAGetLastError()
54#define LWS_EAGAIN WSAEWOULDBLOCK
55#define LWS_EALREADY WSAEALREADY
56#define LWS_EINPROGRESS WSAEINPROGRESS
57#define LWS_EINTR WSAEINTR
58#define LWS_EISCONN WSAEISCONN
59#define LWS_EWOULDBLOCK WSAEWOULDBLOCK
Patrick Ganstererb47f87b2014-03-30 09:18:05 +020060#define LWS_POLLHUP (FD_CLOSE)
Patrick Gansterer0fc37b62014-03-28 15:44:56 +010061#define LWS_POLLIN (FD_READ | FD_ACCEPT)
62#define LWS_POLLOUT (FD_WRITE)
Patrick Gansterer73882e42014-03-29 08:25:58 +010063#define MSG_NOSIGNAL 0
64#define SHUT_RDWR SD_BOTH
65#define SOL_TCP IPPROTO_TCP
Andy Green8c1f6022016-01-26 20:56:56 +080066#define SHUT_WR SD_SEND
Joakim Soderberg4c531232013-02-06 15:26:58 +090067
Andy Green158e8042014-04-02 14:25:10 +080068#define compatible_close(fd) closesocket(fd)
Andy Greenaa775fd2015-12-26 08:56:58 +080069#define lws_set_blocking_send(wsi) wsi->sock_send_blocking = 1
Andy Greenc53f7ca2015-11-14 07:35:27 +080070#define lws_socket_is_valid(x) (!!x)
Andy Green40110e82015-12-14 08:52:03 +080071#define LWS_SOCK_INVALID 0
Peter Hinz56885f32011-03-02 22:03:47 +000072#include <winsock2.h>
Andy Greeneee0d8a2015-12-17 15:15:12 +080073#include <ws2tcpip.h>
Peter Hinz56885f32011-03-02 22:03:47 +000074#include <windows.h>
Joakim Soderbergd2f5b192014-04-07 11:28:08 +020075#include <tchar.h>
=?UTF-8?q?Joakim=20S=C3=B6derberg?=cefab312015-06-24 16:46:02 +020076#ifdef LWS_HAVE_IN6ADDR_H
Andy Green0f58db32014-04-12 11:10:35 +080077#include <in6addr.h>
78#endif
Joakim Soderbergd2f5b192014-04-07 11:28:08 +020079#include <mstcpip.h>
80
81#ifndef __func__
82#define __func__ __FUNCTION__
83#endif
84
Patrick Gansterer6bb4b622014-04-15 18:39:26 +020085#ifdef _WIN32_WCE
86#define vsnprintf _vsnprintf
87#endif
88
Andy Green158e8042014-04-02 14:25:10 +080089#else /* not windows --> */
Andy Green8c0d3c02015-11-02 20:34:12 +080090
Patrick Ganstererb13eed42014-03-30 10:19:23 +020091#include <fcntl.h>
Patrick Ganstererb13eed42014-03-30 10:19:23 +020092#include <strings.h>
93#include <unistd.h>
Andy Greene77ddd82010-11-13 10:03:47 +000094#include <sys/types.h>
Andy Green5f2a8152015-11-02 08:21:08 +080095#ifndef MBED_OPERATORS
Andy Green8c0d3c02015-11-02 20:34:12 +080096#ifndef __cplusplus
97#include <errno.h>
98#endif
Andy Green5f2a8152015-11-02 08:21:08 +080099#include <netdb.h>
100#include <signal.h>
Andy Green7c212cc2010-11-08 20:20:42 +0000101#include <sys/socket.h>
Andy Greene40aa9b2014-04-02 21:02:54 +0800102#ifdef LWS_BUILTIN_GETIFADDRS
103 #include <getifaddrs.h>
104#else
105 #include <ifaddrs.h>
106#endif
Dnyanesh Gate759e50c2014-09-26 05:39:05 +0800107#if defined (__ANDROID__)
108#include <syslog.h>
109#else
Andy Greene40aa9b2014-04-02 21:02:54 +0800110#include <sys/syslog.h>
Dnyanesh Gate759e50c2014-09-26 05:39:05 +0800111#endif
Andy Greene40aa9b2014-04-02 21:02:54 +0800112#include <sys/un.h>
113#include <sys/socket.h>
114#include <netdb.h>
Andy Green7c212cc2010-11-08 20:20:42 +0000115#include <netinet/in.h>
Andy Green6c939552011-03-08 08:56:57 +0000116#include <netinet/tcp.h>
Andy Greenb45993c2010-12-18 15:13:50 +0000117#include <arpa/inet.h>
Andy Green7c212cc2010-11-08 20:20:42 +0000118#include <poll.h>
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800119#ifdef LWS_USE_LIBEV
Andy Green86ed65f2016-02-14 09:27:41 +0800120#include <ev.h>
121#endif
122#ifdef LWS_USE_LIBUV
Alex Hultmana43c2ac2016-02-01 08:31:54 +0800123#include <uv.h>
Andy Green86ed65f2016-02-14 09:27:41 +0800124#endif
Andy Green7c212cc2010-11-08 20:20:42 +0000125#include <sys/mman.h>
Andy Green5f2a8152015-11-02 08:21:08 +0800126
127#endif /* MBED */
128
129#ifndef LWS_NO_FORK
130#ifdef LWS_HAVE_SYS_PRCTL_H
131#include <sys/prctl.h>
132#endif
133#endif
134
Andy Green038d5822011-02-14 20:58:26 +0000135#include <sys/time.h>
Andy Green7c212cc2010-11-08 20:20:42 +0000136
Patrick Gansterer2dbd8372014-02-28 12:37:52 +0100137#define LWS_ERRNO errno
138#define LWS_EAGAIN EAGAIN
139#define LWS_EALREADY EALREADY
140#define LWS_EINPROGRESS EINPROGRESS
141#define LWS_EINTR EINTR
142#define LWS_EISCONN EISCONN
143#define LWS_EWOULDBLOCK EWOULDBLOCK
Patrick Ganstererb47f87b2014-03-30 09:18:05 +0200144#define LWS_POLLHUP (POLLHUP|POLLERR)
145#define LWS_POLLIN (POLLIN)
146#define LWS_POLLOUT (POLLOUT)
Andy Green158e8042014-04-02 14:25:10 +0800147#define compatible_close(fd) close(fd)
Andy Green158e8042014-04-02 14:25:10 +0800148#define lws_set_blocking_send(wsi)
Andy Green2cd30742015-11-02 13:10:33 +0800149
150#ifdef MBED_OPERATORS
151#define lws_socket_is_valid(x) ((x) != NULL)
152#define LWS_SOCK_INVALID (NULL)
153#else
Andy Greenc53f7ca2015-11-14 07:35:27 +0800154#define lws_socket_is_valid(x) (x >= 0)
155#define LWS_SOCK_INVALID (-1)
Peter Hinz56885f32011-03-02 22:03:47 +0000156#endif
Andy Green2cd30742015-11-02 13:10:33 +0800157#endif
Peter Hinz56885f32011-03-02 22:03:47 +0000158
=?UTF-8?q?Joakim=20S=C3=B6derberg?=cefab312015-06-24 16:46:02 +0200159#ifndef LWS_HAVE_BZERO
Andy Greene5ea1f92014-11-18 18:25:24 +0800160#ifndef bzero
Patrick Gansterer4a837272014-02-28 13:17:49 +0100161#define bzero(b, len) (memset((b), '\0', (len)), (void) 0)
162#endif
Andy Greene5ea1f92014-11-18 18:25:24 +0800163#endif
Patrick Gansterer4a837272014-02-28 13:17:49 +0100164
=?UTF-8?q?Joakim=20S=C3=B6derberg?=cefab312015-06-24 16:46:02 +0200165#ifndef LWS_HAVE_STRERROR
Patrick Gansterer9d614912014-02-28 00:59:53 +0100166#define strerror(x) ""
167#endif
168
Andy Green7c212cc2010-11-08 20:20:42 +0000169#ifdef LWS_OPENSSL_SUPPORT
Alexander Bruinesc3bcb892015-08-08 18:54:49 +0200170#ifdef USE_WOLFSSL
ABruines80a70682015-08-09 22:56:32 +0200171#ifdef USE_OLD_CYASSL
172#include <cyassl/openssl/ssl.h>
173#include <cyassl/error-ssl.h>
174#else
Alexander Bruinesc3bcb892015-08-08 18:54:49 +0200175#include <wolfssl/openssl/ssl.h>
176#include <wolfssl/error-ssl.h>
ABruines80a70682015-08-09 22:56:32 +0200177#endif /* not USE_OLD_CYASSL */
Andy Green23c5f2e2013-02-06 15:43:00 +0900178#else
Andy Green7c212cc2010-11-08 20:20:42 +0000179#include <openssl/ssl.h>
180#include <openssl/evp.h>
181#include <openssl/err.h>
Andy Green70dfebd2010-12-20 09:35:03 +0000182#include <openssl/md5.h>
Andy Greene2522172011-01-18 17:14:03 +0000183#include <openssl/sha.h>
Alexander Bruinesc3bcb892015-08-08 18:54:49 +0200184#endif /* not USE_WOLFSSL */
Darin Willitsdb9ba422011-02-14 20:56:24 +0000185#endif
186
Andy Green7c212cc2010-11-08 20:20:42 +0000187#include "libwebsockets.h"
188
Andy Green8c0d3c02015-11-02 20:34:12 +0800189#if defined(MBED_OPERATORS)
190#undef compatible_close
191#define compatible_close(fd) mbed3_delete_tcp_stream_socket(fd)
Andy Green11f27342015-11-08 12:10:26 +0800192#ifndef BIG_ENDIAN
193#define BIG_ENDIAN 4321 /* to show byte order (taken from gcc) */
194#endif
195#ifndef LITTLE_ENDIAN
196#define LITTLE_ENDIAN 1234
197#endif
198#ifndef BYTE_ORDER
199#define BYTE_ORDER LITTLE_ENDIAN
200#endif
Andy Green8c0d3c02015-11-02 20:34:12 +0800201#endif
202
Andy Green158e8042014-04-02 14:25:10 +0800203#if defined(WIN32) || defined(_WIN32)
204
205#ifndef BIG_ENDIAN
206#define BIG_ENDIAN 4321 /* to show byte order (taken from gcc) */
207#endif
208#ifndef LITTLE_ENDIAN
209#define LITTLE_ENDIAN 1234
210#endif
211#ifndef BYTE_ORDER
212#define BYTE_ORDER LITTLE_ENDIAN
213#endif
Andy Green11f27342015-11-08 12:10:26 +0800214#ifndef u_int64_t
Andy Green158e8042014-04-02 14:25:10 +0800215typedef unsigned __int64 u_int64_t;
Andy Green11f27342015-11-08 12:10:26 +0800216#endif
Andy Green158e8042014-04-02 14:25:10 +0800217
218#undef __P
219#ifndef __P
220#if __STDC__
221#define __P(protos) protos
222#else
223#define __P(protos) ()
224#endif
225#endif
226
227#else
228
229#include <sys/stat.h>
230#include <sys/cdefs.h>
231#include <sys/time.h>
232
233#if defined(__APPLE__)
234#include <machine/endian.h>
235#elif defined(__FreeBSD__)
236#include <sys/endian.h>
237#elif defined(__linux__)
238#include <endian.h>
239#endif
240
Andy Green8c0d3c02015-11-02 20:34:12 +0800241#ifdef __cplusplus
242extern "C" {
243#endif
Alejandro Meryead8afe2014-12-07 03:36:11 +0100244#include <stddef.h>
245
246#ifndef container_of
247#define container_of(P,T,M) ((T *)((char *)(P) - offsetof(T, M)))
248#endif
249
emironova49d0842014-09-16 14:05:13 +0400250#if defined(__QNX__)
251 #include <gulliver.h>
252 #if defined(__LITTLEENDIAN__)
253 #define BYTE_ORDER __LITTLEENDIAN__
254 #define LITTLE_ENDIAN __LITTLEENDIAN__
255 #define BIG_ENDIAN 4321 /* to show byte order (taken from gcc); for suppres warning that BIG_ENDIAN is not defined. */
256 #endif
257 #if defined(__BIGENDIAN__)
258 #define BYTE_ORDER __BIGENDIAN__
259 #define LITTLE_ENDIAN 1234 /* to show byte order (taken from gcc); for suppres warning that LITTLE_ENDIAN is not defined. */
260 #define BIG_ENDIAN __BIGENDIAN__
261 #endif
262#endif
263
Andy Green158e8042014-04-02 14:25:10 +0800264#if !defined(BYTE_ORDER)
265# define BYTE_ORDER __BYTE_ORDER
266#endif
267#if !defined(LITTLE_ENDIAN)
268# define LITTLE_ENDIAN __LITTLE_ENDIAN
269#endif
270#if !defined(BIG_ENDIAN)
271# define BIG_ENDIAN __BIG_ENDIAN
272#endif
273
274#endif
275
Darin Willitsc19456f2011-02-14 17:52:39 +0000276/*
277 * Mac OSX as well as iOS do not define the MSG_NOSIGNAL flag,
278 * but happily have something equivalent in the SO_NOSIGPIPE flag.
279 */
280#ifdef __APPLE__
Andy Green6ee372f2012-04-09 15:09:01 +0800281#define MSG_NOSIGNAL SO_NOSIGPIPE
Darin Willitsc19456f2011-02-14 17:52:39 +0000282#endif
283
Bud Davis229bfec2015-01-30 10:13:01 +0800284#ifdef _WIN32
285#ifndef FD_HASHTABLE_MODULUS
286#define FD_HASHTABLE_MODULUS 32
287#endif
288#endif
289
Andy Green8c1f6022016-01-26 20:56:56 +0800290#ifndef LWS_DEF_HEADER_LEN
291#define LWS_DEF_HEADER_LEN 1024
Andy Greenc0d6b632013-01-12 23:42:17 +0800292#endif
Andy Green8c1f6022016-01-26 20:56:56 +0800293#ifndef LWS_DEF_HEADER_POOL
294#define LWS_DEF_HEADER_POOL 16
Andy Green3df58002015-12-25 12:44:12 +0800295#endif
Andy Greenc0d6b632013-01-12 23:42:17 +0800296#ifndef LWS_MAX_PROTOCOLS
Andy Greend91d5e82013-02-10 16:00:47 +0800297#define LWS_MAX_PROTOCOLS 5
Andy Greenc0d6b632013-01-12 23:42:17 +0800298#endif
299#ifndef LWS_MAX_EXTENSIONS_ACTIVE
Andy Greend738f842016-01-19 04:32:14 +0800300#define LWS_MAX_EXTENSIONS_ACTIVE 2
Andy Greenc0d6b632013-01-12 23:42:17 +0800301#endif
Andy Green67112662016-01-11 11:34:01 +0800302#ifndef LWS_MAX_EXT_OFFERS
303#define LWS_MAX_EXT_OFFERS 8
304#endif
Andy Greenc0d6b632013-01-12 23:42:17 +0800305#ifndef SPEC_LATEST_SUPPORTED
Andy Greend85cb202011-09-25 09:32:54 +0100306#define SPEC_LATEST_SUPPORTED 13
Andy Greenc0d6b632013-01-12 23:42:17 +0800307#endif
308#ifndef AWAITING_TIMEOUT
Andy Green8c1f6022016-01-26 20:56:56 +0800309#define AWAITING_TIMEOUT 20
Andy Greenc0d6b632013-01-12 23:42:17 +0800310#endif
311#ifndef CIPHERS_LIST_STRING
David Galeanof177f2a2013-01-10 10:15:19 +0800312#define CIPHERS_LIST_STRING "DEFAULT"
Andy Greenc0d6b632013-01-12 23:42:17 +0800313#endif
Andy Greena824d182013-01-15 20:52:29 +0800314#ifndef LWS_SOMAXCONN
315#define LWS_SOMAXCONN SOMAXCONN
316#endif
Andy Green7c212cc2010-11-08 20:20:42 +0000317
Andy Greene2522172011-01-18 17:14:03 +0000318#define MAX_WEBSOCKET_04_KEY_LEN 128
Andy Green54495112013-02-06 21:10:16 +0900319#define LWS_MAX_SOCKET_IO_BUF 4096
Andy Greenc0d6b632013-01-12 23:42:17 +0800320
321#ifndef SYSTEM_RANDOM_FILEPATH
Andy Green4739e5c2011-01-22 12:51:57 +0000322#define SYSTEM_RANDOM_FILEPATH "/dev/urandom"
Andy Greenc0d6b632013-01-12 23:42:17 +0800323#endif
Andy Greenc0d6b632013-01-12 23:42:17 +0800324
Andy Green5fd55cd2011-04-23 10:54:53 +0100325enum lws_websocket_opcodes_07 {
Andy Green54806b12015-12-17 17:03:59 +0800326 LWSWSOPC_CONTINUATION = 0,
327 LWSWSOPC_TEXT_FRAME = 1,
328 LWSWSOPC_BINARY_FRAME = 2,
Andy Greena41314f2011-05-23 10:00:03 +0100329
Andy Green54806b12015-12-17 17:03:59 +0800330 LWSWSOPC_NOSPEC__MUX = 7,
Andy Greena41314f2011-05-23 10:00:03 +0100331
332 /* control extensions 8+ */
333
Andy Green54806b12015-12-17 17:03:59 +0800334 LWSWSOPC_CLOSE = 8,
335 LWSWSOPC_PING = 9,
336 LWSWSOPC_PONG = 0xa,
Andy Green5fd55cd2011-04-23 10:54:53 +0100337};
338
Andy Greena41314f2011-05-23 10:00:03 +0100339
Andy Green7c212cc2010-11-08 20:20:42 +0000340enum lws_connection_states {
Andy Green54806b12015-12-17 17:03:59 +0800341 LWSS_HTTP,
342 LWSS_HTTP_ISSUING_FILE,
343 LWSS_HTTP_HEADERS,
344 LWSS_HTTP_BODY,
345 LWSS_DEAD_SOCKET,
346 LWSS_ESTABLISHED,
347 LWSS_CLIENT_UNCONNECTED,
348 LWSS_RETURNED_CLOSE_ALREADY,
349 LWSS_AWAITING_CLOSE_ACK,
350 LWSS_FLUSHING_STORED_SEND_BEFORE_CLOSE,
Andy Green8c1f6022016-01-26 20:56:56 +0800351 LWSS_SHUTDOWN,
Andy Green40110e82015-12-14 08:52:03 +0800352
Andy Green54806b12015-12-17 17:03:59 +0800353 LWSS_HTTP2_AWAIT_CLIENT_PREFACE,
354 LWSS_HTTP2_ESTABLISHED_PRE_SETTINGS,
355 LWSS_HTTP2_ESTABLISHED,
Andy Green7c212cc2010-11-08 20:20:42 +0000356};
357
Andrew Canadayafe26cf2014-07-13 01:07:36 -0400358enum http_version {
359 HTTP_VERSION_1_0,
360 HTTP_VERSION_1_1,
361};
362
363enum http_connection_type {
364 HTTP_CONNECTION_CLOSE,
365 HTTP_CONNECTION_KEEP_ALIVE
366};
367
Andy Green024eb6c2014-10-08 12:00:53 +0800368enum lws_pending_protocol_send {
369 LWS_PPS_NONE,
370 LWS_PPS_HTTP2_MY_SETTINGS,
371 LWS_PPS_HTTP2_ACK_SETTINGS,
Andy Greenbbbf07a2014-10-27 16:46:44 +0800372 LWS_PPS_HTTP2_PONG,
Andy Green024eb6c2014-10-08 12:00:53 +0800373};
374
Andy Green7c212cc2010-11-08 20:20:42 +0000375enum lws_rx_parse_state {
376 LWS_RXPS_NEW,
Andy Greene77ddd82010-11-13 10:03:47 +0000377
Andy Green67112662016-01-11 11:34:01 +0800378 LWS_RXPS_04_mask_1,
379 LWS_RXPS_04_mask_2,
380 LWS_RXPS_04_mask_3,
Andy Green3e5eb782011-01-18 18:14:26 +0000381
382 LWS_RXPS_04_FRAME_HDR_1,
Andy Green38e57bb2011-01-19 12:20:27 +0000383 LWS_RXPS_04_FRAME_HDR_LEN,
384 LWS_RXPS_04_FRAME_HDR_LEN16_2,
385 LWS_RXPS_04_FRAME_HDR_LEN16_1,
386 LWS_RXPS_04_FRAME_HDR_LEN64_8,
387 LWS_RXPS_04_FRAME_HDR_LEN64_7,
388 LWS_RXPS_04_FRAME_HDR_LEN64_6,
389 LWS_RXPS_04_FRAME_HDR_LEN64_5,
390 LWS_RXPS_04_FRAME_HDR_LEN64_4,
391 LWS_RXPS_04_FRAME_HDR_LEN64_3,
392 LWS_RXPS_04_FRAME_HDR_LEN64_2,
393 LWS_RXPS_04_FRAME_HDR_LEN64_1,
Andy Green3e5eb782011-01-18 18:14:26 +0000394
Andy Green283d0a22011-04-24 05:46:23 +0100395 LWS_RXPS_07_COLLECT_FRAME_KEY_1,
396 LWS_RXPS_07_COLLECT_FRAME_KEY_2,
397 LWS_RXPS_07_COLLECT_FRAME_KEY_3,
398 LWS_RXPS_07_COLLECT_FRAME_KEY_4,
399
Andy Green7c212cc2010-11-08 20:20:42 +0000400 LWS_RXPS_PAYLOAD_UNTIL_LENGTH_EXHAUSTED
401};
402
403
Andy Green0d338332011-02-12 11:57:43 +0000404enum connection_mode {
Andy Green54806b12015-12-17 17:03:59 +0800405 LWSCM_HTTP_SERVING,
406 LWSCM_HTTP_SERVING_ACCEPTED, /* actual HTTP service going on */
407 LWSCM_PRE_WS_SERVING_ACCEPT,
Andy Greend280b6e2013-01-15 13:40:23 +0800408
Andy Green54806b12015-12-17 17:03:59 +0800409 LWSCM_WS_SERVING,
410 LWSCM_WS_CLIENT,
Andy Green40110e82015-12-14 08:52:03 +0800411
Andy Green54806b12015-12-17 17:03:59 +0800412 LWSCM_HTTP2_SERVING,
Andy Green0d338332011-02-12 11:57:43 +0000413
Andy Greene2160712013-01-28 12:19:10 +0800414 /* transient, ssl delay hiding */
Andy Green54806b12015-12-17 17:03:59 +0800415 LWSCM_SSL_ACK_PENDING,
Andy Green8c1f6022016-01-26 20:56:56 +0800416 LWSCM_SSL_INIT,
Andy Greene2160712013-01-28 12:19:10 +0800417
Andy Greenbe93fef2011-02-14 20:25:43 +0000418 /* transient modes */
Andy Green54806b12015-12-17 17:03:59 +0800419 LWSCM_WSCL_WAITING_CONNECT,
420 LWSCM_WSCL_WAITING_PROXY_REPLY,
421 LWSCM_WSCL_ISSUE_HANDSHAKE,
422 LWSCM_WSCL_ISSUE_HANDSHAKE2,
423 LWSCM_WSCL_WAITING_SSL,
424 LWSCM_WSCL_WAITING_SERVER_REPLY,
425 LWSCM_WSCL_WAITING_EXTENSION_CONNECT,
426 LWSCM_WSCL_PENDING_CANDIDATE_CHILD,
Andy Greenbe93fef2011-02-14 20:25:43 +0000427
Andy Green0d338332011-02-12 11:57:43 +0000428 /* special internal types */
Andy Green54806b12015-12-17 17:03:59 +0800429 LWSCM_SERVER_LISTENER,
Andy Green0d338332011-02-12 11:57:43 +0000430};
431
Andy Greenca0a1292013-03-16 11:24:23 +0800432enum {
433 LWS_RXFLOW_ALLOW = (1 << 0),
434 LWS_RXFLOW_PENDING_CHANGE = (1 << 1),
435};
436
Andy Green1fb95e82015-12-26 17:20:34 +0800437/* this is not usable directly by user code any more, lws_close_reason() */
438#define LWS_WRITE_CLOSE 4
439
Andy Green4b85c1d2015-12-04 11:08:32 +0800440struct lws_protocols;
441struct lws;
Andy Greene92cd172011-01-19 13:11:55 +0000442
Andy Green86ed65f2016-02-14 09:27:41 +0800443#if defined(LWS_USE_LIBEV) || defined(LWS_USE_LIBUV)
444
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800445struct lws_io_watcher {
Andy Green86ed65f2016-02-14 09:27:41 +0800446#ifdef LWS_USE_LIBEV
447 ev_io ev_watcher;
448#endif
449#ifdef LWS_USE_LIBUV
450 uv_poll_t uv_watcher;
451#endif
452 struct lws_context *context;
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800453};
454
455struct lws_signal_watcher {
Andy Green86ed65f2016-02-14 09:27:41 +0800456#ifdef LWS_USE_LIBEV
457 ev_signal ev_watcher;
458#endif
459#ifdef LWS_USE_LIBUV
460 uv_signal_t uv_watcher;
461#endif
462 struct lws_context *context;
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800463};
Andy Green86ed65f2016-02-14 09:27:41 +0800464#endif
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800465
Bud Davis229bfec2015-01-30 10:13:01 +0800466#ifdef _WIN32
467#define LWS_FD_HASH(fd) ((fd ^ (fd >> 8) ^ (fd >> 16)) % FD_HASHTABLE_MODULUS)
Andy Green3ef579b2015-12-04 09:23:56 +0800468struct lws_fd_hashtable {
Andy Green4b85c1d2015-12-04 11:08:32 +0800469 struct lws **wsi;
Bud Davis229bfec2015-01-30 10:13:01 +0800470 int length;
471};
472#endif
473
Andy Green3df58002015-12-25 12:44:12 +0800474/*
475 * This is totally opaque to code using the library. It's exported as a
476 * forward-reference pointer-only declaration; the user can use the pointer with
477 * other APIs to get information out of it.
478 */
479
480struct lws_fragments {
481 unsigned short offset;
482 unsigned short len;
483 unsigned char nfrag; /* which ah->frag[] continues this content, or 0 */
484};
485
486/*
487 * these are assigned from a pool held in the context.
488 * Both client and server mode uses them for http header analysis
489 */
490
491struct allocated_headers {
Andy Greende132b92015-12-25 13:48:20 +0800492 char *data; /* prepared by context init to point to dedicated storage */
493 /*
494 * the randomly ordered fragments, indexed by frag_index and
495 * lws_fragments->nfrag for continuation.
496 */
497 struct lws_fragments frags[WSI_TOKEN_COUNT * 2];
Andy Green8c1f6022016-01-26 20:56:56 +0800498 time_t assigned;
Andy Green3df58002015-12-25 12:44:12 +0800499 /*
500 * for each recognized token, frag_index says which frag[] his data
501 * starts in (0 means the token did not appear)
502 * the actual header data gets dumped as it comes in, into data[]
503 */
504 unsigned char frag_index[WSI_TOKEN_COUNT];
Andy Green3df58002015-12-25 12:44:12 +0800505#ifndef LWS_NO_CLIENT
506 char initial_handshake_hash_base64[30];
507 unsigned short c_port;
508#endif
Andy Greenaa775fd2015-12-26 08:56:58 +0800509
510 unsigned short pos;
511 unsigned char in_use;
512 unsigned char nfrag;
Andy Green3df58002015-12-25 12:44:12 +0800513};
514
Andy Greend3a55052016-01-19 03:34:24 +0800515/*
516 * so we can have n connections being serviced simultaneously,
517 * these things need to be isolated per-thread.
518 */
519
520struct lws_context_per_thread {
Andy Green8c1f6022016-01-26 20:56:56 +0800521#if LWS_MAX_SMP > 1
522 pthread_mutex_t lock;
523#endif
Andy Greend3a55052016-01-19 03:34:24 +0800524 struct lws_pollfd *fds;
525 struct lws *rx_draining_ext_list;
526 struct lws *tx_draining_ext_list;
Andy Green8c1f6022016-01-26 20:56:56 +0800527 struct lws *timeout_list;
528 void *http_header_data;
529 struct allocated_headers *ah_pool;
530 struct lws *ah_wait_list;
531 int ah_wait_list_length;
Andy Greend3a55052016-01-19 03:34:24 +0800532#ifdef LWS_OPENSSL_SUPPORT
533 struct lws *pending_read_list; /* linked list */
534#endif
Andy Green8c1f6022016-01-26 20:56:56 +0800535#ifndef LWS_NO_SERVER
536 struct lws *wsi_listening;
537#endif
Andy Green86ed65f2016-02-14 09:27:41 +0800538#if defined(LWS_USE_LIBEV)
539 struct ev_loop *io_loop_ev;
540#endif
541#if defined(LWS_USE_LIBUV)
542 uv_loop_t *io_loop_uv;
543 uv_signal_t signals[8];
544#endif
545#if defined(LWS_USE_LIBEV)
546 struct lws_io_watcher w_accept;
547#endif
548#if defined(LWS_USE_LIBEV) || defined(LWS_USE_LIBUV)
549 struct lws_signal_watcher w_sigint;
550 unsigned char ev_loop_foreign:1;
551#endif
Andy Green8c1f6022016-01-26 20:56:56 +0800552 lws_sockfd_type lserv_fd;
553
554 unsigned long count_conns;
Andy Greend3a55052016-01-19 03:34:24 +0800555 /*
556 * usable by anything in the service code, but only if the scope
557 * does not last longer than the service action (since next service
558 * of any socket can likewise use it and overwrite)
559 */
560 unsigned char *serv_buf;
561#ifdef _WIN32
562 WSAEVENT *events;
563#else
564 int dummy_pipe_fds[2];
565#endif
Andy Green8c1f6022016-01-26 20:56:56 +0800566 unsigned int fds_count;
567
568 short ah_count_in_use;
Andy Greend3a55052016-01-19 03:34:24 +0800569};
570
571/*
572 * the rest is managed per-context, that includes
573 *
574 * - processwide single fd -> wsi lookup
575 * - contextwide headers pool
576 * - contextwide ssl context
577 * - contextwide proxy
578 */
579
Andy Green4b85c1d2015-12-04 11:08:32 +0800580struct lws_context {
Andy Greenaa775fd2015-12-26 08:56:58 +0800581 time_t last_timeout_check_s;
582 struct lws_plat_file_ops fops;
Andy Greend3a55052016-01-19 03:34:24 +0800583 struct lws_context_per_thread pt[LWS_MAX_SMP];
Bud Davis229bfec2015-01-30 10:13:01 +0800584#ifdef _WIN32
585/* different implementation between unix and windows */
Andy Green3ef579b2015-12-04 09:23:56 +0800586 struct lws_fd_hashtable fd_hashtable[FD_HASHTABLE_MODULUS];
Bud Davis229bfec2015-01-30 10:13:01 +0800587#else
Andy Green4b85c1d2015-12-04 11:08:32 +0800588 struct lws **lws_lookup; /* fd to wsi */
Bud Davis229bfec2015-01-30 10:13:01 +0800589#endif
Mattias Lundberg03bb8f92014-02-18 10:06:57 +0100590 const char *iface;
Andy Greenaa775fd2015-12-26 08:56:58 +0800591 const struct lws_token_limits *token_limits;
592 void *user_space;
Andy Greend738f842016-01-19 04:32:14 +0800593
Andy Greenaa775fd2015-12-26 08:56:58 +0800594 const struct lws_protocols *protocols;
Andy Greend3a55052016-01-19 03:34:24 +0800595
Andy Greenaa775fd2015-12-26 08:56:58 +0800596#ifdef LWS_OPENSSL_SUPPORT
597 SSL_CTX *ssl_ctx;
598 SSL_CTX *ssl_client_ctx;
Andy Greenaa775fd2015-12-26 08:56:58 +0800599#endif
600#ifndef LWS_NO_EXTENSIONS
601 const struct lws_extension *extensions;
602#endif
Andy Green86ed65f2016-02-14 09:27:41 +0800603#if defined(LWS_USE_LIBEV)
604 lws_ev_signal_cb_t * lws_ev_sigint_cb;
605#endif
606#if defined(LWS_USE_LIBUV)
607 lws_uv_signal_cb_t * lws_uv_sigint_cb;
608#endif
Andy Greenaa775fd2015-12-26 08:56:58 +0800609 char http_proxy_address[128];
610 char proxy_basic_auth_token[128];
611 char canonical_hostname[128];
612#ifdef LWS_LATENCY
613 unsigned long worst_latency;
614 char worst_latency_info[256];
615#endif
Andy Greenb8b247d2013-01-22 07:20:08 +0800616
Andy Greenaa775fd2015-12-26 08:56:58 +0800617 int max_fds;
618 int listen_port;
Andy Green86ed65f2016-02-14 09:27:41 +0800619#if defined(LWS_USE_LIBEV) || defined(LWS_USE_LIBUV)
Andy Greenaa775fd2015-12-26 08:56:58 +0800620 int use_ev_sigint;
621#endif
Andy Green24cba922013-01-19 13:56:10 +0800622 int started_with_parent;
623
Andy Green44eee682011-02-10 09:32:24 +0000624 int fd_random;
Andy Green54806b12015-12-17 17:03:59 +0800625 int lserv_mod;
Andy Green86ed65f2016-02-14 09:27:41 +0800626 int count_wsi_allocated;
Andy Greenaa775fd2015-12-26 08:56:58 +0800627 unsigned int http_proxy_port;
628 unsigned int options;
Andy Greend3a55052016-01-19 03:34:24 +0800629 unsigned int fd_limit_per_thread;
Andy Green6ee372f2012-04-09 15:09:01 +0800630
Patrick Gansterer1ee57f62014-03-06 11:57:50 +0100631 /*
632 * set to the Thread ID that's doing the service loop just before entry
633 * to poll indicates service thread likely idling in poll()
634 * volatile because other threads may check it as part of processing
635 * for pollfd event change.
636 */
637 volatile int service_tid;
Andy Greenc35b36b2015-12-24 13:00:54 +0800638 int service_tid_detected;
Patrick Gansterer1ee57f62014-03-06 11:57:50 +0100639
Andy Greenaa775fd2015-12-26 08:56:58 +0800640 int count_protocols;
Andy Greena690cd02013-02-09 12:25:31 +0800641 int ka_time;
642 int ka_probes;
643 int ka_interval;
644
Andy Greenb45993c2010-12-18 15:13:50 +0000645#ifdef LWS_OPENSSL_SUPPORT
646 int use_ssl;
James Devine5b34c972013-12-14 11:41:29 +0800647 int allow_non_ssl_on_ssl_port;
joseph.urciuoli4d9c8fc2014-10-16 08:53:19 +0800648 unsigned int user_supplied_ssl_ctx:1;
Andy Greend3a55052016-01-19 03:34:24 +0800649#define lws_ssl_anybody_has_buffered_read(w) \
650 (w->context->use_ssl && \
651 w->context->pt[(int)w->tsi].pending_read_list)
652#define lws_ssl_anybody_has_buffered_read_tsi(c, t) \
653 (c->use_ssl && \
654 c->pt[(int)t].pending_read_list)
Andy Green52815602015-01-29 08:36:18 +0800655#else
656#define lws_ssl_anybody_has_buffered_read(ctx) (0)
Andy Greend3a55052016-01-19 03:34:24 +0800657#define lws_ssl_anybody_has_buffered_read_tsi(ctx, t) (0)
Andy Greenb45993c2010-12-18 15:13:50 +0000658#endif
Andy Green40110e82015-12-14 08:52:03 +0800659
Andy Green3df58002015-12-25 12:44:12 +0800660 short max_http_header_data;
661 short max_http_header_pool;
Andy Greend3a55052016-01-19 03:34:24 +0800662 short count_threads;
Andy Green9a9d5ea2016-01-18 11:49:41 +0800663
664 unsigned int being_destroyed:1;
Andy Green86ed65f2016-02-14 09:27:41 +0800665 unsigned int requested_kill:1;
Andy Greenb45993c2010-12-18 15:13:50 +0000666};
667
Andy Green86ed65f2016-02-14 09:27:41 +0800668LWS_EXTERN void
669lws_close_free_wsi_final(struct lws *wsi);
670LWS_EXTERN void
671lws_libuv_closehandle(struct lws *wsi);
672
Andy Greena717df22014-04-11 13:14:37 +0800673enum {
674 LWS_EV_READ = (1 << 0),
675 LWS_EV_WRITE = (1 << 1),
676 LWS_EV_START = (1 << 2),
677 LWS_EV_STOP = (1 << 3),
Andy Green86ed65f2016-02-14 09:27:41 +0800678
679 LWS_EV_PREPARE_DELETION = (1 << 31),
Andy Greena717df22014-04-11 13:14:37 +0800680};
681
Andy Green86ed65f2016-02-14 09:27:41 +0800682#if defined(LWS_USE_LIBEV)
Andy Greena717df22014-04-11 13:14:37 +0800683LWS_EXTERN void
Andy Green11c05bf2015-12-16 18:19:08 +0800684lws_libev_accept(struct lws *new_wsi, lws_sockfd_type accept_fd);
Andy Greena717df22014-04-11 13:14:37 +0800685LWS_EXTERN void
Andy Green11c05bf2015-12-16 18:19:08 +0800686lws_libev_io(struct lws *wsi, int flags);
Andy Greena717df22014-04-11 13:14:37 +0800687LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +0800688lws_libev_init_fd_table(struct lws_context *context);
Andy Greena717df22014-04-11 13:14:37 +0800689LWS_EXTERN void
Andy Green86ed65f2016-02-14 09:27:41 +0800690lws_libev_destroyloop(struct lws_context *context, int tsi);
691LWS_EXTERN void
692lws_libev_run(const struct lws_context *context, int tsi);
693#define LWS_LIBEV_ENABLED(context) (context->options & LWS_SERVER_OPTION_LIBEV)
694LWS_EXTERN void lws_feature_status_libev(struct lws_context_creation_info *info);
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800695#else
Andy Green86ed65f2016-02-14 09:27:41 +0800696#define lws_libev_accept(_a, _b) ((void) 0)
697#define lws_libev_io(_a, _b) ((void) 0)
698#define lws_libev_init_fd_table(_a) (0)
699#define lws_libev_run(_a, _b) ((void) 0)
700#define lws_libev_destroyloop(_a, _b) ((void) 0)
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800701#define LWS_LIBEV_ENABLED(context) (0)
Andy Green86ed65f2016-02-14 09:27:41 +0800702#if LWS_POSIX
Andy Greena717df22014-04-11 13:14:37 +0800703#define lws_feature_status_libev(_a) \
704 lwsl_notice("libev support not compiled in\n")
Andy Green8c0d3c02015-11-02 20:34:12 +0800705#else
706#define lws_feature_status_libev(_a)
707#endif
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800708#endif
709
Andy Green86ed65f2016-02-14 09:27:41 +0800710#if defined(LWS_USE_LIBUV)
711LWS_EXTERN void
712lws_libuv_accept(struct lws *new_wsi, lws_sockfd_type accept_fd);
713LWS_EXTERN void
714lws_libuv_io(struct lws *wsi, int flags);
715LWS_EXTERN int
716lws_libuv_init_fd_table(struct lws_context *context);
717LWS_EXTERN void
718lws_libuv_run(const struct lws_context *context, int tsi);
719LWS_EXTERN void
720lws_libuv_destroyloop(struct lws_context *context, int tsi);
721#define LWS_LIBUV_ENABLED(context) (context->options & LWS_SERVER_OPTION_LIBUV)
722LWS_EXTERN void lws_feature_status_libuv(struct lws_context_creation_info *info);
723#else
724#define lws_libuv_accept(_a, _b) ((void) 0)
725#define lws_libuv_io(_a, _b) ((void) 0)
726#define lws_libuv_init_fd_table(_a) (0)
727#define lws_libuv_run(_a, _b) ((void) 0)
728#define lws_libuv_destroyloop(_a, _b) ((void) 0)
729#define LWS_LIBUV_ENABLED(context) (0)
730#if LWS_POSIX
731#define lws_feature_status_libuv(_a) \
732 lwsl_notice("libuv support not compiled in\n")
733#else
734#define lws_feature_status_libuv(_a)
735#endif
736#endif
737
738
Andy Green055f2972014-03-24 16:09:25 +0800739#ifdef LWS_USE_IPV6
Andy Greendc8a3a82015-12-06 09:15:27 +0800740#define LWS_IPV6_ENABLED(context) \
741 (!(context->options & LWS_SERVER_OPTION_DISABLE_IPV6))
James Devine3f13ea22014-03-24 16:09:25 +0800742#else
743#define LWS_IPV6_ENABLED(context) (0)
744#endif
Andrew Canaday9769f4f2014-03-23 13:25:07 +0800745
Andy Greenb1a9e502013-11-10 15:15:21 +0800746enum uri_path_states {
747 URIPS_IDLE,
748 URIPS_SEEN_SLASH,
749 URIPS_SEEN_SLASH_DOT,
750 URIPS_SEEN_SLASH_DOT_DOT,
751};
752
753enum uri_esc_states {
754 URIES_IDLE,
755 URIES_SEEN_PERCENT,
756 URIES_SEEN_PERCENT_H1,
757};
Andy Greenf3d3b402011-02-09 07:16:34 +0000758
Andy Green024eb6c2014-10-08 12:00:53 +0800759/* notice that these union members:
Andy Green40110e82015-12-14 08:52:03 +0800760 *
Andy Green024eb6c2014-10-08 12:00:53 +0800761 * hdr
762 * http
763 * http2
Andy Green40110e82015-12-14 08:52:03 +0800764 *
Andy Green024eb6c2014-10-08 12:00:53 +0800765 * all have a pointer to allocated_headers struct as their first member.
Andy Green40110e82015-12-14 08:52:03 +0800766 *
Andy Green024eb6c2014-10-08 12:00:53 +0800767 * It means for allocated_headers access, the three union paths can all be
Peter Pentchevbb085da2015-12-03 15:55:11 +0200768 * used interchangeably to access the same data
Andy Green024eb6c2014-10-08 12:00:53 +0800769 */
770
Andy Green84fd9492013-11-09 11:40:32 +0800771struct _lws_http_mode_related {
Andy Green024eb6c2014-10-08 12:00:53 +0800772 /* MUST be first in struct */
Andy Green84fd9492013-11-09 11:40:32 +0800773 struct allocated_headers *ah; /* mirroring _lws_header_related */
Andy Green8c1f6022016-01-26 20:56:56 +0800774 struct lws *ah_wait_list;
775 struct lws *new_wsi_list;
Andy Green84fd9492013-11-09 11:40:32 +0800776 unsigned long filepos;
777 unsigned long filelen;
Andy Greenaa775fd2015-12-26 08:56:58 +0800778 lws_filefd_type fd;
kapejodce64fb02013-11-19 13:38:16 +0100779
Andrew Canadayafe26cf2014-07-13 01:07:36 -0400780 enum http_version request_version;
781 enum http_connection_type connection_type;
Andy Green2cd30742015-11-02 13:10:33 +0800782 unsigned int content_length;
783 unsigned int content_remain;
Andy Green84fd9492013-11-09 11:40:32 +0800784};
785
Andy Green024eb6c2014-10-08 12:00:53 +0800786#ifdef LWS_USE_HTTP2
787
788enum lws_http2_settings {
789 LWS_HTTP2_SETTINGS__HEADER_TABLE_SIZE = 1,
790 LWS_HTTP2_SETTINGS__ENABLE_PUSH,
791 LWS_HTTP2_SETTINGS__MAX_CONCURRENT_STREAMS,
792 LWS_HTTP2_SETTINGS__INITIAL_WINDOW_SIZE,
793 LWS_HTTP2_SETTINGS__MAX_FRAME_SIZE,
794 LWS_HTTP2_SETTINGS__MAX_HEADER_LIST_SIZE,
Andy Green40110e82015-12-14 08:52:03 +0800795
Andy Green024eb6c2014-10-08 12:00:53 +0800796 LWS_HTTP2_SETTINGS__COUNT /* always last */
Andy Greena54f2322014-09-30 09:43:14 +0800797};
798
Andy Green024eb6c2014-10-08 12:00:53 +0800799enum lws_http2_wellknown_frame_types {
800 LWS_HTTP2_FRAME_TYPE_DATA,
801 LWS_HTTP2_FRAME_TYPE_HEADERS,
802 LWS_HTTP2_FRAME_TYPE_PRIORITY,
803 LWS_HTTP2_FRAME_TYPE_RST_STREAM,
804 LWS_HTTP2_FRAME_TYPE_SETTINGS,
805 LWS_HTTP2_FRAME_TYPE_PUSH_PROMISE,
806 LWS_HTTP2_FRAME_TYPE_PING,
807 LWS_HTTP2_FRAME_TYPE_GOAWAY,
808 LWS_HTTP2_FRAME_TYPE_WINDOW_UPDATE,
809 LWS_HTTP2_FRAME_TYPE_CONTINUATION,
Andy Green40110e82015-12-14 08:52:03 +0800810
Andy Green024eb6c2014-10-08 12:00:53 +0800811 LWS_HTTP2_FRAME_TYPE_COUNT /* always last */
812};
813
Andy Green91b05892014-10-17 08:38:44 +0800814enum lws_http2_flags {
815 LWS_HTTP2_FLAG_END_STREAM = 1,
816 LWS_HTTP2_FLAG_END_HEADERS = 4,
817 LWS_HTTP2_FLAG_PADDED = 8,
818 LWS_HTTP2_FLAG_PRIORITY = 0x20,
819
820 LWS_HTTP2_FLAG_SETTINGS_ACK = 1,
821};
822
Andy Green024eb6c2014-10-08 12:00:53 +0800823#define LWS_HTTP2_STREAM_ID_MASTER 0
824#define LWS_HTTP2_FRAME_HEADER_LENGTH 9
825#define LWS_HTTP2_SETTINGS_LENGTH 6
826
827struct http2_settings {
828 unsigned int setting[LWS_HTTP2_SETTINGS__COUNT];
829};
830
Andy Greenecc2e722014-10-09 16:57:47 +0800831enum http2_hpack_state {
Andy Green40110e82015-12-14 08:52:03 +0800832
Andy Green200f3852014-10-18 12:23:05 +0800833 /* optional before first header block */
834 HPKS_OPT_PADDING,
835 HKPS_OPT_E_DEPENDENCY,
836 HKPS_OPT_WEIGHT,
Andy Green40110e82015-12-14 08:52:03 +0800837
Andy Green200f3852014-10-18 12:23:05 +0800838 /* header block */
Andy Greenecc2e722014-10-09 16:57:47 +0800839 HPKS_TYPE,
Andy Green40110e82015-12-14 08:52:03 +0800840
Andy Green2add6342014-10-12 08:38:16 +0800841 HPKS_IDX_EXT,
Andy Green40110e82015-12-14 08:52:03 +0800842
Andy Greenecc2e722014-10-09 16:57:47 +0800843 HPKS_HLEN,
844 HPKS_HLEN_EXT,
845
846 HPKS_DATA,
Andy Green40110e82015-12-14 08:52:03 +0800847
Andy Green200f3852014-10-18 12:23:05 +0800848 /* optional after last header block */
849 HKPS_OPT_DISCARD_PADDING,
Andy Greenecc2e722014-10-09 16:57:47 +0800850};
851
Andy Green2add6342014-10-12 08:38:16 +0800852enum http2_hpack_type {
853 HPKT_INDEXED_HDR_7,
854 HPKT_INDEXED_HDR_6_VALUE_INCR,
855 HPKT_LITERAL_HDR_VALUE_INCR,
856 HPKT_INDEXED_HDR_4_VALUE,
857 HPKT_LITERAL_HDR_VALUE,
858 HPKT_SIZE_5
859};
860
Andy Green200f3852014-10-18 12:23:05 +0800861struct hpack_dt_entry {
862 int token; /* additions that don't map to a token are ignored */
863 int arg_offset;
864 int arg_len;
865};
866
867struct hpack_dynamic_table {
868 struct hpack_dt_entry *entries;
869 char *args;
870 int pos;
871 int next;
872 int num_entries;
873 int args_length;
874};
875
Andy Green024eb6c2014-10-08 12:00:53 +0800876struct _lws_http2_related {
Andy Green40110e82015-12-14 08:52:03 +0800877 /*
Andy Green024eb6c2014-10-08 12:00:53 +0800878 * having this first lets us also re-use all HTTP union code
879 * and in turn, http_mode_related has allocated headers in right
880 * place so we can use the header apis on the wsi directly still
881 */
882 struct _lws_http_mode_related http; /* MUST BE FIRST IN STRUCT */
883
884 struct http2_settings my_settings;
885 struct http2_settings peer_settings;
Andy Green40110e82015-12-14 08:52:03 +0800886
Andy Green4b85c1d2015-12-04 11:08:32 +0800887 struct lws *parent_wsi;
888 struct lws *next_child_wsi;
Andy Green024eb6c2014-10-08 12:00:53 +0800889
Andy Green200f3852014-10-18 12:23:05 +0800890 struct hpack_dynamic_table *hpack_dyn_table;
Andy Greenaa775fd2015-12-26 08:56:58 +0800891 struct lws *stream_wsi;
892 unsigned char ping_payload[8];
893 unsigned char one_setting[LWS_HTTP2_SETTINGS_LENGTH];
Andy Green40110e82015-12-14 08:52:03 +0800894
Andy Green024eb6c2014-10-08 12:00:53 +0800895 unsigned int count;
Andy Green024eb6c2014-10-08 12:00:53 +0800896 unsigned int length;
897 unsigned int stream_id;
Andy Greenaa775fd2015-12-26 08:56:58 +0800898 enum http2_hpack_state hpack;
899 enum http2_hpack_type hpack_type;
900 unsigned int header_index;
901 unsigned int hpack_len;
902 unsigned int hpack_e_dep;
903 int tx_credit;
904 unsigned int my_stream_id;
905 unsigned int child_count;
906 int my_priority;
Andy Green67112662016-01-11 11:34:01 +0800907
Andy Green91b05892014-10-17 08:38:44 +0800908 unsigned int END_STREAM:1;
909 unsigned int END_HEADERS:1;
Andy Green1cea5812014-10-19 07:36:20 +0800910 unsigned int send_END_STREAM:1;
Andy Green7df53c52014-10-22 15:37:28 +0800911 unsigned int GOING_AWAY;
912 unsigned int requested_POLLOUT:1;
Andy Green97ee57f2014-10-29 09:39:08 +0800913 unsigned int waiting_tx_credit:1;
Andy Greenecc2e722014-10-09 16:57:47 +0800914 unsigned int huff:1;
915 unsigned int value:1;
Andy Green40110e82015-12-14 08:52:03 +0800916
Andy Greenaa775fd2015-12-26 08:56:58 +0800917 unsigned short round_robin_POLLOUT;
918 unsigned short count_POLLOUT_children;
919 unsigned short hpack_pos;
920
921 unsigned char type;
922 unsigned char flags;
923 unsigned char frame_state;
924 unsigned char padding;
925 unsigned char hpack_m;
Andy Green024eb6c2014-10-08 12:00:53 +0800926 unsigned char initialized;
Andy Green024eb6c2014-10-08 12:00:53 +0800927};
928
Andy Green7df53c52014-10-22 15:37:28 +0800929#define HTTP2_IS_TOPLEVEL_WSI(wsi) (!wsi->u.http2.parent_wsi)
Andy Green024eb6c2014-10-08 12:00:53 +0800930
931#endif
932
Andy Green623a98d2013-01-21 11:04:23 +0800933struct _lws_header_related {
Andy Green024eb6c2014-10-08 12:00:53 +0800934 /* MUST be first in struct */
Andy Green16ab3182013-02-10 18:02:31 +0800935 struct allocated_headers *ah;
Andy Green8c1f6022016-01-26 20:56:56 +0800936 struct lws *ah_wait_list;
Andy Greenb1a9e502013-11-10 15:15:21 +0800937 enum uri_path_states ups;
938 enum uri_esc_states ues;
Andy Greende132b92015-12-25 13:48:20 +0800939 short lextable_pos;
940 unsigned short current_token_limit;
Andy Greenb1a9e502013-11-10 15:15:21 +0800941 char esc_stash;
Andy Green3ba035d2015-12-18 15:40:03 +0800942 char post_literal_equal;
Andy Greende132b92015-12-25 13:48:20 +0800943 unsigned char parser_state; /* enum lws_token_indexes */
Andy Green809d69a2016-01-14 11:37:56 +0800944 char redirects;
Andy Green4019aab2016-01-30 11:43:10 +0800945 char more_rx_waiting;
Andy Green623a98d2013-01-21 11:04:23 +0800946};
947
948struct _lws_websocket_related {
Andy Green67112662016-01-11 11:34:01 +0800949 char *rx_ubuf;
Andy Green4019aab2016-01-30 11:43:10 +0800950 unsigned int rx_ubuf_alloc;
Andy Green67112662016-01-11 11:34:01 +0800951 struct lws *rx_draining_ext_list;
952 struct lws *tx_draining_ext_list;
Andy Greende132b92015-12-25 13:48:20 +0800953 size_t rx_packet_length;
Andy Green67112662016-01-11 11:34:01 +0800954 unsigned int rx_ubuf_head;
955 unsigned char mask[4];
Andy Green1fb95e82015-12-26 17:20:34 +0800956 /* Also used for close content... control opcode == < 128 */
Andy Green67112662016-01-11 11:34:01 +0800957 unsigned char ping_payload_buf[128 - 3 + LWS_PRE];
Andy Green1fb95e82015-12-26 17:20:34 +0800958
Andy Greenaa775fd2015-12-26 08:56:58 +0800959 unsigned char ping_payload_len;
Andy Green67112662016-01-11 11:34:01 +0800960 unsigned char mask_idx;
Andy Green623a98d2013-01-21 11:04:23 +0800961 unsigned char opcode;
Andy Green623a98d2013-01-21 11:04:23 +0800962 unsigned char rsv;
Andy Green67112662016-01-11 11:34:01 +0800963 unsigned char rsv_first_msg;
Andy Green1fb95e82015-12-26 17:20:34 +0800964 /* zero if no info, or length including 2-byte close code */
965 unsigned char close_in_ping_buffer_len;
Andy Green0c7b38b2015-12-29 09:46:03 +0800966 unsigned char utf8;
Andy Green67112662016-01-11 11:34:01 +0800967 unsigned char stashed_write_type;
968 unsigned char tx_draining_stashed_wp;
Andy Greende132b92015-12-25 13:48:20 +0800969
970 unsigned int final:1;
Andy Greend91d5e82013-02-10 16:00:47 +0800971 unsigned int frame_is_binary:1;
Andy Greend91d5e82013-02-10 16:00:47 +0800972 unsigned int all_zero_nonce:1;
Andy Greend91d5e82013-02-10 16:00:47 +0800973 unsigned int this_frame_masked:1;
Andy Green1f4267b2013-10-17 08:09:19 +0800974 unsigned int inside_frame:1; /* next write will be more of frame */
975 unsigned int clean_buffer:1; /* buffer not rewritten by extension */
Andy Green40d5abc2015-04-17 20:29:58 +0800976 unsigned int payload_is_close:1; /* process as PONG, but it is close */
Andy Greenba38a7e2015-12-25 13:14:09 +0800977 unsigned int ping_pending_flag:1;
Andy Green977734e2015-12-28 16:51:08 +0800978 unsigned int continuation_possible:1;
Andy Green91d624e2015-12-28 17:05:40 +0800979 unsigned int owed_a_fin:1;
Andy Green9b81d3c2015-12-29 12:28:48 +0800980 unsigned int check_utf8:1;
981 unsigned int defeat_check_utf8:1;
Andy Green67112662016-01-11 11:34:01 +0800982 unsigned int pmce_compressed_message:1;
983 unsigned int stashed_write_pending:1;
984 unsigned int rx_draining_ext:1;
985 unsigned int tx_draining_ext:1;
Andy Green623a98d2013-01-21 11:04:23 +0800986};
987
Andy Green4b85c1d2015-12-04 11:08:32 +0800988struct lws {
Andy Green623a98d2013-01-21 11:04:23 +0800989
Andy Greenaa775fd2015-12-26 08:56:58 +0800990 /* structs */
991 /* members with mutually exclusive lifetimes are unionized */
992
993 union u {
994 struct _lws_http_mode_related http;
995#ifdef LWS_USE_HTTP2
996 struct _lws_http2_related http2;
997#endif
998 struct _lws_header_related hdr;
999 struct _lws_websocket_related ws;
1000 } u;
1001
Andy Green623a98d2013-01-21 11:04:23 +08001002 /* lifetime members */
1003
Andy Green86ed65f2016-02-14 09:27:41 +08001004#if defined(LWS_USE_LIBEV) || defined(LWS_USE_LIBUV)
Andy Green40110e82015-12-14 08:52:03 +08001005 struct lws_io_watcher w_read;
Andy Green86ed65f2016-02-14 09:27:41 +08001006#endif
1007#if defined(LWS_USE_LIBEV)
Andy Green40110e82015-12-14 08:52:03 +08001008 struct lws_io_watcher w_write;
Andy Green86ed65f2016-02-14 09:27:41 +08001009#endif
Andy Greende132b92015-12-25 13:48:20 +08001010 time_t pending_timeout_limit;
Andy Greenaa775fd2015-12-26 08:56:58 +08001011
1012 /* pointers */
1013
Andy Green40110e82015-12-14 08:52:03 +08001014 struct lws_context *context;
Andy Green4b85c1d2015-12-04 11:08:32 +08001015 const struct lws_protocols *protocol;
Andy Greend738f842016-01-19 04:32:14 +08001016 struct lws *timeout_list;
1017 struct lws **timeout_list_prev;
Andy Greende132b92015-12-25 13:48:20 +08001018 void *user_space;
1019 /* rxflow handling */
1020 unsigned char *rxflow_buffer;
1021 /* truncated send handling */
1022 unsigned char *trunc_alloc; /* non-NULL means buffering in progress */
Andy Green3182ece2013-01-20 17:08:31 +08001023#ifndef LWS_NO_EXTENSIONS
Andy Greend2ac22c2015-12-11 10:45:35 +08001024 const struct lws_extension *active_extensions[LWS_MAX_EXTENSIONS_ACTIVE];
Andy Green67112662016-01-11 11:34:01 +08001025 void *act_ext_user[LWS_MAX_EXTENSIONS_ACTIVE];
Andy Green3182ece2013-01-20 17:08:31 +08001026#endif
Andy Greende132b92015-12-25 13:48:20 +08001027#ifdef LWS_OPENSSL_SUPPORT
1028 SSL *ssl;
1029 BIO *client_bio;
1030 struct lws *pending_read_list_prev, *pending_read_list_next;
1031#endif
Andy Greenaa775fd2015-12-26 08:56:58 +08001032#ifdef LWS_LATENCY
1033 unsigned long action_start;
1034 unsigned long latency_start;
1035#endif
1036 /* pointer / int */
Andy Green3b193862015-11-02 13:13:44 +08001037 lws_sockfd_type sock;
Andy Greende132b92015-12-25 13:48:20 +08001038
Andy Greenaa775fd2015-12-26 08:56:58 +08001039 /* ints */
Andy Greendfb23042013-01-17 12:26:48 +08001040 int position_in_fds_table;
Andy Green024eb6c2014-10-08 12:00:53 +08001041 int rxflow_len;
1042 int rxflow_pos;
Andy Green54806b12015-12-17 17:03:59 +08001043 unsigned int trunc_alloc_len; /* size of malloc */
1044 unsigned int trunc_offset; /* where we are in terms of spilling */
1045 unsigned int trunc_len; /* how much is buffered */
Andy Green2764eba2013-12-09 14:16:17 +08001046
Andy Greende132b92015-12-25 13:48:20 +08001047 unsigned int hdr_parsing_completed:1;
1048 unsigned int user_space_externally_allocated:1;
1049 unsigned int socket_is_permanently_unusable:1;
1050 unsigned int rxflow_change_to:2;
1051#ifndef LWS_NO_EXTENSIONS
1052 unsigned int extension_data_pending:1;
1053#endif
1054#ifdef LWS_OPENSSL_SUPPORT
1055 unsigned int use_ssl:2;
1056 unsigned int upgraded:1;
1057#endif
Andy Greenaa775fd2015-12-26 08:56:58 +08001058#ifdef _WIN32
1059 unsigned int sock_send_blocking:1;
1060#endif
Andy Greende132b92015-12-25 13:48:20 +08001061
Andy Greenaa775fd2015-12-26 08:56:58 +08001062 /* chars */
Andy Greende132b92015-12-25 13:48:20 +08001063#ifndef LWS_NO_EXTENSIONS
Andy Green67112662016-01-11 11:34:01 +08001064 unsigned char count_act_ext;
Andy Greende132b92015-12-25 13:48:20 +08001065#endif
1066 unsigned char ietf_spec_revision;
1067 char mode; /* enum connection_mode */
1068 char state; /* enum lws_connection_states */
Andy Green8c1f6022016-01-26 20:56:56 +08001069 char state_pre_close;
Andy Greende132b92015-12-25 13:48:20 +08001070 char lws_rx_parse_state; /* enum lws_rx_parse_state */
1071 char rx_frame_type; /* enum lws_write_protocol */
1072 char pending_timeout; /* enum pending_timeout */
Andy Greend738f842016-01-19 04:32:14 +08001073 char pps; /* enum lws_pending_protocol_send */
Andy Greend3a55052016-01-19 03:34:24 +08001074 char tsi; /* thread service index we belong to */
Andy Green7c212cc2010-11-08 20:20:42 +00001075};
1076
Andy Green3d67f512014-04-03 07:29:50 +08001077LWS_EXTERN int log_level;
1078
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001079LWS_EXTERN void
Andy Green6b5de702015-12-15 21:15:58 +08001080lws_close_free_wsi(struct lws *wsi, enum lws_close_status);
Andy Green508946c2013-02-12 10:19:08 +08001081
Andy Green34f3dd22014-04-03 07:42:50 +08001082LWS_EXTERN int
Andy Green6b5de702015-12-15 21:15:58 +08001083remove_wsi_socket_from_fds(struct lws *wsi);
Andy Green024eb6c2014-10-08 12:00:53 +08001084LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001085lws_rxflow_cache(struct lws *wsi, unsigned char *buf, int n, int len);
Andy Green34f3dd22014-04-03 07:42:50 +08001086
Andy Greend636e352013-01-29 12:36:17 +08001087#ifndef LWS_LATENCY
Andy Greendc8a3a82015-12-06 09:15:27 +08001088static inline void
1089lws_latency(struct lws_context *context, struct lws *wsi, const char *action,
1090 int ret, int completion) {
Andy Green40110e82015-12-14 08:52:03 +08001091 do {
1092 (void)context; (void)wsi; (void)action; (void)ret;
1093 (void)completion;
Andy Greendc8a3a82015-12-06 09:15:27 +08001094 } while (0);
1095}
1096static inline void
1097lws_latency_pre(struct lws_context *context, struct lws *wsi) {
1098 do { (void)context; (void)wsi; } while (0);
1099}
Andy Greend636e352013-01-29 12:36:17 +08001100#else
1101#define lws_latency_pre(_context, _wsi) lws_latency(_context, _wsi, NULL, 0, 0)
1102extern void
Andy Greendc8a3a82015-12-06 09:15:27 +08001103lws_latency(struct lws_context *context, struct lws *wsi, const char *action,
1104 int ret, int completion);
Andy Greend636e352013-01-29 12:36:17 +08001105#endif
1106
Andy Greendc8a3a82015-12-06 09:15:27 +08001107LWS_EXTERN void
Andy Green6b5de702015-12-15 21:15:58 +08001108lws_set_protocol_write_pending(struct lws *wsi,
Andy Greendc8a3a82015-12-06 09:15:27 +08001109 enum lws_pending_protocol_send pend);
Andy Greene99a83c2016-01-20 16:56:06 +08001110LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001111lws_client_rx_sm(struct lws *wsi, unsigned char c);
Andy Green4739e5c2011-01-22 12:51:57 +00001112
Andy Greene99a83c2016-01-20 16:56:06 +08001113LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001114lws_parse(struct lws *wsi, unsigned char c);
Andy Greenb45993c2010-12-18 15:13:50 +00001115
Andy Greene99a83c2016-01-20 16:56:06 +08001116LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001117lws_http_action(struct lws *wsi);
Andy Green024eb6c2014-10-08 12:00:53 +08001118
1119LWS_EXTERN int
Andy Greendf736162011-01-18 15:39:02 +00001120lws_b64_selftest(void);
Andy Greenbfb051f2011-02-09 08:49:14 +00001121
Andy Green2cd30742015-11-02 13:10:33 +08001122#if defined(_WIN32) || defined(MBED_OPERATORS)
Andy Green4b85c1d2015-12-04 11:08:32 +08001123LWS_EXTERN struct lws *
Andy Green1fa76852015-12-14 11:17:16 +08001124wsi_from_fd(const struct lws_context *context, lws_sockfd_type fd);
Andy Green0d338332011-02-12 11:57:43 +00001125
Andy Green40110e82015-12-14 08:52:03 +08001126LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001127insert_wsi(struct lws_context *context, struct lws *wsi);
Bud Davis229bfec2015-01-30 10:13:01 +08001128
1129LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001130delete_from_fd(struct lws_context *context, lws_sockfd_type fd);
Bud Davis229bfec2015-01-30 10:13:01 +08001131#else
Andy Green40110e82015-12-14 08:52:03 +08001132#define wsi_from_fd(A,B) A->lws_lookup[B]
Andy Green8c1f6022016-01-26 20:56:56 +08001133#define insert_wsi(A,B) assert(A->lws_lookup[B->sock] == 0); A->lws_lookup[B->sock]=B
Bud Davis229bfec2015-01-30 10:13:01 +08001134#define delete_from_fd(A,B) A->lws_lookup[B]=0
1135#endif
1136
Andy Greene99a83c2016-01-20 16:56:06 +08001137LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Greendc8a3a82015-12-06 09:15:27 +08001138insert_wsi_socket_into_fds(struct lws_context *context, struct lws *wsi);
Darin Willitsc19456f2011-02-14 17:52:39 +00001139
Andy Greene99a83c2016-01-20 16:56:06 +08001140LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001141lws_issue_raw(struct lws *wsi, unsigned char *buf, size_t len);
Andy Greend44bf7f2011-03-06 10:29:38 +00001142
Andy Green95a7b5d2011-03-06 10:29:39 +00001143
Andy Greene99a83c2016-01-20 16:56:06 +08001144LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001145lws_service_timeout_check(struct lws *wsi, unsigned int sec);
Andy Greena41314f2011-05-23 10:00:03 +01001146
Andy Greene99a83c2016-01-20 16:56:06 +08001147LWS_EXTERN struct lws * LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001148lws_client_connect_2(struct lws *wsi);
Andy Greena41314f2011-05-23 10:00:03 +01001149
Andy Greene99a83c2016-01-20 16:56:06 +08001150LWS_VISIBLE struct lws * LWS_WARN_UNUSED_RESULT
1151lws_client_reset(struct lws *wsi, int ssl, const char *address, int port,
1152 const char *path, const char *host);
Andy Green809d69a2016-01-14 11:37:56 +08001153
Andy Greene99a83c2016-01-20 16:56:06 +08001154LWS_EXTERN struct lws * LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001155lws_create_new_server_wsi(struct lws_context *context);
Andy Greena41314f2011-05-23 10:00:03 +01001156
Andy Greene99a83c2016-01-20 16:56:06 +08001157LWS_EXTERN char * LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001158lws_generate_client_handshake(struct lws *wsi, char *pkt);
Andy Greena41314f2011-05-23 10:00:03 +01001159
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001160LWS_EXTERN int
Andy Green6b5de702015-12-15 21:15:58 +08001161lws_handle_POLLOUT_event(struct lws *wsi, struct lws_pollfd *pollfd);
Andy Green91b05892014-10-17 08:38:44 +08001162
Andy Greencdb9bf92014-04-12 10:07:02 +08001163/*
1164 * EXTENSIONS
1165 */
1166
Andy Green3182ece2013-01-20 17:08:31 +08001167#ifndef LWS_NO_EXTENSIONS
Andy Greencdb9bf92014-04-12 10:07:02 +08001168LWS_VISIBLE void
1169lws_context_init_extensions(struct lws_context_creation_info *info,
Andy Greendc8a3a82015-12-06 09:15:27 +08001170 struct lws_context *context);
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001171LWS_EXTERN int
Andy Greene99a83c2016-01-20 16:56:06 +08001172lws_any_extension_handled(struct lws *wsi, enum lws_extension_callback_reasons r,
Andy Green6ee372f2012-04-09 15:09:01 +08001173 void *v, size_t len);
Andy Greena41314f2011-05-23 10:00:03 +01001174
Andy Green2c24ec02014-04-02 19:45:42 +08001175LWS_EXTERN int
Andy Greene99a83c2016-01-20 16:56:06 +08001176lws_ext_cb_active(struct lws *wsi, int reason, void *buf, int len);
Andy Green2c24ec02014-04-02 19:45:42 +08001177LWS_EXTERN int
Andy Greene99a83c2016-01-20 16:56:06 +08001178lws_ext_cb_all_exts(struct lws_context *context, struct lws *wsi, int reason,
1179 void *arg, int len);
Andy Green67112662016-01-11 11:34:01 +08001180
Andy Green2c24ec02014-04-02 19:45:42 +08001181#else
Andy Green6b5de702015-12-15 21:15:58 +08001182#define lws_any_extension_handled(_a, _b, _c, _d) (0)
Andy Green67112662016-01-11 11:34:01 +08001183#define lws_ext_cb_active(_a, _b, _c, _d) (0)
Andy Green54806b12015-12-17 17:03:59 +08001184#define lws_ext_cb_all_exts(_a, _b, _c, _d, _e) (0)
Andy Greenb49a9952014-04-03 10:11:04 +08001185#define lws_issue_raw_ext_access lws_issue_raw
Andy Greencdb9bf92014-04-12 10:07:02 +08001186#define lws_context_init_extensions(_a, _b)
Andy Green3182ece2013-01-20 17:08:31 +08001187#endif
Andy Greena41314f2011-05-23 10:00:03 +01001188
Andy Greene99a83c2016-01-20 16:56:06 +08001189LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001190lws_client_interpret_server_handshake(struct lws *wsi);
Andy Greena41314f2011-05-23 10:00:03 +01001191
Andy Greene99a83c2016-01-20 16:56:06 +08001192LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001193lws_rx_sm(struct lws *wsi, unsigned char c);
Andy Greena41314f2011-05-23 10:00:03 +01001194
Andy Greene99a83c2016-01-20 16:56:06 +08001195LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Greendc8a3a82015-12-06 09:15:27 +08001196lws_issue_raw_ext_access(struct lws *wsi, unsigned char *buf, size_t len);
Andy Green09226502011-05-28 10:19:19 +01001197
Andy Green44c11612014-11-08 11:18:47 +08001198LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001199lws_union_transition(struct lws *wsi, enum connection_mode mode);
Andy Green44c11612014-11-08 11:18:47 +08001200
Andy Greene99a83c2016-01-20 16:56:06 +08001201LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Denis Osvald034e5142015-12-21 18:06:38 +01001202user_callback_handle_rxflow(lws_callback_function, struct lws *wsi,
Andy Green00c6d152015-12-17 07:54:44 +08001203 enum lws_callback_reasons reason, void *user,
1204 void *in, size_t len);
Andy Green024eb6c2014-10-08 12:00:53 +08001205#ifdef LWS_USE_HTTP2
Andy Green4b85c1d2015-12-04 11:08:32 +08001206LWS_EXTERN struct lws *lws_http2_get_network_wsi(struct lws *wsi);
1207struct lws * lws_http2_get_nth_child(struct lws *wsi, int n);
Andy Green024eb6c2014-10-08 12:00:53 +08001208LWS_EXTERN int
Andy Greendc8a3a82015-12-06 09:15:27 +08001209lws_http2_interpret_settings_payload(struct http2_settings *settings,
1210 unsigned char *buf, int len);
Andy Green024eb6c2014-10-08 12:00:53 +08001211LWS_EXTERN void lws_http2_init(struct http2_settings *settings);
1212LWS_EXTERN int
Andy Green11c05bf2015-12-16 18:19:08 +08001213lws_http2_parser(struct lws *wsi, unsigned char c);
Andy Greendc8a3a82015-12-06 09:15:27 +08001214LWS_EXTERN int lws_http2_do_pps_send(struct lws_context *context,
1215 struct lws *wsi);
1216LWS_EXTERN int lws_http2_frame_write(struct lws *wsi, int type, int flags,
1217 unsigned int sid, unsigned int len,
1218 unsigned char *buf);
Andy Green4b85c1d2015-12-04 11:08:32 +08001219LWS_EXTERN struct lws *
1220lws_http2_wsi_from_id(struct lws *wsi, unsigned int sid);
Andy Green11c05bf2015-12-16 18:19:08 +08001221LWS_EXTERN int lws_hpack_interpret(struct lws *wsi,
Andy Green2add6342014-10-12 08:38:16 +08001222 unsigned char c);
Andy Green917f43a2014-10-12 14:31:47 +08001223LWS_EXTERN int
Andy Green11c05bf2015-12-16 18:19:08 +08001224lws_add_http2_header_by_name(struct lws *wsi,
Andy Greendc8a3a82015-12-06 09:15:27 +08001225 const unsigned char *name,
1226 const unsigned char *value, int length,
1227 unsigned char **p, unsigned char *end);
Andy Green917f43a2014-10-12 14:31:47 +08001228LWS_EXTERN int
Andy Green11c05bf2015-12-16 18:19:08 +08001229lws_add_http2_header_by_token(struct lws *wsi,
Andy Green917f43a2014-10-12 14:31:47 +08001230 enum lws_token_indexes token,
Andy Greendc8a3a82015-12-06 09:15:27 +08001231 const unsigned char *value, int length,
1232 unsigned char **p, unsigned char *end);
Andy Green917f43a2014-10-12 14:31:47 +08001233LWS_EXTERN int
Andy Green11c05bf2015-12-16 18:19:08 +08001234lws_add_http2_header_status(struct lws *wsi,
Andy Greendc8a3a82015-12-06 09:15:27 +08001235 unsigned int code, unsigned char **p,
Andy Green917f43a2014-10-12 14:31:47 +08001236 unsigned char *end);
Andy Green7df53c52014-10-22 15:37:28 +08001237LWS_EXTERN
Andy Green4b85c1d2015-12-04 11:08:32 +08001238void lws_http2_configure_if_upgraded(struct lws *wsi);
Andy Green7df53c52014-10-22 15:37:28 +08001239#else
1240#define lws_http2_configure_if_upgraded(x)
Andy Green024eb6c2014-10-08 12:00:53 +08001241#endif
Andy Green706961d2013-01-17 16:50:35 +08001242
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001243LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001244lws_plat_set_socket_options(struct lws_context *context, lws_sockfd_type fd);
Andy Greena690cd02013-02-09 12:25:31 +08001245
Andy Greene99a83c2016-01-20 16:56:06 +08001246LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001247lws_allocate_header_table(struct lws *wsi);
Andy Green16ab3182013-02-10 18:02:31 +08001248
Andrew Canaday37718812014-11-07 11:20:59 +08001249LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001250lws_free_header_table(struct lws *wsi);
Andrew Canaday37718812014-11-07 11:20:59 +08001251
Andy Green4019aab2016-01-30 11:43:10 +08001252LWS_EXTERN void
1253lws_reset_header_table(struct lws *wsi);
1254
Andy Greene99a83c2016-01-20 16:56:06 +08001255LWS_EXTERN char * LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001256lws_hdr_simple_ptr(struct lws *wsi, enum lws_token_indexes h);
Andy Green16ab3182013-02-10 18:02:31 +08001257
Andy Greene99a83c2016-01-20 16:56:06 +08001258LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green11c05bf2015-12-16 18:19:08 +08001259lws_hdr_simple_create(struct lws *wsi, enum lws_token_indexes h, const char *s);
Andy Greenb5b23192013-02-11 17:13:32 +08001260
Andy Green2af4d5b2013-02-18 16:30:10 +08001261LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001262lws_ensure_user_space(struct lws *wsi);
Andy Green2af4d5b2013-02-18 16:30:10 +08001263
Andy Green158e8042014-04-02 14:25:10 +08001264LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001265lws_change_pollfd(struct lws *wsi, int _and, int _or);
Andy Green91f19d82013-12-21 11:18:34 +08001266
Andy Greenb5b23192013-02-11 17:13:32 +08001267#ifndef LWS_NO_SERVER
Andy Greene38031a2014-04-03 08:24:29 +08001268int lws_context_init_server(struct lws_context_creation_info *info,
Andy Greenaa775fd2015-12-26 08:56:58 +08001269 struct lws_context *context);
Andy Greend7340c12014-04-10 14:08:10 +08001270LWS_EXTERN int
Andy Greendc8a3a82015-12-06 09:15:27 +08001271handshake_0405(struct lws_context *context, struct lws *wsi);
Andy Greene99a83c2016-01-20 16:56:06 +08001272LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green67112662016-01-11 11:34:01 +08001273lws_interpret_incoming_packet(struct lws *wsi, unsigned char **buf, size_t len);
Andy Greencdb9bf92014-04-12 10:07:02 +08001274LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001275lws_server_get_canonical_hostname(struct lws_context *context,
Andy Greendc8a3a82015-12-06 09:15:27 +08001276 struct lws_context_creation_info *info);
Andy Greene38031a2014-04-03 08:24:29 +08001277#else
1278#define lws_context_init_server(_a, _b) (0)
Andy Green3ef579b2015-12-04 09:23:56 +08001279#define lws_interpret_incoming_packet(_a, _b, _c) (0)
Andy Greencdb9bf92014-04-12 10:07:02 +08001280#define lws_server_get_canonical_hostname(_a, _b)
Andy Greenb5b23192013-02-11 17:13:32 +08001281#endif
1282
1283#ifndef LWS_NO_DAEMONIZE
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001284LWS_EXTERN int get_daemonize_pid();
Andy Greencdb9bf92014-04-12 10:07:02 +08001285#else
1286#define get_daemonize_pid() (0)
Andy Greenb5b23192013-02-11 17:13:32 +08001287#endif
1288
Andy Green2cd30742015-11-02 13:10:33 +08001289#if !defined(MBED_OPERATORS)
Andy Greene99a83c2016-01-20 16:56:06 +08001290LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Greendc8a3a82015-12-06 09:15:27 +08001291interface_to_sa(struct lws_context *context, const char *ifname,
1292 struct sockaddr_in *addr, size_t addrlen);
Andy Green2cd30742015-11-02 13:10:33 +08001293#endif
Andy Greenb25b85f2014-04-03 23:34:09 +08001294LWS_EXTERN void lwsl_emit_stderr(int level, const char *line);
1295
Andy Green1a308e42014-04-08 07:26:30 +01001296enum lws_ssl_capable_status {
1297 LWS_SSL_CAPABLE_ERROR = -1,
1298 LWS_SSL_CAPABLE_MORE_SERVICE = -2,
1299};
1300
Darin Willitsc19456f2011-02-14 17:52:39 +00001301#ifndef LWS_OPENSSL_SUPPORT
Andy Greencdb9bf92014-04-12 10:07:02 +08001302#define LWS_SSL_ENABLED(context) (0)
Andy Greenc57037a2014-04-03 10:17:00 +08001303#define lws_context_init_server_ssl(_a, _b) (0)
1304#define lws_ssl_destroy(_a)
Andy Green2eedea92014-04-03 14:33:48 +08001305#define lws_context_init_http2_ssl(_a)
Andy Green02138122014-04-06 06:26:35 +01001306#define lws_ssl_capable_read lws_ssl_capable_read_no_ssl
1307#define lws_ssl_capable_write lws_ssl_capable_write_no_ssl
=?UTF-8?q?Jos=C3=A9=20Luis=20Mill=C3=A1n?=4c0ba022015-08-19 16:23:33 +02001308#define lws_ssl_pending lws_ssl_pending_no_ssl
Andy Green8c1f6022016-01-26 20:56:56 +08001309#define lws_server_socket_service_ssl(_b, _c) (0)
Andy Greencdb9bf92014-04-12 10:07:02 +08001310#define lws_ssl_close(_a) (0)
1311#define lws_ssl_context_destroy(_a)
Andy Green6b5de702015-12-15 21:15:58 +08001312#define lws_ssl_remove_wsi_from_buffered_list(_a)
Andy Greenb5b23192013-02-11 17:13:32 +08001313#else
Andy Greencdb9bf92014-04-12 10:07:02 +08001314#define LWS_SSL_ENABLED(context) (context->use_ssl)
Joakim Soderbergf272cb02013-02-13 09:29:26 +08001315LWS_EXTERN int openssl_websocket_private_data_index;
Andy Greene99a83c2016-01-20 16:56:06 +08001316LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001317lws_ssl_capable_read(struct lws *wsi, unsigned char *buf, int len);
Andy Greene99a83c2016-01-20 16:56:06 +08001318LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001319lws_ssl_capable_write(struct lws *wsi, unsigned char *buf, int len);
Andy Greene99a83c2016-01-20 16:56:06 +08001320LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001321lws_ssl_pending(struct lws *wsi);
Andy Greene99a83c2016-01-20 16:56:06 +08001322LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green8c1f6022016-01-26 20:56:56 +08001323lws_server_socket_service_ssl(struct lws *new_wsi, lws_sockfd_type accept_fd);
Andy Greencdb9bf92014-04-12 10:07:02 +08001324LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001325lws_ssl_close(struct lws *wsi);
Andy Greencdb9bf92014-04-12 10:07:02 +08001326LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001327lws_ssl_context_destroy(struct lws_context *context);
Andy Green52815602015-01-29 08:36:18 +08001328LWS_VISIBLE void
Andy Green6b5de702015-12-15 21:15:58 +08001329lws_ssl_remove_wsi_from_buffered_list(struct lws *wsi);
Andy Greenc57037a2014-04-03 10:17:00 +08001330#ifndef LWS_NO_SERVER
1331LWS_EXTERN int
1332lws_context_init_server_ssl(struct lws_context_creation_info *info,
Andy Greendc8a3a82015-12-06 09:15:27 +08001333 struct lws_context *context);
Andy Greenc57037a2014-04-03 10:17:00 +08001334#else
1335#define lws_context_init_server_ssl(_a, _b) (0)
1336#endif
1337LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001338lws_ssl_destroy(struct lws_context *context);
Andy Green2eedea92014-04-03 14:33:48 +08001339
1340/* HTTP2-related */
1341
1342#ifdef LWS_USE_HTTP2
1343LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001344lws_context_init_http2_ssl(struct lws_context *context);
Andy Green2eedea92014-04-03 14:33:48 +08001345#else
1346#define lws_context_init_http2_ssl(_a)
1347#endif
Darin Willitsc19456f2011-02-14 17:52:39 +00001348#endif
Andy Greena654fc02014-04-03 07:16:40 +08001349
Andy Green8c1f6022016-01-26 20:56:56 +08001350#if LWS_MAX_SMP > 1
1351static LWS_INLINE void
1352lws_pt_mutex_init(struct lws_context_per_thread *pt)
1353{
1354 pthread_mutex_init(&pt->lock, NULL);
1355}
1356static LWS_INLINE void
1357lws_pt_lock(struct lws_context_per_thread *pt)
1358{
1359 pthread_mutex_lock(&pt->lock);
1360}
1361
1362static LWS_INLINE void
1363lws_pt_unlock(struct lws_context_per_thread *pt)
1364{
1365 pthread_mutex_unlock(&pt->lock);
1366}
1367#else
1368#define lws_pt_mutex_init(_a) (void)(_a)
1369#define lws_pt_lock(_a) (void)(_a)
1370#define lws_pt_unlock(_a) (void)(_a)
1371#endif
1372
Andy Greene99a83c2016-01-20 16:56:06 +08001373LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green6b5de702015-12-15 21:15:58 +08001374lws_ssl_capable_read_no_ssl(struct lws *wsi, unsigned char *buf, int len);
Patrick Gansterera6b019a2014-04-15 18:40:31 +02001375
Andy Greene99a83c2016-01-20 16:56:06 +08001376LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001377lws_ssl_capable_write_no_ssl(struct lws *wsi, unsigned char *buf, int len);
Patrick Gansterera6b019a2014-04-15 18:40:31 +02001378
Andy Greene99a83c2016-01-20 16:56:06 +08001379LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green4b85c1d2015-12-04 11:08:32 +08001380lws_ssl_pending_no_ssl(struct lws *wsi);
=?UTF-8?q?Jos=C3=A9=20Luis=20Mill=C3=A1n?=4c0ba022015-08-19 16:23:33 +02001381
Andy Greena654fc02014-04-03 07:16:40 +08001382#ifndef LWS_NO_CLIENT
Andy Greendc8a3a82015-12-06 09:15:27 +08001383LWS_EXTERN int lws_client_socket_service(struct lws_context *context,
1384 struct lws *wsi,
1385 struct lws_pollfd *pollfd);
Andy Greenc57037a2014-04-03 10:17:00 +08001386#ifdef LWS_OPENSSL_SUPPORT
Andy Greendc8a3a82015-12-06 09:15:27 +08001387LWS_EXTERN int
1388lws_context_init_client_ssl(struct lws_context_creation_info *info,
Andy Green4b85c1d2015-12-04 11:08:32 +08001389 struct lws_context *context);
Andy Greene38031a2014-04-03 08:24:29 +08001390#else
Andy Greenc57037a2014-04-03 10:17:00 +08001391 #define lws_context_init_client_ssl(_a, _b) (0)
1392#endif
Andy Greene99a83c2016-01-20 16:56:06 +08001393LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Greendc8a3a82015-12-06 09:15:27 +08001394lws_handshake_client(struct lws *wsi, unsigned char **buf, size_t len);
1395LWS_EXTERN void
1396lws_decode_ssl_error(void);
Andy Greenc57037a2014-04-03 10:17:00 +08001397#else
1398#define lws_context_init_client_ssl(_a, _b) (0)
Andy Greenaad2eac2014-04-03 09:03:37 +08001399#define lws_handshake_client(_a, _b, _c) (0)
Andy Greena654fc02014-04-03 07:16:40 +08001400#endif
Andy Green44e0b082015-12-28 14:24:49 +08001401
1402LWS_EXTERN int
1403_lws_rx_flow_control(struct lws *wsi);
1404
Andy Green8c1f6022016-01-26 20:56:56 +08001405LWS_EXTERN int
1406_lws_change_pollfd(struct lws *wsi, int _and, int _or, struct lws_pollargs *pa);
1407
Andy Greena654fc02014-04-03 07:16:40 +08001408#ifndef LWS_NO_SERVER
Andy Greendc8a3a82015-12-06 09:15:27 +08001409LWS_EXTERN int
1410lws_server_socket_service(struct lws_context *context, struct lws *wsi,
1411 struct lws_pollfd *pollfd);
1412LWS_EXTERN int
Andy Green11c05bf2015-12-16 18:19:08 +08001413lws_handshake_server(struct lws *wsi, unsigned char **buf, size_t len);
Andy Greenb3d21f12015-12-25 09:12:08 +08001414LWS_EXTERN int
Andy Green8c1f6022016-01-26 20:56:56 +08001415_lws_server_listen_accept_flow_control(struct lws *twsi, int on);
Andy Greend99476b2014-04-03 08:40:05 +08001416#else
Andy Greenb49a9952014-04-03 10:11:04 +08001417#define lws_server_socket_service(_a, _b, _c) (0)
Andy Green11c05bf2015-12-16 18:19:08 +08001418#define lws_handshake_server(_a, _b, _c) (0)
Andy Greenb3d21f12015-12-25 09:12:08 +08001419#define _lws_server_listen_accept_flow_control(a, b) (0)
Andy Greena654fc02014-04-03 07:16:40 +08001420#endif
Andy Green40110e82015-12-14 08:52:03 +08001421
Andy Greendc8a3a82015-12-06 09:15:27 +08001422LWS_EXTERN int
1423lws_get_addresses(struct lws_context *context, void *ads, char *name,
1424 int name_len, char *rip, int rip_len);
Andy Greena654fc02014-04-03 07:16:40 +08001425
1426/*
Alejandro Merycdc97172014-12-04 23:15:27 +01001427 * custom allocator
1428 */
Andy Greene99a83c2016-01-20 16:56:06 +08001429LWS_EXTERN void *
Alejandro Merycdc97172014-12-04 23:15:27 +01001430lws_realloc(void *ptr, size_t size);
1431
Andy Greene99a83c2016-01-20 16:56:06 +08001432LWS_EXTERN void * LWS_WARN_UNUSED_RESULT
Alejandro Merycdc97172014-12-04 23:15:27 +01001433lws_zalloc(size_t size);
1434
1435#define lws_malloc(S) lws_realloc(NULL, S)
1436#define lws_free(P) lws_realloc(P, 0)
Andy Green54806b12015-12-17 17:03:59 +08001437#define lws_free_set_NULL(P) do { lws_realloc(P, 0); (P) = NULL; } while(0)
Alejandro Merycdc97172014-12-04 23:15:27 +01001438
Andy Greendc8a3a82015-12-06 09:15:27 +08001439/* lws_plat_ */
Andy Greena654fc02014-04-03 07:16:40 +08001440LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001441lws_plat_delete_socket_from_fds(struct lws_context *context,
Andy Greendc8a3a82015-12-06 09:15:27 +08001442 struct lws *wsi, int m);
Andy Greena654fc02014-04-03 07:16:40 +08001443LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001444lws_plat_insert_socket_into_fds(struct lws_context *context,
Andy Greendc8a3a82015-12-06 09:15:27 +08001445 struct lws *wsi);
Andy Greena654fc02014-04-03 07:16:40 +08001446LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001447lws_plat_service_periodic(struct lws_context *context);
Andy Greena654fc02014-04-03 07:16:40 +08001448
1449LWS_EXTERN int
Andy Greendc8a3a82015-12-06 09:15:27 +08001450lws_plat_change_pollfd(struct lws_context *context, struct lws *wsi,
1451 struct lws_pollfd *pfd);
Andy Greena654fc02014-04-03 07:16:40 +08001452LWS_EXTERN int
1453lws_plat_context_early_init(void);
1454LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001455lws_plat_context_early_destroy(struct lws_context *context);
Andy Greena654fc02014-04-03 07:16:40 +08001456LWS_EXTERN void
Andy Green4b85c1d2015-12-04 11:08:32 +08001457lws_plat_context_late_destroy(struct lws_context *context);
Andy Greena654fc02014-04-03 07:16:40 +08001458LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001459lws_poll_listen_fd(struct lws_pollfd *fd);
Andy Greena654fc02014-04-03 07:16:40 +08001460LWS_EXTERN int
Andy Green4b85c1d2015-12-04 11:08:32 +08001461lws_plat_service(struct lws_context *context, int timeout_ms);
Andy Greend3a55052016-01-19 03:34:24 +08001462LWS_EXTERN LWS_VISIBLE int
1463lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi);
Andy Greena654fc02014-04-03 07:16:40 +08001464LWS_EXTERN int
Andy Green4386e362015-12-10 07:14:16 +08001465lws_plat_init(struct lws_context *context,
1466 struct lws_context_creation_info *info);
Andy Greena654fc02014-04-03 07:16:40 +08001467LWS_EXTERN void
1468lws_plat_drop_app_privileges(struct lws_context_creation_info *info);
1469LWS_EXTERN unsigned long long
1470time_in_microseconds(void);
Andy Greene99a83c2016-01-20 16:56:06 +08001471LWS_EXTERN const char * LWS_WARN_UNUSED_RESULT
Andy Greena654fc02014-04-03 07:16:40 +08001472lws_plat_inet_ntop(int af, const void *src, char *dst, int cnt);
Andy Green8c0d3c02015-11-02 20:34:12 +08001473
Andy Greene99a83c2016-01-20 16:56:06 +08001474LWS_EXTERN int LWS_WARN_UNUSED_RESULT
Andy Green86c1ef12015-12-30 11:43:36 +08001475lws_check_utf8(unsigned char *state, unsigned char *buf, size_t len);
1476
Andy Green8c0d3c02015-11-02 20:34:12 +08001477#ifdef __cplusplus
1478};
1479#endif