Update libwebsocket to 3.0.1

This commit is contained in:
Fabio Alessandrelli 2018-11-03 15:00:17 +01:00
parent 121cead38e
commit 3703655ce2
26 changed files with 790 additions and 632 deletions

View file

@ -264,9 +264,7 @@ File extracted from upstream source:
- From `roles/ws` exclude `ext` folder. - From `roles/ws` exclude `ext` folder.
- From `tls` exclude `openssl` folder. - From `tls` exclude `openssl` folder.
- Also copy `win32helpers/` from `win32port/` inside `thirdparty/libwebsockets` - Also copy `win32helpers/` from `win32port/` inside `thirdparty/libwebsockets`
- A small fix has been added in `libwebsockets/libwebsockets.h` to `#include <sys/socket.h>` for the BSD family. - A fix has been added to allow building for 32-bits UWP, replacing `GetFileSize[Ex]` and `CreateFileW` with supported functions.
This change has been PRed upstream, and should be merged before the next update. Remember to check and remove this line.
- Another fix has been added to allow building for 32-bits UWP, replacing `GetFileSize[Ex]` and `CreateFileW` with supported functions.
There is a diff for this change in `thirdparty/libwebsockets/uwp_fixes.diff` There is a diff for this change in `thirdparty/libwebsockets/uwp_fixes.diff`
Important: `lws_config.h` and `lws_config_private.h` contains custom Important: `lws_config.h` and `lws_config_private.h` contains custom

View file

