mirror of
https://github.com/warmcat/libwebsockets.git
synced 2025-03-16 00:00:07 +01:00

Presently a vh is allocated per trust store at policy parsing-time, this is no problem on a linux-class device or if you decide you need a dynamic policy for functionality reasons. However if you're in a constrained enough situation that the static policy makes sense, in the case your trust stores do not have 100% duty cycle, ie, are anyway always in use, the currently-unused vhosts and their x.509 stack are sitting there taking up heap for no immediate benefit. This patch modifies behaviour in ..._STATIC_POLICY_ONLY so that vhosts and associated x.509 tls contexts are not instantiated until a secure stream using them is created; they are refcounted, and when the last logical secure stream using a vhost is destroyed, the vhost and its tls context is also destroyed. If another ss connection is created that wants to use the trust store, the vhost and x.509 context is regenerated again as needed. Currently the refcounting is by ss, it's also possible to move the refcounting to be by connection. The choice is between the delay to generate the vh being visisble at logical ss creation-time, or at connection-time. It's anyway not preferable to have ss instantiated and taking up space with no associated connection or connection attempt underway. NB you will need to reprocess any static policies after this patch so they conform to the trust_store changes.
427 lines
11 KiB
C
427 lines
11 KiB
C
/*
|
|
* libwebsockets - small server side websockets and web server implementation
|
|
*
|
|
* Copyright (C) 2019 - 2020 Andy Green <andy@warmcat.com>
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to
|
|
* deal in the Software without restriction, including without limitation the
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
* IN THE SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
* Secure Stream state
|
|
*/
|
|
|
|
typedef enum {
|
|
SSSEQ_IDLE,
|
|
SSSEQ_TRY_CONNECT,
|
|
SSSEQ_TRY_CONNECT_NAUTH,
|
|
SSSEQ_TRY_CONNECT_SAUTH,
|
|
SSSEQ_RECONNECT_WAIT,
|
|
SSSEQ_DO_RETRY,
|
|
SSSEQ_CONNECTED,
|
|
} lws_ss_seq_state_t;
|
|
|
|
|
|
/**
|
|
* lws_ss_handle_t: publicly-opaque secure stream object implementation
|
|
*/
|
|
|
|
typedef struct lws_ss_handle {
|
|
lws_ss_info_t info; /**< copy of stream creation info */
|
|
struct lws_dll2 list; /**< pt lists active ss */
|
|
struct lws_dll2 to_list; /**< pt lists ss with pending to-s */
|
|
|
|
struct lws_dll2_owner src_list; /**< sink's list of bound sources */
|
|
|
|
struct lws_context *context; /**< lws context we are created on */
|
|
const lws_ss_policy_t *policy; /**< system policy for stream */
|
|
|
|
struct lws_sequencer *seq; /**< owning sequencer if any */
|
|
struct lws *wsi; /**< the stream wsi if any */
|
|
|
|
void *nauthi; /**< the nauth plugin instance data */
|
|
void *sauthi; /**< the sauth plugin instance data */
|
|
|
|
lws_ss_metadata_t *metadata;
|
|
const lws_ss_policy_t *rideshare;
|
|
|
|
struct lws_ss_handle *h_sink; /**< sink we are bound to, or NULL */
|
|
void *sink_obj;/**< sink's private object representing us */
|
|
|
|
lws_sorted_usec_list_t sul_timeout;
|
|
lws_sorted_usec_list_t sul;
|
|
lws_ss_tx_ordinal_t txord;
|
|
|
|
/* protocol-specific connection helpers */
|
|
|
|
union {
|
|
|
|
/* ...for http-related protocols... */
|
|
|
|
struct {
|
|
|
|
/* common to all http-related protocols */
|
|
|
|
/* incoming multipart parsing */
|
|
|
|
char boundary[24]; /* --boundary from headers */
|
|
uint8_t boundary_len; /* length of --boundary */
|
|
uint8_t boundary_seq; /* current match amount */
|
|
uint8_t boundary_dashes; /* check for -- after */
|
|
uint8_t boundary_post; /* swallow post CRLF */
|
|
|
|
uint8_t som:1; /* SOM has been sent */
|
|
uint8_t any:1; /* any content has been sent */
|
|
|
|
|
|
uint8_t good_respcode:1; /* 200 type response code */
|
|
|
|
union {
|
|
struct { /* LWSSSP_H1 */
|
|
#if defined(WIN32)
|
|
uint8_t dummy;
|
|
#endif
|
|
} h1;
|
|
struct { /* LWSSSP_H2 */
|
|
#if defined(WIN32)
|
|
uint8_t dummy;
|
|
#endif
|
|
} h2;
|
|
struct { /* LWSSSP_WS */
|
|
#if defined(WIN32)
|
|
uint8_t dummy;
|
|
#endif
|
|
} ws;
|
|
} u;
|
|
} http;
|
|
|
|
/* details for non-http related protocols... */
|
|
#if defined(LWS_ROLE_MQTT)
|
|
struct {
|
|
lws_mqtt_topic_elem_t topic_qos;
|
|
lws_mqtt_topic_elem_t sub_top;
|
|
lws_mqtt_subscribe_param_t sub_info;
|
|
/* allocation that must be destroyed with conn */
|
|
void *heap_baggage;
|
|
const char *subscribe_to;
|
|
size_t subscribe_to_len;
|
|
} mqtt;
|
|
#endif
|
|
#if defined(LWS_WITH_SYS_SMD)
|
|
struct {
|
|
struct lws_smd_peer *smd_peer;
|
|
lws_sorted_usec_list_t sul_write;
|
|
} smd;
|
|
#endif
|
|
} u;
|
|
|
|
unsigned long writeable_len;
|
|
|
|
lws_ss_constate_t connstate;/**< public connection state */
|
|
lws_ss_seq_state_t seqstate; /**< private connection state */
|
|
|
|
uint16_t retry; /**< retry / backoff tracking */
|
|
int16_t temp16;
|
|
|
|
uint8_t tsi; /**< service thread idx, usually 0 */
|
|
uint8_t subseq; /**< emulate SOM tracking */
|
|
uint8_t txn_ok; /**< 1 = transaction was OK */
|
|
|
|
uint8_t hanging_som:1;
|
|
uint8_t inside_msg:1;
|
|
uint8_t being_serialized:1; /* we are not the consumer */
|
|
} lws_ss_handle_t;
|
|
|
|
/* connection helper that doesn't need to hang around after connection starts */
|
|
|
|
union lws_ss_contemp {
|
|
#if defined(LWS_ROLE_MQTT)
|
|
lws_mqtt_client_connect_param_t ccp;
|
|
#else
|
|
#if defined(WIN32)
|
|
uint8_t dummy;
|
|
#endif
|
|
#endif
|
|
};
|
|
|
|
/*
|
|
* When allocating the opaque handle, we overallocate for:
|
|
*
|
|
* 1) policy->nauth_plugin->alloc (.nauthi) if any
|
|
* 2) policy->sauth_plugin->alloc (.sauthi) if any
|
|
* 3) copy of creation info stream type pointed to by info.streamtype... this
|
|
* may be arbitrarily long and since it may be coming from socket ipc and be
|
|
* temporary at creation time, we need a place for the copy to stay in scope
|
|
* 4) copy of info->streamtype contents
|
|
*/
|
|
|
|
|
|
/* the user object allocation is immediately after the ss object allocation */
|
|
#define ss_to_userobj(ss) ((void *)&(ss)[1])
|
|
|
|
/*
|
|
* serialization parser state
|
|
*/
|
|
|
|
enum {
|
|
KIND_C_TO_P,
|
|
KIND_SS_TO_P,
|
|
};
|
|
|
|
struct lws_ss_serialization_parser {
|
|
char streamtype[32];
|
|
char rideshare[32];
|
|
char metadata_name[32];
|
|
|
|
uint64_t ust_pwait;
|
|
|
|
lws_ss_metadata_t *ssmd;
|
|
|
|
int ps;
|
|
int ctr;
|
|
|
|
uint32_t usd_phandling;
|
|
uint32_t flags;
|
|
int32_t temp32;
|
|
|
|
int32_t txcr_out;
|
|
int32_t txcr_in;
|
|
uint16_t rem;
|
|
|
|
uint8_t type;
|
|
uint8_t frag1;
|
|
uint8_t slen;
|
|
uint8_t rsl_pos;
|
|
uint8_t rsl_idx;
|
|
};
|
|
|
|
/*
|
|
* Unlike locally-fulfilled SS, SSS doesn't have to hold metadata on client side
|
|
* but pass it through to the proxy. The client side doesn't know the real
|
|
* metadata names that are available in the policy (since it's hardcoded in code
|
|
* no point passing them back to the client from the policy). Because of that,
|
|
* it doesn't know how many to allocate when we create the sspc_handle either.
|
|
*
|
|
* So we use a linked-list of changed-but-not-yet-proxied metadata allocated
|
|
* on the heap and items removed as they are proxied out. Anything on the list
|
|
* is sent to the proxy before any requested tx is handled.
|
|
*
|
|
* This is also used to queue tx credit changes
|
|
*/
|
|
|
|
typedef struct lws_sspc_metadata {
|
|
lws_dll2_t list;
|
|
char name[32]; /* empty string, then actually TCXR */
|
|
size_t len;
|
|
int tx_cr_adjust;
|
|
|
|
/* the value of length .len is overallocated after this */
|
|
} lws_sspc_metadata_t;
|
|
|
|
|
|
typedef struct lws_sspc_handle {
|
|
char rideshare_list[128];
|
|
lws_ss_info_t ssi;
|
|
lws_sorted_usec_list_t sul_retry;
|
|
|
|
struct lws_ss_serialization_parser parser;
|
|
|
|
lws_dll2_owner_t metadata_owner;
|
|
|
|
struct lws_dll2 client_list;
|
|
struct lws_tx_credit txc;
|
|
|
|
struct lws *cwsi;
|
|
|
|
struct lws_dsh *dsh;
|
|
struct lws_context *context;
|
|
|
|
lws_usec_t us_earliest_write_req;
|
|
|
|
lws_ss_conn_states_t state;
|
|
|
|
uint32_t timeout_ms;
|
|
uint32_t ord;
|
|
|
|
int16_t temp16;
|
|
|
|
uint8_t rideshare_ofs[4];
|
|
uint8_t conn_req;
|
|
uint8_t rsidx;
|
|
|
|
uint8_t destroying:1;
|
|
uint8_t non_wsi:1;
|
|
uint8_t ignore_txc:1;
|
|
uint8_t pending_timeout_update:1;
|
|
} lws_sspc_handle_t;
|
|
|
|
typedef struct backoffs {
|
|
struct backoffs *next;
|
|
const char *name;
|
|
lws_retry_bo_t r;
|
|
} backoff_t;
|
|
|
|
union u {
|
|
backoff_t *b;
|
|
lws_ss_x509_t *x;
|
|
lws_ss_trust_store_t *t;
|
|
lws_ss_policy_t *p;
|
|
};
|
|
|
|
enum {
|
|
LTY_BACKOFF,
|
|
LTY_X509,
|
|
LTY_TRUSTSTORE,
|
|
LTY_POLICY,
|
|
|
|
_LTY_COUNT /* always last */
|
|
};
|
|
|
|
|
|
struct policy_cb_args {
|
|
struct lejp_ctx jctx;
|
|
struct lws_context *context;
|
|
struct lwsac *ac;
|
|
|
|
const char *socks5_proxy;
|
|
|
|
struct lws_b64state b64;
|
|
|
|
union u heads[_LTY_COUNT];
|
|
union u curr[_LTY_COUNT];
|
|
|
|
uint8_t *p;
|
|
|
|
int count;
|
|
};
|
|
|
|
#if defined(LWS_WITH_SYS_SMD)
|
|
extern const lws_ss_policy_t pol_smd;
|
|
#endif
|
|
|
|
int
|
|
lws_ss_deserialize_parse(struct lws_ss_serialization_parser *par,
|
|
struct lws_context *context,
|
|
struct lws_dsh *dsh, const uint8_t *cp, size_t len,
|
|
lws_ss_conn_states_t *state, void *parconn,
|
|
lws_ss_handle_t **pss, lws_ss_info_t *ssi, char client);
|
|
int
|
|
lws_ss_serialize_rx_payload(struct lws_dsh *dsh, const uint8_t *buf,
|
|
size_t len, int flags, const char *rsp);
|
|
int
|
|
lws_ss_deserialize_tx_payload(struct lws_dsh *dsh, struct lws *wsi,
|
|
lws_ss_tx_ordinal_t ord, uint8_t *buf,
|
|
size_t *len, int *flags);
|
|
int
|
|
lws_ss_serialize_state(struct lws_dsh *dsh, lws_ss_constate_t state,
|
|
lws_ss_tx_ordinal_t ack);
|
|
|
|
void
|
|
lws_ss_serialize_state_transition(lws_ss_conn_states_t *state, int new_state);
|
|
|
|
const lws_ss_policy_t *
|
|
lws_ss_policy_lookup(const struct lws_context *context, const char *streamtype);
|
|
|
|
/* can be used as a cb from lws_dll2_foreach_safe() to destroy ss */
|
|
int
|
|
lws_ss_destroy_dll(struct lws_dll2 *d, void *user);
|
|
|
|
int
|
|
lws_sspc_destroy_dll(struct lws_dll2 *d, void *user);
|
|
|
|
int
|
|
lws_ss_policy_set(struct lws_context *context, const char *name);
|
|
|
|
int
|
|
lws_ss_sys_fetch_policy(struct lws_context *context);
|
|
|
|
int
|
|
lws_ss_event_helper(lws_ss_handle_t *h, lws_ss_constate_t cs);
|
|
|
|
int
|
|
lws_ss_backoff(lws_ss_handle_t *h);
|
|
|
|
int
|
|
lws_ss_set_timeout_us(lws_ss_handle_t *h, lws_usec_t us);
|
|
|
|
void
|
|
ss_proxy_onward_txcr(void *userobj, int bump);
|
|
|
|
int
|
|
lws_ss_serialize_txcr(struct lws_dsh *dsh, int txcr);
|
|
|
|
int
|
|
lws_ss_sys_auth_api_amazon_com(struct lws_context *context);
|
|
|
|
lws_ss_metadata_t *
|
|
lws_ss_get_handle_metadata(struct lws_ss_handle *h, const char *name);
|
|
lws_ss_metadata_t *
|
|
lws_ss_policy_metadata_index(const lws_ss_policy_t *p, size_t index);
|
|
|
|
lws_ss_metadata_t *
|
|
lws_ss_policy_metadata(const lws_ss_policy_t *p, const char *name);
|
|
|
|
int
|
|
lws_ss_exp_cb_metadata(void *priv, const char *name, char *out, size_t *pos,
|
|
size_t olen, size_t *exp_ofs);
|
|
|
|
int
|
|
_lws_ss_client_connect(lws_ss_handle_t *h, int is_retry);
|
|
|
|
struct lws_vhost *
|
|
lws_ss_policy_ref_trust_store(struct lws_context *context,
|
|
const lws_ss_policy_t *pol, char doref);
|
|
|
|
#if defined(LWS_WITH_SECURE_STREAMS_STATIC_POLICY_ONLY)
|
|
int
|
|
lws_ss_policy_unref_trust_store(struct lws_context *context,
|
|
const lws_ss_policy_t *pol);
|
|
#endif
|
|
|
|
int
|
|
lws_ss_sys_cpd(struct lws_context *cx);
|
|
|
|
typedef int (* const secstream_protocol_connect_munge_t)(lws_ss_handle_t *h,
|
|
char *buf, size_t len, struct lws_client_connect_info *i,
|
|
union lws_ss_contemp *ct);
|
|
|
|
typedef int (* const secstream_protocol_add_txcr_t)(lws_ss_handle_t *h, int add);
|
|
|
|
typedef int (* const secstream_protocol_get_txcr_t)(lws_ss_handle_t *h);
|
|
|
|
struct ss_pcols {
|
|
const char *name;
|
|
const char *alpn;
|
|
const char *protocol_name;
|
|
secstream_protocol_connect_munge_t munge;
|
|
secstream_protocol_add_txcr_t tx_cr_add;
|
|
secstream_protocol_get_txcr_t tx_cr_est;
|
|
};
|
|
|
|
extern const struct ss_pcols ss_pcol_h1;
|
|
extern const struct ss_pcols ss_pcol_h2;
|
|
extern const struct ss_pcols ss_pcol_ws;
|
|
extern const struct ss_pcols ss_pcol_mqtt;
|
|
extern const struct ss_pcols ss_pcol_raw;
|
|
|
|
extern const struct lws_protocols protocol_secstream_h1;
|
|
extern const struct lws_protocols protocol_secstream_h2;
|
|
extern const struct lws_protocols protocol_secstream_ws;
|
|
extern const struct lws_protocols protocol_secstream_mqtt;
|
|
extern const struct lws_protocols protocol_secstream_raw;
|
|
|