@ -134,7 +134,7 @@ lws_protocol_vh_priv_get(struct lws_vhost *vhost,
{ {
int n = 0; int n = 0;
if (!vhost || !vhost->protocol_vh_privs) if (!vhost || !vhost->protocol_vh_privs || !prot)
return NULL; return NULL;
while (n < vhost->count_protocols && &vhost->protocols[n] != prot) while (n < vhost->count_protocols && &vhost->protocols[n] != prot)
@ -808,7 +808,7 @@ lws_create_vhost(struct lws_context *context,
#ifdef LWS_WITH_ACCESS_LOG #ifdef LWS_WITH_ACCESS_LOG
if (info->log_filepath) { if (info->log_filepath) {
vh->log_fd = open(info->log_filepath, vh->log_fd = lws_open(info->log_filepath,
O_CREAT | O_APPEND | O_RDWR, 0600); O_CREAT | O_APPEND | O_RDWR, 0600);
if (vh->log_fd == (int)LWS_INVALID_FILE) { if (vh->log_fd == (int)LWS_INVALID_FILE) {
lwsl_err("unable to open log filepath %s\n", lwsl_err("unable to open log filepath %s\n",
@ -936,24 +936,29 @@ lws_create_event_pipes(struct lws_context *context)
wsi->tsi = n; wsi->tsi = n;
wsi->vhost = NULL; wsi->vhost = NULL;
wsi->event_pipe = 1; wsi->event_pipe = 1;
wsi->desc.sockfd = LWS_SOCK_INVALID;
context->pt[n].pipe_wsi = wsi;
context->count_wsi_allocated++;
if (lws_plat_pipe_create(wsi)) { if (lws_plat_pipe_create(wsi))
lws_free(wsi); /*
* platform code returns 0 if it actually created pipes
* and initialized pt->dummy_pipe_fds[]. If it used
* some other mechanism outside of signaling in the
* normal event loop, we skip treating the pipe as
* related to dummy_pipe_fds[], adding it to the fds,
* etc.
*/
continue; continue;
}
wsi->desc.sockfd = context->pt[n].dummy_pipe_fds[0]; wsi->desc.sockfd = context->pt[n].dummy_pipe_fds[0];
lwsl_debug("event pipe fd %d\n", wsi->desc.sockfd); lwsl_debug("event pipe fd %d\n", wsi->desc.sockfd);
context->pt[n].pipe_wsi = wsi;
if (context->event_loop_ops->accept) if (context->event_loop_ops->accept)
context->event_loop_ops->accept(wsi); context->event_loop_ops->accept(wsi);
if (__insert_wsi_socket_into_fds(context, wsi)) if (__insert_wsi_socket_into_fds(context, wsi))
return 1; return 1;
//lws_change_pollfd(context->pt[n].pipe_wsi, 0, LWS_POLLIN);
context->count_wsi_allocated++;
} }
return 0; return 0;

View file

@ -28,7 +28,7 @@
#ifdef LWS_WITH_IPV6 #ifdef LWS_WITH_IPV6
#if defined(WIN32) || defined(_WIN32) #if defined(WIN32) || defined(_WIN32)
#include <wincrypt.h> #include <wincrypt.h>
#include <Iphlpapi.h> #include <iphlpapi.h>
#else #else
#include <net/if.h> #include <net/if.h>
#endif #endif
@ -58,6 +58,28 @@ static const char * const log_level_names[] = {
}; };
#endif #endif
int lws_open(const char *__file, int __oflag, ...)
{
va_list ap;
int n;
va_start(ap, __oflag);
if (((__oflag & O_CREAT) == O_CREAT)
#if defined(O_TMPFILE)
|| ((__oflag & O_TMPFILE) == O_TMPFILE)
#endif
)
/* last arg is really a mode_t. But windows... */
n = open(__file, __oflag, va_arg(ap, uint32_t));
else
n = open(__file, __oflag);
va_end(ap);
lws_plat_apply_FD_CLOEXEC(n);
return n;
}
#if defined (_DEBUG) #if defined (_DEBUG)
void lwsi_set_role(struct lws *wsi, lws_wsi_state_t role) void lwsi_set_role(struct lws *wsi, lws_wsi_state_t role)
{ {
@ -826,7 +848,15 @@ just_kill_connection:
if (!wsi->protocol) if (!wsi->protocol)
pro = &wsi->vhost->protocols[0]; pro = &wsi->vhost->protocols[0];
pro->callback(wsi, if (!wsi->upgraded_to_http2 || !lwsi_role_client(wsi))
/*
* The network wsi for a client h2 connection shouldn't
* call back for its role: the child stream connections
* own the role. Otherwise h2 will call back closed
* one too many times as the children do it and then
* the closing network stream.
*/
pro->callback(wsi,
wsi->role_ops->close_cb[lwsi_role_server(wsi)], wsi->role_ops->close_cb[lwsi_role_server(wsi)],
wsi->user_space, NULL, 0); wsi->user_space, NULL, 0);
wsi->told_user_closed = 1; wsi->told_user_closed = 1;
@ -1453,7 +1483,7 @@ lws_vfs_select_fops(const struct lws_plat_file_ops *fops, const char *vfs_path,
pf = fops->next; pf = fops->next;
while (pf) { while (pf) {
n = 0; n = 0;
while (n < (int)ARRAY_SIZE(pf->fi) && pf->fi[n].sig) { while (n < (int)LWS_ARRAY_SIZE(pf->fi) && pf->fi[n].sig) {
if (p >= vfs_path + pf->fi[n].len) if (p >= vfs_path + pf->fi[n].len)
if (!strncmp(p - (pf->fi[n].len - 1), if (!strncmp(p - (pf->fi[n].len - 1),
pf->fi[n].sig, pf->fi[n].sig,
@ -1935,9 +1965,9 @@ static const char * const colours[] = {
"[32;1m", /* LLL_INFO */ "[32;1m", /* LLL_INFO */
"[34;1m", /* LLL_DEBUG */ "[34;1m", /* LLL_DEBUG */
"[33;1m", /* LLL_PARSER */ "[33;1m", /* LLL_PARSER */
"[33;1m", /* LLL_HEADER */ "[33m", /* LLL_HEADER */
"[33;1m", /* LLL_EXT */ "[33m", /* LLL_EXT */
"[33;1m", /* LLL_CLIENT */ "[33m", /* LLL_CLIENT */
"[33;1m", /* LLL_LATENCY */ "[33;1m", /* LLL_LATENCY */
"[30;1m", /* LLL_USER */ "[30;1m", /* LLL_USER */
}; };
@ -1946,14 +1976,14 @@ LWS_VISIBLE void lwsl_emit_stderr(int level, const char *line)
{ {
char buf[50]; char buf[50];
static char tty = 3; static char tty = 3;
int n, m = ARRAY_SIZE(colours) - 1; int n, m = LWS_ARRAY_SIZE(colours) - 1;
if (!tty) if (!tty)
tty = isatty(2) | 2; tty = isatty(2) | 2;
lwsl_timestamp(level, buf, sizeof(buf)); lwsl_timestamp(level, buf, sizeof(buf));
if (tty == 3) { if (tty == 3) {
n = 1 << (ARRAY_SIZE(colours) - 1); n = 1 << (LWS_ARRAY_SIZE(colours) - 1);
while (n) { while (n) {
if (level & n) if (level & n)
break; break;
@ -2060,7 +2090,9 @@ lwsl_hexdump_level(int hexdump_level, const void *vbuf, size_t len)
LWS_VISIBLE void LWS_VISIBLE void
lwsl_hexdump(const void *vbuf, size_t len) lwsl_hexdump(const void *vbuf, size_t len)
{ {
#if defined(_DEBUG)
lwsl_hexdump_level(LLL_DEBUG, vbuf, len); lwsl_hexdump_level(LLL_DEBUG, vbuf, len);
#endif
} }
LWS_VISIBLE int LWS_VISIBLE int
@ -2091,6 +2123,8 @@ lws_partial_buffered(struct lws *wsi)
LWS_VISIBLE lws_fileofs_t LWS_VISIBLE lws_fileofs_t
lws_get_peer_write_allowance(struct lws *wsi) lws_get_peer_write_allowance(struct lws *wsi)
{ {
if (!wsi->role_ops->tx_credit)
return -1;
return wsi->role_ops->tx_credit(wsi); return wsi->role_ops->tx_credit(wsi);
} }
@ -3364,7 +3398,7 @@ lws_stats_log_dump(struct lws_context *context)
wl = pt->http.ah_wait_list; wl = pt->http.ah_wait_list;
while (wl) { while (wl) {
m++; m++;
wl = wl->ah_wait_list; wl = wl->http.ah_wait_list;
} }
lwsl_notice(" AH wait list count / actual: %d / %d\n", lwsl_notice(" AH wait list count / actual: %d / %d\n",
@ -3401,7 +3435,8 @@ lws_stats_log_dump(struct lws_context *context)
strcpy(buf, "unknown"); strcpy(buf, "unknown");
#if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2) #if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)
lwsl_notice(" peer %s: count wsi: %d, count ah: %d\n", lwsl_notice(" peer %s: count wsi: %d, count ah: %d\n",
buf, df->count_wsi, df->count_ah); buf, df->count_wsi,
df->http.count_ah);
#else #else
lwsl_notice(" peer %s: count wsi: %d\n", lwsl_notice(" peer %s: count wsi: %d\n",
buf, df->count_wsi); buf, df->count_wsi);

View file

@ -126,6 +126,18 @@ int lws_issue_raw(struct lws *wsi, unsigned char *buf, size_t len)
lwsl_info("** %p signalling to close now\n", wsi); lwsl_info("** %p signalling to close now\n", wsi);
return -1; /* retry closing now */ return -1; /* retry closing now */
} }
#if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)
#if !defined(LWS_WITHOUT_SERVER)
if (wsi->http.deferred_transaction_completed) {
lwsl_notice("%s: partial completed, doing "
"deferred transaction completed\n",
__func__);
wsi->http.deferred_transaction_completed = 0;
return lws_http_transaction_completed(wsi);
}
#endif
#endif
} }
/* always callback on writeable */ /* always callback on writeable */
lws_callback_on_writable(wsi); lws_callback_on_writable(wsi);

View file

@ -232,7 +232,7 @@
#endif #endif
#else /* not windows */ #else /* not windows */
static inline int compatible_close(int fd) { return close(fd); } static LWS_INLINE int compatible_close(int fd) { return close(fd); }
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/time.h> #include <sys/time.h>
@ -351,7 +351,15 @@ extern "C" {
#define LWS_H2_RX_SCRATCH_SIZE 512 #define LWS_H2_RX_SCRATCH_SIZE 512
#if defined(WIN32) || defined(_WIN32)
// Visual studio older than 2015 and WIN_CE has only _stricmp
#if (defined(_MSC_VER) && _MSC_VER < 1900) || defined(_WIN32_WCE)
#define strcasecmp _stricmp
#elif !defined(__MINGW32__)
#define strcasecmp stricmp
#endif
#define getdtablesize() 30000
#endif
/* /*
* All lws_tls...() functions must return this type, converting the * All lws_tls...() functions must return this type, converting the
@ -863,6 +871,7 @@ struct lws_context {
unsigned int timeout_secs; unsigned int timeout_secs;
unsigned int pt_serv_buf_size; unsigned int pt_serv_buf_size;
int max_http_header_data; int max_http_header_data;
int max_http_header_pool;
int simultaneous_ssl_restriction; int simultaneous_ssl_restriction;
int simultaneous_ssl; int simultaneous_ssl;
#if defined(LWS_WITH_PEER_LIMITS) #if defined(LWS_WITH_PEER_LIMITS)
@ -889,7 +898,6 @@ struct lws_context {
volatile int service_tid; volatile int service_tid;
int service_tid_detected; int service_tid_detected;
short max_http_header_pool;
short count_threads; short count_threads;
short plugin_protocol_count; short plugin_protocol_count;
short plugin_extension_count; short plugin_extension_count;
@ -1216,7 +1224,7 @@ LWS_EXTERN int
lws_rxflow_cache(struct lws *wsi, unsigned char *buf, int n, int len); lws_rxflow_cache(struct lws *wsi, unsigned char *buf, int n, int len);
#ifndef LWS_LATENCY #ifndef LWS_LATENCY
static inline void static LWS_INLINE void
lws_latency(struct lws_context *context, struct lws *wsi, const char *action, lws_latency(struct lws_context *context, struct lws *wsi, const char *action,
int ret, int completion) { int ret, int completion) {
do { do {
@ -1224,7 +1232,7 @@ lws_latency(struct lws_context *context, struct lws *wsi, const char *action,
(void)completion; (void)completion;
} while (0); } while (0);
} }
static inline void static LWS_INLINE void
lws_latency_pre(struct lws_context *context, struct lws *wsi) { lws_latency_pre(struct lws_context *context, struct lws *wsi) {
do { (void)context; (void)wsi; } while (0); do { (void)context; (void)wsi; } while (0);
} }
@ -1597,6 +1605,9 @@ void lws_free(void *p);
#define lws_free_set_NULL(P) do { lws_realloc(P, 0, "free"); (P) = NULL; } while(0) #define lws_free_set_NULL(P) do { lws_realloc(P, 0, "free"); (P) = NULL; } while(0)
#endif #endif
char *
lws_strdup(const char *s);
int int
lws_plat_pipe_create(struct lws *wsi); lws_plat_pipe_create(struct lws *wsi);
int int
@ -1606,6 +1617,9 @@ lws_plat_pipe_close(struct lws *wsi);
int int
lws_create_event_pipes(struct lws_context *context); lws_create_event_pipes(struct lws_context *context);
int lws_open(const char *__file, int __oflag, ...);
void lws_plat_apply_FD_CLOEXEC(int n);
const struct lws_plat_file_ops * const struct lws_plat_file_ops *
lws_vfs_select_fops(const struct lws_plat_file_ops *fops, const char *vfs_path, lws_vfs_select_fops(const struct lws_plat_file_ops *fops, const char *vfs_path,
const char **vpath); const char **vpath);
@ -1673,10 +1687,10 @@ lws_broadcast(struct lws_context *context, int reason, void *in, size_t len);
lws_stats_atomic_max(struct lws_context * context, lws_stats_atomic_max(struct lws_context * context,
struct lws_context_per_thread *pt, int index, uint64_t val); struct lws_context_per_thread *pt, int index, uint64_t val);
#else #else
static inline uint64_t lws_stats_atomic_bump(struct lws_context * context, static LWS_INLINE uint64_t lws_stats_atomic_bump(struct lws_context * context,
struct lws_context_per_thread *pt, int index, uint64_t bump) { struct lws_context_per_thread *pt, int index, uint64_t bump) {
(void)context; (void)pt; (void)index; (void)bump; return 0; } (void)context; (void)pt; (void)index; (void)bump; return 0; }
static inline uint64_t lws_stats_atomic_max(struct lws_context * context, static LWS_INLINE uint64_t lws_stats_atomic_max(struct lws_context * context,
struct lws_context_per_thread *pt, int index, uint64_t val) { struct lws_context_per_thread *pt, int index, uint64_t val) {
(void)context; (void)pt; (void)index; (void)val; return 0; } (void)context; (void)pt; (void)index; (void)val; return 0; }
#endif #endif
@ -1703,6 +1717,11 @@ void
lws_peer_dump_from_wsi(struct lws *wsi); lws_peer_dump_from_wsi(struct lws *wsi);
#endif #endif
#ifdef LWS_WITH_HTTP_PROXY
hubbub_error
html_parser_cb(const hubbub_token *token, void *pw);
#endif
void void
__lws_remove_from_timeout_list(struct lws *wsi); __lws_remove_from_timeout_list(struct lws *wsi);

View file

@ -66,14 +66,6 @@ typedef unsigned long long lws_intptr_t;
#define O_RDONLY _O_RDONLY #define O_RDONLY _O_RDONLY
#endif #endif
// Visual studio older than 2015 and WIN_CE has only _stricmp
#if (defined(_MSC_VER) && _MSC_VER < 1900) || defined(_WIN32_WCE)
#define strcasecmp _stricmp
#elif !defined(__MINGW32__)
#define strcasecmp stricmp
#endif
#define getdtablesize() 30000
#define LWS_INLINE __inline #define LWS_INLINE __inline
#define LWS_VISIBLE #define LWS_VISIBLE
#define LWS_WARN_UNUSED_RESULT #define LWS_WARN_UNUSED_RESULT
@ -150,6 +142,7 @@ typedef unsigned long long lws_intptr_t;
#endif #endif
#if defined(__ANDROID__) #if defined(__ANDROID__)
#include <netinet/in.h>
#include <unistd.h> #include <unistd.h>
#define getdtablesize() sysconf(_SC_OPEN_MAX) #define getdtablesize() sysconf(_SC_OPEN_MAX)
#endif #endif
@ -164,6 +157,9 @@ typedef unsigned long long lws_intptr_t;
#ifdef LWS_HAVE_UV_VERSION_H #ifdef LWS_HAVE_UV_VERSION_H
#include <uv-version.h> #include <uv-version.h>
#endif #endif
#ifdef LWS_HAVE_NEW_UV_VERSION_H
#include <uv/version.h>
#endif
#endif /* LWS_WITH_LIBUV */ #endif /* LWS_WITH_LIBUV */
#if defined(LWS_WITH_LIBEVENT) #if defined(LWS_WITH_LIBEVENT)
#include <event2/event.h> #include <event2/event.h>
@ -456,9 +452,6 @@ lwsl_visible(int level);
#endif #endif
struct lws; struct lws;
#ifndef ARRAY_SIZE
#define ARRAY_SIZE(x) (sizeof(x) / sizeof(x[0]))
#endif
typedef int64_t lws_usec_t; typedef int64_t lws_usec_t;
@ -540,7 +533,7 @@ struct timer_mapping {
#define lws_uv_getloop(a, b) (NULL) #define lws_uv_getloop(a, b) (NULL)
static inline void uv_timer_init(void *l, uv_timer_t *t) static LWS_INLINE void uv_timer_init(void *l, uv_timer_t *t)
{ {
(void)l; (void)l;
*t = NULL; *t = NULL;
@ -548,7 +541,7 @@ static inline void uv_timer_init(void *l, uv_timer_t *t)
extern void esp32_uvtimer_cb(TimerHandle_t t); extern void esp32_uvtimer_cb(TimerHandle_t t);
static inline void uv_timer_start(uv_timer_t *t, uv_cb_t *cb, int first, int rep) static LWS_INLINE void uv_timer_start(uv_timer_t *t, uv_cb_t *cb, int first, int rep)
{ {
struct timer_mapping *tm = (struct timer_mapping *)malloc(sizeof(*tm)); struct timer_mapping *tm = (struct timer_mapping *)malloc(sizeof(*tm));
@ -563,12 +556,12 @@ static inline void uv_timer_start(uv_timer_t *t, uv_cb_t *cb, int first, int rep
xTimerStart(*t, 0); xTimerStart(*t, 0);
} }
static inline void uv_timer_stop(uv_timer_t *t) static LWS_INLINE void uv_timer_stop(uv_timer_t *t)
{ {
xTimerStop(*t, 0); xTimerStop(*t, 0);
} }
static inline void uv_close(uv_handle_t *h, void *v) static LWS_INLINE void uv_close(uv_handle_t *h, void *v)
{ {
free(pvTimerGetTimerID((uv_timer_t)h)); free(pvTimerGetTimerID((uv_timer_t)h));
xTimerDelete(*(uv_timer_t *)h, 0); xTimerDelete(*(uv_timer_t *)h, 0);
@ -1757,7 +1750,7 @@ lws_genhash_destroy(struct lws_genhash_ctx *ctx, void *result);
* If the return is nonzero, it failed and there is nothing needing to be * If the return is nonzero, it failed and there is nothing needing to be
* destroyed. * destroyed.
*/ */
int LWS_VISIBLE LWS_EXTERN int LWS_WARN_UNUSED_RESULT
lws_genhmac_init(struct lws_genhmac_ctx *ctx, enum lws_genhmac_types type, lws_genhmac_init(struct lws_genhmac_ctx *ctx, enum lws_genhmac_types type,
const uint8_t *key, size_t key_len); const uint8_t *key, size_t key_len);
@ -1771,7 +1764,7 @@ lws_genhmac_init(struct lws_genhmac_ctx *ctx, enum lws_genhmac_types type,
* *
* If the return is nonzero, it failed and needs destroying. * If the return is nonzero, it failed and needs destroying.
*/ */
int LWS_VISIBLE LWS_EXTERN int LWS_WARN_UNUSED_RESULT
lws_genhmac_update(struct lws_genhmac_ctx *ctx, const void *in, size_t len); lws_genhmac_update(struct lws_genhmac_ctx *ctx, const void *in, size_t len);
/** lws_genhmac_destroy() - copy out the result digest and destroy the ctx /** lws_genhmac_destroy() - copy out the result digest and destroy the ctx
@ -1785,7 +1778,7 @@ lws_genhmac_update(struct lws_genhmac_ctx *ctx, const void *in, size_t len);
* NULL result is supported so that you can destroy the ctx cleanly on error * NULL result is supported so that you can destroy the ctx cleanly on error
* conditions, where there is no valid result. * conditions, where there is no valid result.
*/ */
int LWS_VISIBLE LWS_EXTERN int
lws_genhmac_destroy(struct lws_genhmac_ctx *ctx, void *result); lws_genhmac_destroy(struct lws_genhmac_ctx *ctx, void *result);
///@} ///@}
@ -2812,7 +2805,7 @@ struct lws_context_creation_info {
/**< VHOST: pointer to optional linked list of per-vhost /**< VHOST: pointer to optional linked list of per-vhost
* options made accessible to protocols */ * options made accessible to protocols */
int keepalive_timeout; int keepalive_timeout;
/**< VHOST: (default = 0 = 60s) seconds to allow remote /**< VHOST: (default = 0 = 5s) seconds to allow remote
* client to hold on to an idle HTTP/1.1 connection */ * client to hold on to an idle HTTP/1.1 connection */
const char *log_filepath; const char *log_filepath;
/**< VHOST: filepath to append logs to... this is opened before /**< VHOST: filepath to append logs to... this is opened before
@ -4943,7 +4936,7 @@ lws_write(struct lws *wsi, unsigned char *buf, size_t len,
lws_write(wsi, (unsigned char *)(buf), len, LWS_WRITE_HTTP) lws_write(wsi, (unsigned char *)(buf), len, LWS_WRITE_HTTP)
/* helper for multi-frame ws message flags */ /* helper for multi-frame ws message flags */
static inline int static LWS_INLINE int
lws_write_ws_flags(int initial, int is_start, int is_end) lws_write_ws_flags(int initial, int is_start, int is_end)
{ {
int r; int r;
@ -5621,13 +5614,13 @@ struct lws_dll_lws { /* typed as struct lws * */
#define lws_dll_is_null(___dll) (!(___dll)->prev && !(___dll)->next) #define lws_dll_is_null(___dll) (!(___dll)->prev && !(___dll)->next)
static inline void static LWS_INLINE void
lws_dll_lws_add_front(struct lws_dll_lws *_a, struct lws_dll_lws *_head) lws_dll_lws_add_front(struct lws_dll_lws *_a, struct lws_dll_lws *_head)
{ {
lws_dll_add_front((struct lws_dll *)_a, (struct lws_dll *)_head); lws_dll_add_front((struct lws_dll *)_a, (struct lws_dll *)_head);
} }
static inline void static LWS_INLINE void
lws_dll_lws_remove(struct lws_dll_lws *_a) lws_dll_lws_remove(struct lws_dll_lws *_a)
{ {
lws_dll_remove((struct lws_dll *)_a); lws_dll_remove((struct lws_dll *)_a);
@ -7064,9 +7057,6 @@ lws_email_destroy(struct lws_email *email);
//@{ //@{
struct lejp_ctx; struct lejp_ctx;
#ifndef ARRAY_SIZE
#define ARRAY_SIZE(_x) (sizeof(_x) / sizeof(_x[0]))
#endif
#define LWS_ARRAY_SIZE(_x) (sizeof(_x) / sizeof(_x[0])) #define LWS_ARRAY_SIZE(_x) (sizeof(_x) / sizeof(_x[0]))
#define LEJP_FLAG_WS_KEEP 64 #define LEJP_FLAG_WS_KEEP 64
#define LEJP_FLAG_WS_COMMENTLINE 32 #define LEJP_FLAG_WS_COMMENTLINE 32
@ -7219,7 +7209,7 @@ typedef signed char (*lejp_callback)(struct lejp_ctx *ctx, char reason);
#endif #endif
#ifndef LEJP_STRING_CHUNK #ifndef LEJP_STRING_CHUNK
/* must be >= 30 to assemble floats */ /* must be >= 30 to assemble floats */
#define LEJP_STRING_CHUNK 255 #define LEJP_STRING_CHUNK 254
#endif #endif
enum num_flags { enum num_flags {
@ -7253,7 +7243,7 @@ struct lejp_ctx {
uint16_t i[LEJP_MAX_INDEX_DEPTH]; /* index array */ uint16_t i[LEJP_MAX_INDEX_DEPTH]; /* index array */
uint16_t wild[LEJP_MAX_INDEX_DEPTH]; /* index array */ uint16_t wild[LEJP_MAX_INDEX_DEPTH]; /* index array */
char path[LEJP_MAX_PATH]; char path[LEJP_MAX_PATH];
char buf[LEJP_STRING_CHUNK]; char buf[LEJP_STRING_CHUNK + 1];
/* int */ /* int */

View file

@ -20,6 +20,7 @@
*/ */
#include <libwebsockets.h> #include <libwebsockets.h>
#include "core/private.h"
#include <string.h> #include <string.h>
#include <stdio.h> #include <stdio.h>
@ -30,7 +31,7 @@
* \param callback: your user callback which will received parsed tokens * \param callback: your user callback which will received parsed tokens
* \param user: optional user data pointer untouched by lejp * \param user: optional user data pointer untouched by lejp
* \param paths: your array of name elements you are interested in * \param paths: your array of name elements you are interested in
* \param count_paths: ARRAY_SIZE() of @paths * \param count_paths: LWS_ARRAY_SIZE() of @paths
* *
* Prepares your context struct for use with lejp * Prepares your context struct for use with lejp
*/ */
@ -250,7 +251,7 @@ lejp_parse(struct lejp_ctx *ctx, const unsigned char *json, int len)
case LEJP_MP_STRING: case LEJP_MP_STRING:
if (c == '\"') { if (c == '\"') {
if (!ctx->sp) { if (!ctx->sp) { /* JSON can't end on quote */
ret = LEJP_REJECT_MP_STRING_UNDERRUN; ret = LEJP_REJECT_MP_STRING_UNDERRUN;
goto reject; goto reject;
} }
@ -417,7 +418,7 @@ lejp_parse(struct lejp_ctx *ctx, const unsigned char *json, int len)
goto reject; goto reject;
} }
ctx->i[ctx->ipos++] = 0; ctx->i[ctx->ipos++] = 0;
if (ctx->ipos > ARRAY_SIZE(ctx->i)) { if (ctx->ipos > LWS_ARRAY_SIZE(ctx->i)) {
ret = LEJP_REJECT_MP_DELIM_ISTACK; ret = LEJP_REJECT_MP_DELIM_ISTACK;
goto reject; goto reject;
} }
@ -425,17 +426,23 @@ lejp_parse(struct lejp_ctx *ctx, const unsigned char *json, int len)
case ']': case ']':
/* pop */ /* pop */
if (!ctx->sp) { /* JSON can't end on ] */
ret = LEJP_REJECT_MP_C_OR_E_UNDERF;
goto reject;
}
ctx->sp--; ctx->sp--;
if (ctx->st[ctx->sp].s != LEJP_MP_ARRAY_END) { if (ctx->st[ctx->sp].s != LEJP_MP_ARRAY_END) {
ret = LEJP_REJECT_MP_C_OR_E_NOTARRAY; ret = LEJP_REJECT_MP_C_OR_E_NOTARRAY;
goto reject; goto reject;
} }
/* drop the path [n] bit */ /* drop the path [n] bit */
ctx->ppos = ctx->st[ctx->sp - 1].p; if (ctx->sp) {
ctx->ipos = ctx->st[ctx->sp - 1].i; ctx->ppos = ctx->st[ctx->sp - 1].p;
ctx->ipos = ctx->st[ctx->sp - 1].i;
}
ctx->path[ctx->ppos] = '\0'; ctx->path[ctx->ppos] = '\0';
if (ctx->path_match && if (ctx->path_match &&
ctx->ppos <= ctx->path_match_len) ctx->ppos <= ctx->path_match_len)
/* /*
* we shrank the path to be * we shrank the path to be
* smaller than the matching point * smaller than the matching point
@ -603,7 +610,7 @@ lejp_parse(struct lejp_ctx *ctx, const unsigned char *json, int len)
break; break;
} }
if (c == ']') { if (c == ']') {
if (!ctx->sp) { if (!ctx->sp) { /* JSON can't end on ] */
ret = LEJP_REJECT_MP_C_OR_E_UNDERF; ret = LEJP_REJECT_MP_C_OR_E_UNDERF;
goto reject; goto reject;
} }
@ -631,7 +638,7 @@ lejp_parse(struct lejp_ctx *ctx, const unsigned char *json, int len)
goto redo_character; goto redo_character;
} }
if (c == '}') { if (c == '}') {
if (ctx->sp == 0) { if (!ctx->sp) {
lejp_check_path_match(ctx); lejp_check_path_match(ctx);
if (ctx->callback(ctx, LEJPCB_OBJECT_END)) { if (ctx->callback(ctx, LEJPCB_OBJECT_END)) {
ret = LEJP_REJECT_CALLBACK; ret = LEJP_REJECT_CALLBACK;
@ -716,7 +723,7 @@ add_stack_level:
ctx->st[ctx->sp].p = ctx->ppos; ctx->st[ctx->sp].p = ctx->ppos;
ctx->st[ctx->sp].i = ctx->ipos; ctx->st[ctx->sp].i = ctx->ipos;
if (++ctx->sp == ARRAY_SIZE(ctx->st)) { if (++ctx->sp == LWS_ARRAY_SIZE(ctx->st)) {
ret = LEJP_REJECT_STACK_OVERFLOW; ret = LEJP_REJECT_STACK_OVERFLOW;
goto reject; goto reject;
} }

View file

@ -30,6 +30,12 @@
#endif #endif
#include <dirent.h> #include <dirent.h>
void lws_plat_apply_FD_CLOEXEC(int n)
{
if (n != -1)
fcntl(n, F_SETFD, FD_CLOEXEC );
}
int int
lws_plat_socket_offset(void) lws_plat_socket_offset(void)
{ {
@ -330,6 +336,8 @@ lws_plat_set_socket_options(struct lws_vhost *vhost, int fd)
struct protoent *tcp_proto; struct protoent *tcp_proto;
#endif #endif
fcntl(fd, F_SETFD, FD_CLOEXEC);
if (vhost->ka_time) { if (vhost->ka_time) {
/* enable keepalive on this socket */ /* enable keepalive on this socket */
optval = 1; optval = 1;
@ -952,7 +960,7 @@ lws_plat_write_file(const char *filename, void *buf, int len)
LWS_VISIBLE int LWS_VISIBLE int
lws_plat_read_file(const char *filename, void *buf, int len) lws_plat_read_file(const char *filename, void *buf, int len)
{ {
int n, fd = open(filename, O_RDONLY); int n, fd = lws_open(filename, O_RDONLY);
if (fd == -1) if (fd == -1)
return -1; return -1;

View file

@ -3,6 +3,10 @@
#endif #endif
#include "core/private.h" #include "core/private.h"
void lws_plat_apply_FD_CLOEXEC(int n)
{
}
int int
lws_plat_socket_offset(void) lws_plat_socket_offset(void)
{ {
@ -54,7 +58,7 @@ time_in_microseconds()
memcpy(&datetime, &filetime, sizeof(datetime)); memcpy(&datetime, &filetime, sizeof(datetime));
/* Windows file times are in 100s of nanoseconds. */ /* Windows file times are in 100s of nanoseconds. */
return (datetime.QuadPart - DELTA_EPOCH_IN_MICROSECS) / 10; return (datetime.QuadPart / 10) - DELTA_EPOCH_IN_MICROSECS;
} }
#ifdef _WIN32_WCE #ifdef _WIN32_WCE
@ -229,23 +233,21 @@ _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
continue; continue;
wsi = wsi_from_fd(context, pfd->fd); wsi = wsi_from_fd(context, pfd->fd);
if (wsi->listener) if (!wsi || wsi->listener)
continue; continue;
if (!wsi || wsi->sock_send_blocking) if (wsi->sock_send_blocking)
continue; continue;
pfd->revents = LWS_POLLOUT; pfd->revents = LWS_POLLOUT;
n = lws_service_fd(context, pfd); n = lws_service_fd(context, pfd);
if (n < 0) if (n < 0)
return -1; return -1;
/* Force WSAWaitForMultipleEvents() to check events and then return immediately. */
timeout_ms = 0;
/* if something closed, retry this slot */ /* if something closed, retry this slot */
if (n) if (n)
i--; i--;
/*
* any wsi has truncated, force him signalled
*/
if (wsi->trunc_len)
WSASetEvent(pt->events[0]);
} }
/* /*
@ -261,9 +263,11 @@ _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
} }
if (timeout_ms) { if (timeout_ms) {
lws_usec_t t;
lws_pt_lock(pt, __func__); lws_pt_lock(pt, __func__);
/* don't stay in poll wait longer than next hr timeout */ /* don't stay in poll wait longer than next hr timeout */
lws_usec_t t = __lws_hrtimer_service(pt); t = __lws_hrtimer_service(pt);
if ((lws_usec_t)timeout_ms * 1000 > t) if ((lws_usec_t)timeout_ms * 1000 > t)
timeout_ms = (int)(t / 1000); timeout_ms = (int)(t / 1000);
@ -310,8 +314,10 @@ _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
if (pfd->revents & LWS_POLLHUP) if (pfd->revents & LWS_POLLHUP)
--eIdx; --eIdx;
if (pfd->revents) if (pfd->revents) {
recv(pfd->fd, NULL, 0, 0);
lws_service_fd_tsi(context, pfd, tsi); lws_service_fd_tsi(context, pfd, tsi);
}
} }
} }
@ -637,7 +643,7 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename,
lws_fop_fd_t fop_fd; lws_fop_fd_t fop_fd;
FILE_STANDARD_INFO fInfo = {0}; FILE_STANDARD_INFO fInfo = {0};
MultiByteToWideChar(CP_UTF8, 0, filename, -1, buf, ARRAY_SIZE(buf)); MultiByteToWideChar(CP_UTF8, 0, filename, -1, buf, LWS_ARRAY_SIZE(buf));
#if defined(_WIN32_WINNT) && _WIN32_WINNT >= 0x0602 // Windows 8 (minimum when UWP_ENABLED, but can be used in Windows builds) #if defined(_WIN32_WINNT) && _WIN32_WINNT >= 0x0602 // Windows 8 (minimum when UWP_ENABLED, but can be used in Windows builds)
CREATEFILE2_EXTENDED_PARAMETERS extParams = {0}; CREATEFILE2_EXTENDED_PARAMETERS extParams = {0};
@ -810,7 +816,7 @@ lws_plat_write_file(const char *filename, void *buf, int len)
{ {
int m, fd; int m, fd;
fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0600); fd = lws_open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0600);
if (fd == -1) if (fd == -1)
return -1; return -1;
@ -824,7 +830,7 @@ lws_plat_write_file(const char *filename, void *buf, int len)
LWS_VISIBLE int LWS_VISIBLE int
lws_plat_read_file(const char *filename, void *buf, int len) lws_plat_read_file(const char *filename, void *buf, int len)
{ {
int n, fd = open(filename, O_RDONLY); int n, fd = lws_open(filename, O_RDONLY);
if (fd == -1) if (fd == -1)
return -1; return -1;

View file

@ -195,6 +195,7 @@ postbody_completion:
} }
break; break;
case LRS_RETURNED_CLOSE:
case LRS_AWAITING_CLOSE_ACK: case LRS_AWAITING_CLOSE_ACK:
case LRS_WAITING_TO_SEND_CLOSE: case LRS_WAITING_TO_SEND_CLOSE:
case LRS_SHUTDOWN: case LRS_SHUTDOWN:
@ -444,6 +445,16 @@ try_pollout:
if (lwsi_state(wsi) != LRS_ISSUING_FILE) { if (lwsi_state(wsi) != LRS_ISSUING_FILE) {
if (wsi->trunc_len) {
//lwsl_notice("%s: completing partial\n", __func__);
if (lws_issue_raw(wsi, wsi->trunc_alloc + wsi->trunc_offset,
wsi->trunc_len) < 0) {
lwsl_info("%s signalling to close\n", __func__);
goto fail;
}
return LWS_HPI_RET_HANDLED;
}
lws_stats_atomic_bump(wsi->context, pt, lws_stats_atomic_bump(wsi->context, pt,
LWSSTATS_C_WRITEABLE_CB, 1); LWSSTATS_C_WRITEABLE_CB, 1);
#if defined(LWS_WITH_STATS) #if defined(LWS_WITH_STATS)
@ -655,6 +666,9 @@ rops_destroy_role_h1(struct lws *wsi)
ah = ah->next; ah = ah->next;
} }
#ifdef LWS_ROLE_WS
lws_free_set_NULL(wsi->ws);
#endif
return 0; return 0;
} }

View file

@ -162,7 +162,7 @@ create_new_conn:
if (!wsi->client_hostname_copy) if (!wsi->client_hostname_copy)
wsi->client_hostname_copy = wsi->client_hostname_copy =
strdup(lws_hdr_simple_ptr(wsi, lws_strdup(lws_hdr_simple_ptr(wsi,
_WSI_TOKEN_CLIENT_PEER_ADDRESS)); _WSI_TOKEN_CLIENT_PEER_ADDRESS));
/* /*
@ -654,13 +654,13 @@ lws_client_reset(struct lws **pwsi, int ssl, const char *address, int port,
lws_ssl_close(wsi); lws_ssl_close(wsi);
#endif #endif
__remove_wsi_socket_from_fds(wsi);
if (wsi->context->event_loop_ops->close_handle_manually) if (wsi->context->event_loop_ops->close_handle_manually)
wsi->context->event_loop_ops->close_handle_manually(wsi); wsi->context->event_loop_ops->close_handle_manually(wsi);
else else
compatible_close(wsi->desc.sockfd); compatible_close(wsi->desc.sockfd);
__remove_wsi_socket_from_fds(wsi);
#if defined(LWS_WITH_TLS) #if defined(LWS_WITH_TLS)
wsi->tls.use_ssl = ssl; wsi->tls.use_ssl = ssl;
#else #else
@ -717,7 +717,7 @@ lws_client_reset(struct lws **pwsi, int ssl, const char *address, int port,
} }
#ifdef LWS_WITH_HTTP_PROXY #ifdef LWS_WITH_HTTP_PROXY
static hubbub_error hubbub_error
html_parser_cb(const hubbub_token *token, void *pw) html_parser_cb(const hubbub_token *token, void *pw)
{ {
struct lws_rewrite *r = (struct lws_rewrite *)pw; struct lws_rewrite *r = (struct lws_rewrite *)pw;
@ -846,7 +846,7 @@ html_parser_cb(const hubbub_token *token, void *pw)
#endif #endif
static char * char *
lws_strdup(const char *s) lws_strdup(const char *s)
{ {
char *d = lws_malloc(strlen(s) + 1, "strdup"); char *d = lws_malloc(strlen(s) + 1, "strdup");

View file

@ -93,7 +93,7 @@ lws_client_socket_service(struct lws *wsi, struct lws_pollfd *pollfd,
char *sb = p; char *sb = p;
int n = 0; int n = 0;
#if defined(LWS_WITH_SOCKS5) #if defined(LWS_WITH_SOCKS5)
char conn_mode = 0, pending_timeout = 0; int conn_mode = 0, pending_timeout = 0;
#endif #endif
if ((pollfd->revents & LWS_POLLOUT) && if ((pollfd->revents & LWS_POLLOUT) &&
@ -252,6 +252,8 @@ socks_reply_fail:
/* clear his proxy connection timeout */ /* clear his proxy connection timeout */
lws_set_timeout(wsi, NO_PENDING_TIMEOUT, 0); lws_set_timeout(wsi, NO_PENDING_TIMEOUT, 0);
goto start_ws_handshake; goto start_ws_handshake;
default:
break;
} }
break; break;
#endif #endif
@ -578,6 +580,8 @@ lws_http_transaction_completed_client(struct lws *wsi)
"queued client done"); "queued client done");
} }
_lws_header_table_reset(wsi->http.ah);
/* after the first one, they can only be coming from the queue */ /* after the first one, they can only be coming from the queue */
wsi->transaction_from_pipeline_queue = 1; wsi->transaction_from_pipeline_queue = 1;
@ -629,12 +633,20 @@ lws_http_transaction_completed_client(struct lws *wsi)
} }
LWS_VISIBLE LWS_EXTERN unsigned int LWS_VISIBLE LWS_EXTERN unsigned int
lws_http_client_http_response(struct lws *wsi) lws_http_client_http_response(struct lws *_wsi)
{ {
if (!wsi->http.ah) struct lws *wsi;
return 0; unsigned int resp;
return wsi->http.ah->http_response; if (_wsi->http.ah && _wsi->http.ah->http_response)
return _wsi->http.ah->http_response;
lws_vhost_lock(_wsi->vhost);
wsi = _lws_client_wsi_master(_wsi);
resp = wsi->http.ah->http_response;
lws_vhost_unlock(_wsi->vhost);
return resp;
} }
#endif #endif
#if defined(LWS_PLAT_OPTEE) #if defined(LWS_PLAT_OPTEE)
@ -781,7 +793,7 @@ lws_client_interpret_server_handshake(struct lws *wsi)
q = strrchr(new_path, '/'); q = strrchr(new_path, '/');
if (q) if (q)
lws_strncpy(q + 1, p, sizeof(new_path) - lws_strncpy(q + 1, p, sizeof(new_path) -
(q - new_path)); (q - new_path) - 1);
else else
path = p; path = p;
} }
@ -910,9 +922,9 @@ lws_client_interpret_server_handshake(struct lws *wsi)
* we seem to be good to go, give client last chance to check * we seem to be good to go, give client last chance to check
* headers and OK it * headers and OK it
*/ */
if (wsi->protocol->callback(wsi, if (w->protocol->callback(w,
LWS_CALLBACK_CLIENT_FILTER_PRE_ESTABLISH, LWS_CALLBACK_CLIENT_FILTER_PRE_ESTABLISH,
wsi->user_space, NULL, 0)) { w->user_space, NULL, 0)) {
cce = "HS: disallowed by client filter"; cce = "HS: disallowed by client filter";
goto bail2; goto bail2;
@ -924,9 +936,9 @@ lws_client_interpret_server_handshake(struct lws *wsi)
wsi->rxflow_change_to = LWS_RXFLOW_ALLOW; wsi->rxflow_change_to = LWS_RXFLOW_ALLOW;
/* call him back to inform him he is up */ /* call him back to inform him he is up */
if (wsi->protocol->callback(wsi, if (w->protocol->callback(w,
LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP, LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP,
wsi->user_space, NULL, 0)) { w->user_space, NULL, 0)) {
cce = "HS: disallowed at ESTABLISHED"; cce = "HS: disallowed at ESTABLISHED";
goto bail3; goto bail3;
} }
@ -964,9 +976,9 @@ bail2:
n = 0; n = 0;
if (cce) if (cce)
n = (int)strlen(cce); n = (int)strlen(cce);
wsi->protocol->callback(wsi, w->protocol->callback(w,
LWS_CALLBACK_CLIENT_CONNECTION_ERROR, LWS_CALLBACK_CLIENT_CONNECTION_ERROR,
wsi->user_space, (void *)cce, w->user_space, (void *)cce,
(unsigned int)n); (unsigned int)n);
} }
wsi->already_did_cce = 1; wsi->already_did_cce = 1;
@ -1228,4 +1240,4 @@ completed:
return 0; return 0;
} }
#endif #endif

View file

@ -26,7 +26,7 @@
const unsigned char * const unsigned char *
lws_token_to_string(enum lws_token_indexes token) lws_token_to_string(enum lws_token_indexes token)
{ {
if ((unsigned int)token >= ARRAY_SIZE(set)) if ((unsigned int)token >= LWS_ARRAY_SIZE(set))
return NULL; return NULL;
return (unsigned char *)set[token]; return (unsigned char *)set[token];
@ -149,9 +149,17 @@ lws_add_http_common_headers(struct lws *wsi, unsigned int code,
(int)strlen(content_type), p, end)) (int)strlen(content_type), p, end))
return 1; return 1;
if (content_len != LWS_ILLEGAL_HTTP_CONTENT_LEN && if (content_len != LWS_ILLEGAL_HTTP_CONTENT_LEN) {
lws_add_http_header_content_length(wsi, content_len, p, end)) if (lws_add_http_header_content_length(wsi, content_len, p, end))
return 1; return 1;
} else {
if (lws_add_http_header_by_token(wsi, WSI_TOKEN_CONNECTION,
(unsigned char *)"close", 5,
p, end))
return 1;
wsi->http.connection_type = HTTP_CONNECTION_CLOSE;
}
return 0; return 0;
} }
@ -204,34 +212,40 @@ lws_add_http_header_status(struct lws *wsi, unsigned int _code,
#endif #endif
#ifdef LWS_WITH_HTTP2 #ifdef LWS_WITH_HTTP2
if (lwsi_role_h2(wsi) || lwsi_role_h2_ENCAPSULATION(wsi)) if (lwsi_role_h2(wsi) || lwsi_role_h2_ENCAPSULATION(wsi)) {
return lws_add_http2_header_status(wsi, code, p, end); n = lws_add_http2_header_status(wsi, code, p, end);
if (n)
return n;
} else
#endif #endif
if (code >= 400 && code < (400 + ARRAY_SIZE(err400))) {
description = err400[code - 400]; if (code >= 400 && code < (400 + LWS_ARRAY_SIZE(err400)))
if (code >= 500 && code < (500 + ARRAY_SIZE(err500))) description = err400[code - 400];
description = err500[code - 500]; if (code >= 500 && code < (500 + LWS_ARRAY_SIZE(err500)))
description = err500[code - 500];
if (code == 100) if (code == 100)
description = "Continue"; description = "Continue";
if (code == 200) if (code == 200)
description = "OK"; description = "OK";
if (code == 304) if (code == 304)
description = "Not Modified"; description = "Not Modified";
else else
if (code >= 300 && code < 400) if (code >= 300 && code < 400)
description = "Redirect"; description = "Redirect";
if (wsi->http.request_version < ARRAY_SIZE(hver)) if (wsi->http.request_version < LWS_ARRAY_SIZE(hver))
p1 = hver[wsi->http.request_version]; p1 = hver[wsi->http.request_version];
else else
p1 = hver[0]; p1 = hver[0];
n = sprintf((char *)code_and_desc, "%s %u %s", p1, code, description); n = sprintf((char *)code_and_desc, "%s %u %s", p1, code,
description);
if (lws_add_http_header_by_name(wsi, NULL, code_and_desc, n, p, end))
return 1;
if (lws_add_http_header_by_name(wsi, NULL, code_and_desc, n, p,
end))
return 1;
}
headers = wsi->vhost->headers; headers = wsi->vhost->headers;
while (headers) { while (headers) {
if (lws_add_http_header_by_name(wsi, if (lws_add_http_header_by_name(wsi,

View file

@ -227,6 +227,7 @@ struct _lws_http_mode_related {
#if defined(LWS_WITH_HTTP_PROXY) #if defined(LWS_WITH_HTTP_PROXY)
unsigned int perform_rewrite:1; unsigned int perform_rewrite:1;
#endif #endif
unsigned int deferred_transaction_completed:1;
}; };

View file

@ -205,6 +205,7 @@ struct jpargs {
unsigned int enable_client_ssl:1; unsigned int enable_client_ssl:1;
unsigned int fresh_mount:1; unsigned int fresh_mount:1;
unsigned int any_vhosts:1; unsigned int any_vhosts:1;
unsigned int chunk:1;
}; };
static void * static void *
@ -213,6 +214,8 @@ lwsws_align(struct jpargs *a)
if ((lws_intptr_t)(a->p) & 15) if ((lws_intptr_t)(a->p) & 15)
a->p += 16 - ((lws_intptr_t)(a->p) & 15); a->p += 16 - ((lws_intptr_t)(a->p) & 15);
a->chunk = 0;
return a->p; return a->p;
} }
@ -225,7 +228,7 @@ arg_to_bool(const char *s)
if (n) if (n)
return 1; return 1;
for (n = 0; n < (int)ARRAY_SIZE(on); n++) for (n = 0; n < (int)LWS_ARRAY_SIZE(on); n++)
if (!strcasecmp(s, on[n])) if (!strcasecmp(s, on[n]))
return 1; return 1;
@ -413,25 +416,30 @@ lejp_vhosts_cb(struct lejp_ctx *ctx, char reason)
} }
/* this catches, eg, vhosts[].headers[].xxx */ /* this catches, eg, vhosts[].headers[].xxx */
if (reason == LEJPCB_VAL_STR_END && if ((reason == LEJPCB_VAL_STR_END || reason == LEJPCB_VAL_STR_CHUNK) &&
ctx->path_match == LEJPVP_HEADERS_NAME + 1) { ctx->path_match == LEJPVP_HEADERS_NAME + 1) {
headers = lwsws_align(a); if (!a->chunk) {
a->p += sizeof(*headers); headers = lwsws_align(a);
a->p += sizeof(*headers);
n = lejp_get_wildcard(ctx, 0, a->p, a->end - a->p); n = lejp_get_wildcard(ctx, 0, a->p,
/* ie, enable this protocol, no options yet */ lws_ptr_diff(a->end, a->p));
headers->next = a->info->headers; /* ie, add this header */
a->info->headers = headers; headers->next = a->info->headers;
headers->name = a->p; a->info->headers = headers;
// lwsl_notice(" adding header %s=%s\n", a->p, ctx->buf); headers->name = a->p;
a->p += n - 1;
*(a->p++) = ':'; lwsl_notice(" adding header %s=%s\n", a->p, ctx->buf);
if (a->p < a->end) a->p += n - 1;
*(a->p++) = '\0'; *(a->p++) = ':';
else if (a->p < a->end)
*(a->p - 1) = '\0'; *(a->p++) = '\0';
headers->value = a->p; else
headers->options = NULL; *(a->p - 1) = '\0';
headers->value = a->p;
headers->options = NULL;
}
a->chunk = reason == LEJPCB_VAL_STR_CHUNK;
goto dostring; goto dostring;
} }
@ -502,7 +510,7 @@ lejp_vhosts_cb(struct lejp_ctx *ctx, char reason)
if (a->last) if (a->last)
a->last->mount_next = m; a->last->mount_next = m;
for (n = 0; n < (int)ARRAY_SIZE(mount_protocols); n++) for (n = 0; n < (int)LWS_ARRAY_SIZE(mount_protocols); n++)
if (!strncmp(a->m.origin, mount_protocols[n], if (!strncmp(a->m.origin, mount_protocols[n],
strlen(mount_protocols[n]))) { strlen(mount_protocols[n]))) {
lwsl_info("----%s\n", a->m.origin); lwsl_info("----%s\n", a->m.origin);
@ -512,7 +520,7 @@ lejp_vhosts_cb(struct lejp_ctx *ctx, char reason)
break; break;
} }
if (n == (int)ARRAY_SIZE(mount_protocols)) { if (n == (int)LWS_ARRAY_SIZE(mount_protocols)) {
lwsl_err("unsupported protocol:// %s\n", a->m.origin); lwsl_err("unsupported protocol:// %s\n", a->m.origin);
return 1; return 1;
} }
@ -750,6 +758,7 @@ lejp_vhosts_cb(struct lejp_ctx *ctx, char reason)
dostring: dostring:
p = ctx->buf; p = ctx->buf;
p[LEJP_STRING_CHUNK] = '\0';
p1 = strstr(p, ESC_INSTALL_DATADIR); p1 = strstr(p, ESC_INSTALL_DATADIR);
if (p1) { if (p1) {
n = p1 - p; n = p1 - p;
@ -762,7 +771,8 @@ dostring:
} }
a->p += lws_snprintf(a->p, a->end - a->p, "%s", p); a->p += lws_snprintf(a->p, a->end - a->p, "%s", p);
*(a->p)++ = '\0'; if (reason == LEJPCB_VAL_STR_END)
*(a->p)++ = '\0';
return 0; return 0;
} }
@ -779,7 +789,7 @@ lwsws_get_config(void *user, const char *f, const char * const *paths,
struct lejp_ctx ctx; struct lejp_ctx ctx;
int n, m, fd; int n, m, fd;
fd = open(f, O_RDONLY); fd = lws_open(f, O_RDONLY);
if (fd < 0) { if (fd < 0) {
lwsl_err("Cannot open %s\n", f); lwsl_err("Cannot open %s\n", f);
return 2; return 2;
@ -927,11 +937,11 @@ lwsws_get_config_globals(struct lws_context_creation_info *info, const char *d,
lws_snprintf(dd, sizeof(dd) - 1, "%s/conf", d); lws_snprintf(dd, sizeof(dd) - 1, "%s/conf", d);
if (lwsws_get_config(&a, dd, paths_global, if (lwsws_get_config(&a, dd, paths_global,
ARRAY_SIZE(paths_global), lejp_globals_cb) > 1) LWS_ARRAY_SIZE(paths_global), lejp_globals_cb) > 1)
return 1; return 1;
lws_snprintf(dd, sizeof(dd) - 1, "%s/conf.d", d); lws_snprintf(dd, sizeof(dd) - 1, "%s/conf.d", d);
if (lwsws_get_config_d(&a, dd, paths_global, if (lwsws_get_config_d(&a, dd, paths_global,
ARRAY_SIZE(paths_global), lejp_globals_cb) > 1) LWS_ARRAY_SIZE(paths_global), lejp_globals_cb) > 1)
return 1; return 1;
a.plugin_dirs[a.count_plugin_dirs] = NULL; a.plugin_dirs[a.count_plugin_dirs] = NULL;
@ -962,11 +972,11 @@ lwsws_get_config_vhosts(struct lws_context *context,
lws_snprintf(dd, sizeof(dd) - 1, "%s/conf", d); lws_snprintf(dd, sizeof(dd) - 1, "%s/conf", d);
if (lwsws_get_config(&a, dd, paths_vhosts, if (lwsws_get_config(&a, dd, paths_vhosts,
ARRAY_SIZE(paths_vhosts), lejp_vhosts_cb) > 1) LWS_ARRAY_SIZE(paths_vhosts), lejp_vhosts_cb) > 1)
return 1; return 1;
lws_snprintf(dd, sizeof(dd) - 1, "%s/conf.d", d); lws_snprintf(dd, sizeof(dd) - 1, "%s/conf.d", d);
if (lwsws_get_config_d(&a, dd, paths_vhosts, if (lwsws_get_config_d(&a, dd, paths_vhosts,
ARRAY_SIZE(paths_vhosts), lejp_vhosts_cb) > 1) LWS_ARRAY_SIZE(paths_vhosts), lejp_vhosts_cb) > 1)
return 1; return 1;
*cs = a.p; *cs = a.p;

View file

@ -563,7 +563,7 @@ int LWS_WARN_UNUSED_RESULT
lws_hdr_simple_create(struct lws *wsi, enum lws_token_indexes h, const char *s) lws_hdr_simple_create(struct lws *wsi, enum lws_token_indexes h, const char *s)
{ {
wsi->http.ah->nfrag++; wsi->http.ah->nfrag++;
if (wsi->http.ah->nfrag == ARRAY_SIZE(wsi->http.ah->frags)) { if (wsi->http.ah->nfrag == LWS_ARRAY_SIZE(wsi->http.ah->frags)) {
lwsl_warn("More hdr frags than we can deal with, dropping\n"); lwsl_warn("More hdr frags than we can deal with, dropping\n");
return -1; return -1;
} }
@ -677,18 +677,16 @@ lws_parse_urldecode(struct lws *wsi, uint8_t *_c)
return -1; return -1;
/* genuine delimiter */ /* genuine delimiter */
if ((c == '&' || c == ';') && !enc) { if ((c == '&' || c == ';') && !enc) {
if (issue_char(wsi, c) < 0) if (issue_char(wsi, '\0') < 0)
return -1; return -1;
/* swallow the terminator */
ah->frags[ah->nfrag].len--;
/* link to next fragment */ /* link to next fragment */
ah->frags[ah->nfrag].nfrag = ah->nfrag + 1; ah->frags[ah->nfrag].nfrag = ah->nfrag + 1;
ah->nfrag++; ah->nfrag++;
if (ah->nfrag >= ARRAY_SIZE(ah->frags)) if (ah->nfrag >= LWS_ARRAY_SIZE(ah->frags))
goto excessive; goto excessive;
/* start next fragment after the & */ /* start next fragment after the & */
ah->post_literal_equal = 0; ah->post_literal_equal = 0;
ah->frags[ah->nfrag].offset = ah->pos; ah->frags[ah->nfrag].offset = ++ah->pos;
ah->frags[ah->nfrag].len = 0; ah->frags[ah->nfrag].len = 0;
ah->frags[ah->nfrag].nfrag = 0; ah->frags[ah->nfrag].nfrag = 0;
goto swallow; goto swallow;
@ -787,9 +785,9 @@ lws_parse_urldecode(struct lws *wsi, uint8_t *_c)
/* move to using WSI_TOKEN_HTTP_URI_ARGS */ /* move to using WSI_TOKEN_HTTP_URI_ARGS */
ah->nfrag++; ah->nfrag++;
if (ah->nfrag >= ARRAY_SIZE(ah->frags)) if (ah->nfrag >= LWS_ARRAY_SIZE(ah->frags))
goto excessive; goto excessive;
ah->frags[ah->nfrag].offset = ah->pos; ah->frags[ah->nfrag].offset = ++ah->pos;
ah->frags[ah->nfrag].len = 0; ah->frags[ah->nfrag].len = 0;
ah->frags[ah->nfrag].nfrag = 0; ah->frags[ah->nfrag].nfrag = 0;
@ -852,10 +850,10 @@ lws_parse(struct lws *wsi, unsigned char *buf, int *len)
c == ' ') c == ' ')
break; break;
for (m = 0; m < ARRAY_SIZE(methods); m++) for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
if (ah->parser_state == methods[m]) if (ah->parser_state == methods[m])
break; break;
if (m == ARRAY_SIZE(methods)) if (m == LWS_ARRAY_SIZE(methods))
/* it was not any of the methods */ /* it was not any of the methods */
goto check_eol; goto check_eol;
@ -983,7 +981,7 @@ nope:
if (ah->lextable_pos < 0 && lwsi_role_h1(wsi) && if (ah->lextable_pos < 0 && lwsi_role_h1(wsi) &&
lwsi_role_server(wsi)) { lwsi_role_server(wsi)) {
/* this is not a header we know about */ /* this is not a header we know about */
for (m = 0; m < ARRAY_SIZE(methods); m++) for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
if (ah->frag_index[methods[m]]) { if (ah->frag_index[methods[m]]) {
/* /*
* already had the method, no idea what * already had the method, no idea what
@ -996,7 +994,7 @@ nope:
* hm it's an unknown http method from a client in fact, * hm it's an unknown http method from a client in fact,
* it cannot be valid http * it cannot be valid http
*/ */
if (m == ARRAY_SIZE(methods)) { if (m == LWS_ARRAY_SIZE(methods)) {
/* /*
* are we set up to accept raw in these cases? * are we set up to accept raw in these cases?
*/ */
@ -1025,7 +1023,7 @@ nope:
lextable[ah->lextable_pos + 1]; lextable[ah->lextable_pos + 1];
lwsl_parser("known hdr %d\n", n); lwsl_parser("known hdr %d\n", n);
for (m = 0; m < ARRAY_SIZE(methods); m++) for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
if (n == methods[m] && if (n == methods[m] &&
ah->frag_index[methods[m]]) { ah->frag_index[methods[m]]) {
lwsl_warn("Duplicated method\n"); lwsl_warn("Duplicated method\n");
@ -1061,7 +1059,7 @@ nope:
start_fragment: start_fragment:
ah->nfrag++; ah->nfrag++;
excessive: excessive:
if (ah->nfrag == ARRAY_SIZE(ah->frags)) { if (ah->nfrag == LWS_ARRAY_SIZE(ah->frags)) {
lwsl_warn("More hdr frags than we can deal with\n"); lwsl_warn("More hdr frags than we can deal with\n");
return -1; return -1;
} }

View file

@ -131,6 +131,17 @@ done_list:
(void)n; (void)n;
#if defined(__linux__) #if defined(__linux__)
#ifdef LWS_WITH_UNIX_SOCK
/*
* A Unix domain sockets cannot be bound for several times, even if we set
* the SO_REUSE* options on.
* However, fortunately, each thread is able to independently listen when
* running on a reasonably new Linux kernel. So we can safely assume
* creating just one listening socket for a multi-threaded environment won't
* fail in most cases.
*/
if (!LWS_UNIX_SOCK_ENABLED(vhost))
#endif
limit = vhost->context->count_threads; limit = vhost->context->count_threads;
#endif #endif
@ -694,7 +705,7 @@ lws_find_string_in_file(const char *filename, const char *string, int stringlen)
char buf[128]; char buf[128];
int fd, match = 0, pos = 0, n = 0, hit = 0; int fd, match = 0, pos = 0, n = 0, hit = 0;
fd = open(filename, O_RDONLY); fd = lws_open(filename, O_RDONLY);
if (fd < 0) { if (fd < 0) {
lwsl_err("can't open auth file: %s\n", filename); lwsl_err("can't open auth file: %s\n", filename);
return 0; return 0;
@ -812,7 +823,7 @@ lws_http_get_uri_and_method(struct lws *wsi, char **puri_ptr, int *puri_len)
{ {
int n, count = 0; int n, count = 0;
for (n = 0; n < (int)ARRAY_SIZE(methods); n++) for (n = 0; n < (int)LWS_ARRAY_SIZE(methods); n++)
if (lws_hdr_total_length(wsi, methods[n])) if (lws_hdr_total_length(wsi, methods[n]))
count++; count++;
if (!count) { if (!count) {
@ -827,7 +838,7 @@ lws_http_get_uri_and_method(struct lws *wsi, char **puri_ptr, int *puri_len)
return -1; return -1;
} }
for (n = 0; n < (int)ARRAY_SIZE(methods); n++) for (n = 0; n < (int)LWS_ARRAY_SIZE(methods); n++)
if (lws_hdr_total_length(wsi, methods[n])) { if (lws_hdr_total_length(wsi, methods[n])) {
*puri_ptr = lws_hdr_simple_ptr(wsi, methods[n]); *puri_ptr = lws_hdr_simple_ptr(wsi, methods[n]);
*puri_len = lws_hdr_total_length(wsi, methods[n]); *puri_len = lws_hdr_total_length(wsi, methods[n]);
@ -857,7 +868,7 @@ lws_http_action(struct lws *wsi)
}; };
meth = lws_http_get_uri_and_method(wsi, &uri_ptr, &uri_len); meth = lws_http_get_uri_and_method(wsi, &uri_ptr, &uri_len);
if (meth < 0 || meth >= (int)ARRAY_SIZE(method_names)) if (meth < 0 || meth >= (int)LWS_ARRAY_SIZE(method_names))
goto bail_nuke_ah; goto bail_nuke_ah;
/* we insist on absolute paths */ /* we insist on absolute paths */
@ -1128,7 +1139,7 @@ lws_http_action(struct lws *wsi)
} }
if (pcolon > pslash) if (pcolon > pslash)
pcolon = NULL; pcolon = NULL;
if (pcolon) if (pcolon)
n = pcolon - hit->origin; n = pcolon - hit->origin;
else else
@ -1142,13 +1153,13 @@ lws_http_action(struct lws *wsi)
i.address = ads; i.address = ads;
i.port = 80; i.port = 80;
if (hit->origin_protocol == LWSMPRO_HTTPS) { if (hit->origin_protocol == LWSMPRO_HTTPS) {
i.port = 443; i.port = 443;
i.ssl_connection = 1; i.ssl_connection = 1;
} }
if (pcolon) if (pcolon)
i.port = atoi(pcolon + 1); i.port = atoi(pcolon + 1);
lws_snprintf(rpath, sizeof(rpath) - 1, "/%s/%s", pslash + 1, lws_snprintf(rpath, sizeof(rpath) - 1, "/%s/%s", pslash + 1,
uri_ptr + hit->mountpoint_len); uri_ptr + hit->mountpoint_len);
lws_clean_url(rpath); lws_clean_url(rpath);
@ -1164,7 +1175,7 @@ lws_http_action(struct lws *wsi)
p++; p++;
} }
} }
i.path = rpath; i.path = rpath;
i.host = i.address; i.host = i.address;
@ -1178,7 +1189,7 @@ lws_http_action(struct lws *wsi)
"from %s, to %s\n", "from %s, to %s\n",
i.address, i.port, i.path, i.ssl_connection, i.address, i.port, i.path, i.ssl_connection,
i.uri_replace_from, i.uri_replace_to); i.uri_replace_from, i.uri_replace_to);
if (!lws_client_connect_via_info(&i)) { if (!lws_client_connect_via_info(&i)) {
lwsl_err("proxy connect fail\n"); lwsl_err("proxy connect fail\n");
return 1; return 1;
@ -1714,12 +1725,31 @@ lws_http_transaction_completed(struct lws *wsi)
{ {
int n = NO_PENDING_TIMEOUT; int n = NO_PENDING_TIMEOUT;
if (wsi->trunc_len) {
/*
* ...so he tried to send something large as the http reply,
* it went as a partial, but he immediately said the
* transaction was completed.
*
* Defer the transaction completed until the last part of the
* partial is sent.
*/
lwsl_notice("%s: deferring due to partial\n", __func__);
wsi->http.deferred_transaction_completed = 1;
return 0;
}
lwsl_info("%s: wsi %p\n", __func__, wsi); lwsl_info("%s: wsi %p\n", __func__, wsi);
lws_access_log(wsi); lws_access_log(wsi);
if (!wsi->hdr_parsing_completed) { if (!wsi->hdr_parsing_completed) {
lwsl_notice("%s: ignoring, ah parsing incomplete\n", __func__); char peer[64];
lws_get_peer_simple(wsi, peer, sizeof(peer) - 1);
peer[sizeof(peer) - 1] = '\0';
lwsl_notice("%s: (from %s) ignoring, ah parsing incomplete\n",
__func__, peer);
return 0; return 0;
} }

View file

@ -450,7 +450,7 @@ ping_drop:
break; break;
case LWSWSOPC_PONG: case LWSWSOPC_PONG:
lwsl_info("client receied pong\n"); lwsl_info("client received pong\n");
lwsl_hexdump(&wsi->ws->rx_ubuf[LWS_PRE], lwsl_hexdump(&wsi->ws->rx_ubuf[LWS_PRE],
wsi->ws->rx_ubuf_head); wsi->ws->rx_ubuf_head);
@ -488,9 +488,6 @@ ping_drop:
ebuf.token = &wsi->ws->rx_ubuf[LWS_PRE]; ebuf.token = &wsi->ws->rx_ubuf[LWS_PRE];
ebuf.len = wsi->ws->rx_ubuf_head; ebuf.len = wsi->ws->rx_ubuf_head;
if (wsi->ws->opcode == LWSWSOPC_PONG && !ebuf.len)
goto already_done;
#if !defined(LWS_WITHOUT_EXTENSIONS) #if !defined(LWS_WITHOUT_EXTENSIONS)
drain_extension: drain_extension:
lwsl_ext("%s: passing %d to ext\n", __func__, ebuf.len); lwsl_ext("%s: passing %d to ext\n", __func__, ebuf.len);
@ -504,14 +501,12 @@ drain_extension:
#endif #endif
lwsl_debug("post inflate ebuf len %d\n", ebuf.len); lwsl_debug("post inflate ebuf len %d\n", ebuf.len);
if (
#if !defined(LWS_WITHOUT_EXTENSIONS) #if !defined(LWS_WITHOUT_EXTENSIONS)
rx_draining_ext && if (rx_draining_ext && !ebuf.len) {
#endif
!ebuf.len) {
lwsl_debug(" --- ending drain on 0 read result\n"); lwsl_debug(" --- ending drain on 0 read result\n");
goto already_done; goto already_done;
} }
#endif
if (wsi->ws->check_utf8 && !wsi->ws->defeat_check_utf8) { if (wsi->ws->check_utf8 && !wsi->ws->defeat_check_utf8) {
if (lws_check_utf8(&wsi->ws->utf8, if (lws_check_utf8(&wsi->ws->utf8,

View file

@ -1246,8 +1246,7 @@ int rops_handle_POLLOUT_ws(struct lws *wsi)
return LWS_HP_RET_BAIL_OK; return LWS_HP_RET_BAIL_OK;
} }
if (lwsi_role_client(wsi) && !wsi->socket_is_permanently_unusable && if (!wsi->socket_is_permanently_unusable && wsi->ws->send_check_ping) {
wsi->ws->send_check_ping) {
lwsl_info("issuing ping on wsi %p\n", wsi); lwsl_info("issuing ping on wsi %p\n", wsi);
wsi->ws->send_check_ping = 0; wsi->ws->send_check_ping = 0;
@ -1282,7 +1281,7 @@ int rops_handle_POLLOUT_ws(struct lws *wsi)
* payload ordering, but since they are always complete * payload ordering, but since they are always complete
* fragments control packets can interleave OK. * fragments control packets can interleave OK.
*/ */
if (lwsi_role_client(wsi) && wsi->ws->tx_draining_ext) { if (wsi->ws->tx_draining_ext) {
lwsl_ext("SERVICING TX EXT DRAINING\n"); lwsl_ext("SERVICING TX EXT DRAINING\n");
if (lws_write(wsi, NULL, 0, LWS_WRITE_CONTINUATION) < 0) if (lws_write(wsi, NULL, 0, LWS_WRITE_CONTINUATION) < 0)
return LWS_HP_RET_BAIL_DIE; return LWS_HP_RET_BAIL_DIE;
@ -1292,8 +1291,10 @@ int rops_handle_POLLOUT_ws(struct lws *wsi)
/* Priority 6: extensions /* Priority 6: extensions
*/ */
if (!wsi->ws->extension_data_pending) if (!wsi->ws->extension_data_pending && !wsi->ws->tx_draining_ext) {
lwsl_ext("%s: !wsi->ws->extension_data_pending\n", __func__);
return LWS_HP_RET_USER_SERVICE; return LWS_HP_RET_USER_SERVICE;
}
/* /*
* check in on the active extensions, see if they * check in on the active extensions, see if they
@ -1412,15 +1413,13 @@ rops_periodic_checks_ws(struct lws_context *context, int tsi, time_t now)
wsi->ws->time_next_ping_check) > wsi->ws->time_next_ping_check) >
context->ws_ping_pong_interval) { context->ws_ping_pong_interval) {
lwsl_info("req pp on wsi %p\n", lwsl_info("req pp on wsi %p\n", wsi);
wsi);
wsi->ws->send_check_ping = 1; wsi->ws->send_check_ping = 1;
lws_set_timeout(wsi, lws_set_timeout(wsi,
PENDING_TIMEOUT_WS_PONG_CHECK_SEND_PING, PENDING_TIMEOUT_WS_PONG_CHECK_SEND_PING,
context->timeout_secs); context->timeout_secs);
lws_callback_on_writable(wsi); lws_callback_on_writable(wsi);
wsi->ws->time_next_ping_check = wsi->ws->time_next_ping_check = now;
now;
} }
wsi = wsi->same_vh_protocol_next; wsi = wsi->same_vh_protocol_next;
} }
@ -1466,6 +1465,9 @@ rops_service_flag_pending_ws(struct lws_context *context, int tsi)
static int static int
rops_close_via_role_protocol_ws(struct lws *wsi, enum lws_close_status reason) rops_close_via_role_protocol_ws(struct lws *wsi, enum lws_close_status reason)
{ {
if (!wsi->ws)
return 0;
if (!wsi->ws->close_in_ping_buffer_len && /* already a reason */ if (!wsi->ws->close_in_ping_buffer_len && /* already a reason */
(reason == LWS_CLOSE_STATUS_NOSTATUS || (reason == LWS_CLOSE_STATUS_NOSTATUS ||
reason == LWS_CLOSE_STATUS_NOSTATUS_CONTEXT_DESTROY)) reason == LWS_CLOSE_STATUS_NOSTATUS_CONTEXT_DESTROY))
@ -1512,7 +1514,7 @@ rops_close_role_ws(struct lws_context_per_thread *pt, struct lws *wsi)
if (wsi->ws->tx_draining_ext) { if (wsi->ws->tx_draining_ext) {
struct lws **w = &pt->ws.tx_draining_ext_list; struct lws **w = &pt->ws.tx_draining_ext_list;
lwsl_notice("%s: CLEARING tx_draining_ext\n", __func__); lwsl_ext("%s: CLEARING tx_draining_ext\n", __func__);
wsi->ws->tx_draining_ext = 0; wsi->ws->tx_draining_ext = 0;
/* remove us from context draining ext list */ /* remove us from context draining ext list */
while (*w) { while (*w) {
@ -1563,7 +1565,7 @@ rops_write_role_protocol_ws(struct lws *wsi, unsigned char *buf, size_t len,
/* remove us from the list */ /* remove us from the list */
struct lws **w = &pt->ws.tx_draining_ext_list; struct lws **w = &pt->ws.tx_draining_ext_list;
lwsl_notice("%s: CLEARING tx_draining_ext\n", __func__); lwsl_ext("%s: CLEARING tx_draining_ext\n", __func__);
wsi->ws->tx_draining_ext = 0; wsi->ws->tx_draining_ext = 0;
/* remove us from context draining ext list */ /* remove us from context draining ext list */
while (*w) { while (*w) {
@ -1588,7 +1590,7 @@ rops_write_role_protocol_ws(struct lws *wsi, unsigned char *buf, size_t len,
if (!(wpt & LWS_WRITE_NO_FIN) && len) if (!(wpt & LWS_WRITE_NO_FIN) && len)
*wp &= ~LWS_WRITE_NO_FIN; *wp &= ~LWS_WRITE_NO_FIN;
lwsl_notice("FORCED draining wp to 0x%02X (stashed 0x%02X, incoming 0x%02X)\n", *wp, lwsl_ext("FORCED draining wp to 0x%02X (stashed 0x%02X, incoming 0x%02X)\n", *wp,
wsi->ws->tx_draining_stashed_wp, wpt); wsi->ws->tx_draining_stashed_wp, wpt);
// assert(0); // assert(0);
} }
@ -1644,7 +1646,7 @@ rops_write_role_protocol_ws(struct lws *wsi, unsigned char *buf, size_t len,
// lwsl_notice("ext processed %d plaintext into %d compressed (wp 0x%x)\n", m, (int)ebuf.len, *wp); // lwsl_notice("ext processed %d plaintext into %d compressed (wp 0x%x)\n", m, (int)ebuf.len, *wp);
if (n && ebuf.len) { if (n && ebuf.len) {
lwsl_notice("write drain len %d (wp 0x%x) SETTING tx_draining_ext\n", (int)ebuf.len, *wp); lwsl_ext("write drain len %d (wp 0x%x) SETTING tx_draining_ext\n", (int)ebuf.len, *wp);
/* extension requires further draining */ /* extension requires further draining */
wsi->ws->tx_draining_ext = 1; wsi->ws->tx_draining_ext = 1;
wsi->ws->tx_draining_ext_list = pt->ws.tx_draining_ext_list; wsi->ws->tx_draining_ext_list = pt->ws.tx_draining_ext_list;

View file

@ -631,7 +631,7 @@ lws_tls_acme_sni_csr_create(struct lws_context *context, const char *elements[],
/* subject must be formatted like "C=TW,O=warmcat,CN=myserver" */ /* subject must be formatted like "C=TW,O=warmcat,CN=myserver" */
for (n = 0; n < (int)ARRAY_SIZE(x5); n++) { for (n = 0; n < (int)LWS_ARRAY_SIZE(x5); n++) {
if (p != subject) if (p != subject)
*p++ = ','; *p++ = ',';
if (elements[n]) if (elements[n])

View file

@ -121,8 +121,6 @@ lws_ssl_capable_read(struct lws *wsi, unsigned char *buf, int len)
if (wsi->vhost) if (wsi->vhost)
wsi->vhost->conn_stats.rx += n; wsi->vhost->conn_stats.rx += n;
lws_restart_ws_ping_pong_timer(wsi);
/* /*
* if it was our buffer that limited what we read, * if it was our buffer that limited what we read,
* check if SSL has additional data pending inside SSL buffers. * check if SSL has additional data pending inside SSL buffers.

View file

@ -37,7 +37,11 @@ typedef void RSA;
typedef void STACK; typedef void STACK;
typedef void BIO; typedef void BIO;
#if defined(WIN32) || defined(_WIN32)
#define ossl_inline __inline
#else
#define ossl_inline inline #define ossl_inline inline
#endif
#define SSL_METHOD_CALL(f, s, ...) s->method->func->ssl_##f(s, ##__VA_ARGS__) #define SSL_METHOD_CALL(f, s, ...) s->method->func->ssl_##f(s, ##__VA_ARGS__)
#define X509_METHOD_CALL(f, x, ...) x->method->x509_##f(x, ##__VA_ARGS__) #define X509_METHOD_CALL(f, x, ...) x->method->x509_##f(x, ##__VA_ARGS__)

View file

@ -1,15 +1,15 @@
diff --git a/thirdparty/libwebsockets/plat/lws-plat-win.c b/thirdparty/libwebsockets/plat/lws-plat-win.c diff --git a/thirdparty/libwebsockets/plat/lws-plat-win.c b/thirdparty/libwebsockets/plat/lws-plat-win.c
index 948db6289..511e29739 100644 index bd513b494..1850b6425 100644
--- a/thirdparty/libwebsockets/plat/lws-plat-win.c --- a/thirdparty/libwebsockets/plat/lws-plat-win.c
+++ b/thirdparty/libwebsockets/plat/lws-plat-win.c +++ b/thirdparty/libwebsockets/plat/lws-plat-win.c
@@ -635,9 +635,20 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename, @@ -641,9 +641,20 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename,
HANDLE ret; HANDLE ret;
WCHAR buf[MAX_PATH]; WCHAR buf[MAX_PATH];
lws_fop_fd_t fop_fd; lws_fop_fd_t fop_fd;
- LARGE_INTEGER llFileSize = {0}; - LARGE_INTEGER llFileSize = {0};
+ FILE_STANDARD_INFO fInfo = {0}; + FILE_STANDARD_INFO fInfo = {0};
MultiByteToWideChar(CP_UTF8, 0, filename, -1, buf, ARRAY_SIZE(buf)); MultiByteToWideChar(CP_UTF8, 0, filename, -1, buf, LWS_ARRAY_SIZE(buf));
+ +
+#if defined(_WIN32_WINNT) && _WIN32_WINNT >= 0x0602 // Windows 8 (minimum when UWP_ENABLED, but can be used in Windows builds) +#if defined(_WIN32_WINNT) && _WIN32_WINNT >= 0x0602 // Windows 8 (minimum when UWP_ENABLED, but can be used in Windows builds)
+ CREATEFILE2_EXTENDED_PARAMETERS extParams = {0}; + CREATEFILE2_EXTENDED_PARAMETERS extParams = {0};
@ -24,7 +24,7 @@ index 948db6289..511e29739 100644
if (((*flags) & 7) == _O_RDONLY) { if (((*flags) & 7) == _O_RDONLY) {
ret = CreateFileW(buf, GENERIC_READ, FILE_SHARE_READ, ret = CreateFileW(buf, GENERIC_READ, FILE_SHARE_READ,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
@@ -645,6 +656,7 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename, @@ -651,6 +662,7 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename,
ret = CreateFileW(buf, GENERIC_WRITE, 0, NULL, ret = CreateFileW(buf, GENERIC_WRITE, 0, NULL,
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
} }
@ -32,7 +32,7 @@ index 948db6289..511e29739 100644
if (ret == LWS_INVALID_FILE) if (ret == LWS_INVALID_FILE)
goto bail; goto bail;
@@ -657,9 +669,9 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename, @@ -663,9 +675,9 @@ _lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename,
fop_fd->fd = ret; fop_fd->fd = ret;
fop_fd->filesystem_priv = NULL; /* we don't use it */ fop_fd->filesystem_priv = NULL; /* we don't use it */
fop_fd->flags = *flags; fop_fd->flags = *flags;

View file

@ -1,153 +1,153 @@
/* $NetBSD: getopt.c,v 1.16 1999/12/02 13:15:56 kleink Exp $ */ /* $NetBSD: getopt.c,v 1.16 1999/12/02 13:15:56 kleink Exp $ */
/* /*
* Copyright (c) 1987, 1993, 1994 * Copyright (c) 1987, 1993, 1994
* The Regents of the University of California. All rights reserved. * The Regents of the University of California. All rights reserved.
* *
* Redistribution and use in source and binary forms, with or without * Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions * modification, are permitted provided that the following conditions
* are met: * are met:
* 1. Redistributions of source code must retain the above copyright * 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer. * notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright * 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the * notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution. * documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software * 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement: * must display the following acknowledgement:
* This product includes software developed by the University of * This product includes software developed by the University of
* California, Berkeley and its contributors. * California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors * 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software * may be used to endorse or promote products derived from this software
* without specific prior written permission. * without specific prior written permission.
* *
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
*/ */
#if 0 #if 0
static char sccsid[] = "@(#)getopt.c 8.3 (Berkeley) 4/27/95"; static char sccsid[] = "@(#)getopt.c 8.3 (Berkeley) 4/27/95";
#endif #endif
#include <assert.h> #include <assert.h>
#include <errno.h> #include <errno.h>
#include <stdio.h> #include <stdio.h>
#include <string.h> #include <string.h>
#define __P(x) x #define __P(x) x
#define _DIAGASSERT(x) assert(x) #define _DIAGASSERT(x) assert(x)
#ifdef __weak_alias #ifdef __weak_alias
__weak_alias(getopt,_getopt); __weak_alias(getopt,_getopt);
#endif #endif
int opterr = 1, /* if error message should be printed */ int opterr = 1, /* if error message should be printed */
optind = 1, /* index into parent argv vector */ optind = 1, /* index into parent argv vector */
optopt, /* character checked for validity */ optopt, /* character checked for validity */
optreset; /* reset getopt */ optreset; /* reset getopt */
char *optarg; /* argument associated with option */ char *optarg; /* argument associated with option */
static char * _progname __P((char *)); static char * _progname __P((char *));
int getopt_internal __P((int, char * const *, const char *)); int getopt_internal __P((int, char * const *, const char *));
static char * static char *
_progname(nargv0) _progname(nargv0)
char * nargv0; char * nargv0;
{ {
char * tmp; char * tmp;
_DIAGASSERT(nargv0 != NULL); _DIAGASSERT(nargv0 != NULL);
tmp = strrchr(nargv0, '/'); tmp = strrchr(nargv0, '/');
if (tmp) if (tmp)
tmp++; tmp++;
else else
tmp = nargv0; tmp = nargv0;
return(tmp); return(tmp);
} }
#define BADCH (int)'?' #define BADCH (int)'?'
#define BADARG (int)':' #define BADARG (int)':'
#define EMSG "" #define EMSG ""
/* /*
* getopt -- * getopt --
* Parse argc/argv argument vector. * Parse argc/argv argument vector.
*/ */
int int
getopt(nargc, nargv, ostr) getopt(nargc, nargv, ostr)
int nargc; int nargc;
char * const nargv[]; char * const nargv[];
const char *ostr; const char *ostr;
{ {
static char *__progname = 0; static char *__progname = 0;
static char *place = EMSG; /* option letter processing */ static char *place = EMSG; /* option letter processing */
char *oli; /* option letter list index */ char *oli; /* option letter list index */
__progname = __progname?__progname:_progname(*nargv); __progname = __progname?__progname:_progname(*nargv);
_DIAGASSERT(nargv != NULL); _DIAGASSERT(nargv != NULL);
_DIAGASSERT(ostr != NULL); _DIAGASSERT(ostr != NULL);
if (optreset || !*place) { /* update scanning pointer */ if (optreset || !*place) { /* update scanning pointer */
optreset = 0; optreset = 0;
if (optind >= nargc || *(place = nargv[optind]) != '-') { if (optind >= nargc || *(place = nargv[optind]) != '-') {
place = EMSG; place = EMSG;
return (-1); return (-1);
} }
if (place[1] && *++place == '-' /* found "--" */ if (place[1] && *++place == '-' /* found "--" */
&& place[1] == '\0') { && place[1] == '\0') {
++optind; ++optind;
place = EMSG; place = EMSG;
return (-1); return (-1);
} }
} /* option letter okay? */ } /* option letter okay? */
if ((optopt = (int)*place++) == (int)':' || if ((optopt = (int)*place++) == (int)':' ||
!(oli = strchr(ostr, optopt))) { !(oli = strchr(ostr, optopt))) {
/* /*
* if the user didn't specify '-' as an option, * if the user didn't specify '-' as an option,
* assume it means -1. * assume it means -1.
*/ */
if (optopt == (int)'-') if (optopt == (int)'-')
return (-1); return (-1);
if (!*place) if (!*place)
++optind; ++optind;
if (opterr && *ostr != ':') if (opterr && *ostr != ':')
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: illegal option -- %c\n", __progname, optopt); "%s: illegal option -- %c\n", __progname, optopt);
return (BADCH); return (BADCH);
} }
if (*++oli != ':') { /* don't need argument */ if (*++oli != ':') { /* don't need argument */
optarg = NULL; optarg = NULL;
if (!*place) if (!*place)
++optind; ++optind;
} }
else { /* need an argument */ else { /* need an argument */
if (*place) /* no white space */ if (*place) /* no white space */
optarg = place; optarg = place;
else if (nargc <= ++optind) { /* no arg */ else if (nargc <= ++optind) { /* no arg */
place = EMSG; place = EMSG;
if (*ostr == ':') if (*ostr == ':')
return (BADARG); return (BADARG);
if (opterr) if (opterr)
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: option requires an argument -- %c\n", "%s: option requires an argument -- %c\n",
__progname, optopt); __progname, optopt);
return (BADCH); return (BADCH);
} }
else /* white space */ else /* white space */
optarg = nargv[optind]; optarg = nargv[optind];
place = EMSG; place = EMSG;
++optind; ++optind;
} }
return (optopt); /* dump back option letter */ return (optopt); /* dump back option letter */
} }

View file

@ -1,240 +1,240 @@
/* /*
* Copyright (c) 1987, 1993, 1994, 1996 * Copyright (c) 1987, 1993, 1994, 1996
* The Regents of the University of California. All rights reserved. * The Regents of the University of California. All rights reserved.
* *
* Redistribution and use in source and binary forms, with or without * Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions * modification, are permitted provided that the following conditions
* are met: * are met:
* 1. Redistributions of source code must retain the above copyright * 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer. * notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright * 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the * notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution. * documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software * 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement: * must display the following acknowledgement:
* This product includes software developed by the University of * This product includes software developed by the University of
* California, Berkeley and its contributors. * California, Berkeley and its contributors.
* 4. Neither the name of the University nor the names of its contributors * 4. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software * may be used to endorse or promote products derived from this software
* without specific prior written permission. * without specific prior written permission.
* *
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
*/ */
#include <assert.h> #include <assert.h>
#include <errno.h> #include <errno.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include "getopt.h" #include "getopt.h"
#define lws_ptr_diff(head, tail) \ #define lws_ptr_diff(head, tail) \
((int)((char *)(head) - (char *)(tail))) ((int)((char *)(head) - (char *)(tail)))
extern int opterr; /* if error message should be printed */ extern int opterr; /* if error message should be printed */
extern int optind; /* index into parent argv vector */ extern int optind; /* index into parent argv vector */
extern int optopt; /* character checked for validity */ extern int optopt; /* character checked for validity */
extern int optreset; /* reset getopt */ extern int optreset; /* reset getopt */
extern char *optarg; /* argument associated with option */ extern char *optarg; /* argument associated with option */
#define __P(x) x #define __P(x) x
#define _DIAGASSERT(x) assert(x) #define _DIAGASSERT(x) assert(x)
static char * __progname __P((char *)); static char * __progname __P((char *));
int getopt_internal __P((int, char * const *, const char *)); int getopt_internal __P((int, char * const *, const char *));
static char * static char *
__progname(nargv0) __progname(nargv0)
char * nargv0; char * nargv0;
{ {
char * tmp; char * tmp;
_DIAGASSERT(nargv0 != NULL); _DIAGASSERT(nargv0 != NULL);
tmp = strrchr(nargv0, '/'); tmp = strrchr(nargv0, '/');
if (tmp) if (tmp)
tmp++; tmp++;
else else
tmp = nargv0; tmp = nargv0;
return(tmp); return(tmp);
} }
#define BADCH (int)'?' #define BADCH (int)'?'
#define BADARG (int)':' #define BADARG (int)':'
#define EMSG "" #define EMSG ""
/* /*
* getopt -- * getopt --
* Parse argc/argv argument vector. * Parse argc/argv argument vector.
*/ */
int int
getopt_internal(nargc, nargv, ostr) getopt_internal(nargc, nargv, ostr)
int nargc; int nargc;
char * const *nargv; char * const *nargv;
const char *ostr; const char *ostr;
{ {
static char *place = EMSG; /* option letter processing */ static char *place = EMSG; /* option letter processing */
char *oli; /* option letter list index */ char *oli; /* option letter list index */
_DIAGASSERT(nargv != NULL); _DIAGASSERT(nargv != NULL);
_DIAGASSERT(ostr != NULL); _DIAGASSERT(ostr != NULL);
if (optreset || !*place) { /* update scanning pointer */ if (optreset || !*place) { /* update scanning pointer */
optreset = 0; optreset = 0;
if (optind >= nargc || *(place = nargv[optind]) != '-') { if (optind >= nargc || *(place = nargv[optind]) != '-') {
place = EMSG; place = EMSG;
return (-1); return (-1);
} }
if (place[1] && *++place == '-') { /* found "--" */ if (place[1] && *++place == '-') { /* found "--" */
/* ++optind; */ /* ++optind; */
place = EMSG; place = EMSG;
return (-2); return (-2);
} }
} /* option letter okay? */ } /* option letter okay? */
if ((optopt = (int)*place++) == (int)':' || if ((optopt = (int)*place++) == (int)':' ||
!(oli = strchr(ostr, optopt))) { !(oli = strchr(ostr, optopt))) {
/* /*
* if the user didn't specify '-' as an option, * if the user didn't specify '-' as an option,
* assume it means -1. * assume it means -1.
*/ */
if (optopt == (int)'-') if (optopt == (int)'-')
return (-1); return (-1);
if (!*place) if (!*place)
++optind; ++optind;
if (opterr && *ostr != ':') if (opterr && *ostr != ':')
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: illegal option -- %c\n", __progname(nargv[0]), optopt); "%s: illegal option -- %c\n", __progname(nargv[0]), optopt);
return (BADCH); return (BADCH);
} }
if (*++oli != ':') { /* don't need argument */ if (*++oli != ':') { /* don't need argument */
optarg = NULL; optarg = NULL;
if (!*place) if (!*place)
++optind; ++optind;
} else { /* need an argument */ } else { /* need an argument */
if (*place) /* no white space */ if (*place) /* no white space */
optarg = place; optarg = place;
else if (nargc <= ++optind) { /* no arg */ else if (nargc <= ++optind) { /* no arg */
place = EMSG; place = EMSG;
if ((opterr) && (*ostr != ':')) if ((opterr) && (*ostr != ':'))
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: option requires an argument -- %c\n", "%s: option requires an argument -- %c\n",
__progname(nargv[0]), optopt); __progname(nargv[0]), optopt);
return (BADARG); return (BADARG);
} else /* white space */ } else /* white space */
optarg = nargv[optind]; optarg = nargv[optind];
place = EMSG; place = EMSG;
++optind; ++optind;
} }
return (optopt); /* dump back option letter */ return (optopt); /* dump back option letter */
} }
#if 0 #if 0
/* /*
* getopt -- * getopt --
* Parse argc/argv argument vector. * Parse argc/argv argument vector.
*/ */
int int
getopt2(nargc, nargv, ostr) getopt2(nargc, nargv, ostr)
int nargc; int nargc;
char * const *nargv; char * const *nargv;
const char *ostr; const char *ostr;
{ {
int retval; int retval;
if ((retval = getopt_internal(nargc, nargv, ostr)) == -2) { if ((retval = getopt_internal(nargc, nargv, ostr)) == -2) {
retval = -1; retval = -1;
++optind; ++optind;
} }
return(retval); return(retval);
} }
#endif #endif
/* /*
* getopt_long -- * getopt_long --
* Parse argc/argv argument vector. * Parse argc/argv argument vector.
*/ */
int int
getopt_long(nargc, nargv, options, long_options, index) getopt_long(nargc, nargv, options, long_options, index)
int nargc; int nargc;
char ** nargv; char ** nargv;
char * options; char * options;
struct option * long_options; struct option * long_options;
int * index; int * index;
{ {
int retval; int retval;
_DIAGASSERT(nargv != NULL); _DIAGASSERT(nargv != NULL);
_DIAGASSERT(options != NULL); _DIAGASSERT(options != NULL);
_DIAGASSERT(long_options != NULL); _DIAGASSERT(long_options != NULL);
/* index may be NULL */ /* index may be NULL */
if ((retval = getopt_internal(nargc, nargv, options)) == -2) { if ((retval = getopt_internal(nargc, nargv, options)) == -2) {
char *current_argv = nargv[optind++] + 2, *has_equal; char *current_argv = nargv[optind++] + 2, *has_equal;
int i, current_argv_len, match = -1; int i, current_argv_len, match = -1;
if (*current_argv == '\0') { if (*current_argv == '\0') {
return(-1); return(-1);
} }
if ((has_equal = strchr(current_argv, '=')) != NULL) { if ((has_equal = strchr(current_argv, '=')) != NULL) {
current_argv_len = lws_ptr_diff(has_equal, current_argv); current_argv_len = lws_ptr_diff(has_equal, current_argv);
has_equal++; has_equal++;
} else } else
current_argv_len = (int)strlen(current_argv); current_argv_len = (int)strlen(current_argv);
for (i = 0; long_options[i].name; i++) { for (i = 0; long_options[i].name; i++) {
if (strncmp(current_argv, long_options[i].name, current_argv_len)) if (strncmp(current_argv, long_options[i].name, current_argv_len))
continue; continue;
if (strlen(long_options[i].name) == (unsigned)current_argv_len) { if (strlen(long_options[i].name) == (unsigned)current_argv_len) {
match = i; match = i;
break; break;
} }
if (match == -1) if (match == -1)
match = i; match = i;
} }
if (match != -1) { if (match != -1) {
if (long_options[match].has_arg == required_argument || if (long_options[match].has_arg == required_argument ||
long_options[match].has_arg == optional_argument) { long_options[match].has_arg == optional_argument) {
if (has_equal) if (has_equal)
optarg = has_equal; optarg = has_equal;
else else
optarg = nargv[optind++]; optarg = nargv[optind++];
} }
if ((long_options[match].has_arg == required_argument) if ((long_options[match].has_arg == required_argument)
&& (optarg == NULL)) { && (optarg == NULL)) {
/* /*
* Missing argument, leading : * Missing argument, leading :
* indicates no error should be generated * indicates no error should be generated
*/ */
if ((opterr) && (*options != ':')) if ((opterr) && (*options != ':'))
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: option requires an argument -- %s\n", "%s: option requires an argument -- %s\n",
__progname(nargv[0]), current_argv); __progname(nargv[0]), current_argv);
return (BADARG); return (BADARG);
} }
} else { /* No matching argument */ } else { /* No matching argument */
if ((opterr) && (*options != ':')) if ((opterr) && (*options != ':'))
(void)fprintf(stderr, (void)fprintf(stderr,
"%s: illegal option -- %s\n", __progname(nargv[0]), current_argv); "%s: illegal option -- %s\n", __progname(nargv[0]), current_argv);
return (BADCH); return (BADCH);
} }
if (long_options[match].flag) { if (long_options[match].flag) {
*long_options[match].flag = long_options[match].val; *long_options[match].flag = long_options[match].val;
retval = 0; retval = 0;
} else } else
retval = long_options[match].val; retval = long_options[match].val;
if (index) if (index)
*index = match; *index = match;
} }
return(retval); return(retval);
} }

View file

@ -1,36 +1,36 @@
#include <time.h> #include <time.h>
#include <windows.h> //I've omitted context line #include <windows.h> //I've omitted context line
#include "gettimeofday.h" #include "gettimeofday.h"
int gettimeofday(struct timeval *tv, struct timezone *tz) int gettimeofday(struct timeval *tv, struct timezone *tz)
{ {
FILETIME ft; FILETIME ft;
unsigned __int64 tmpres = 0; unsigned __int64 tmpres = 0;
static int tzflag; static int tzflag;
if (NULL != tv) { if (NULL != tv) {
GetSystemTimeAsFileTime(&ft); GetSystemTimeAsFileTime(&ft);
tmpres |= ft.dwHighDateTime; tmpres |= ft.dwHighDateTime;
tmpres <<= 32; tmpres <<= 32;
tmpres |= ft.dwLowDateTime; tmpres |= ft.dwLowDateTime;
/*converting file time to unix epoch*/ /*converting file time to unix epoch*/
tmpres /= 10; /*convert into microseconds*/ tmpres /= 10; /*convert into microseconds*/
tmpres -= DELTA_EPOCH_IN_MICROSECS; tmpres -= DELTA_EPOCH_IN_MICROSECS;
tv->tv_sec = (long)(tmpres / 1000000UL); tv->tv_sec = (long)(tmpres / 1000000UL);
tv->tv_usec = (long)(tmpres % 1000000UL); tv->tv_usec = (long)(tmpres % 1000000UL);
} }
if (NULL != tz) { if (NULL != tz) {
if (!tzflag) { if (!tzflag) {
_tzset(); _tzset();
tzflag++; tzflag++;
} }
tz->tz_minuteswest = _timezone / 60; tz->tz_minuteswest = _timezone / 60;
tz->tz_dsttime = _daylight; tz->tz_dsttime = _daylight;
} }
return 0; return 0;
} }