| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776 |
- /*
- * lws-minimal-secure-streams-client
- *
- * Written in 2010-2020 by Andy Green <[email protected]>
- *
- * This file is made available under the Creative Commons CC0 1.0
- * Universal Public Domain Dedication.
- *
- *
- * This client does not perform any INET networking... instead it opens a unix
- * domain socket on a proxy that is listening for it, and that creates the
- * actual secure stream connection.
- *
- * We are able to use the usual secure streams api in the client process, with
- * payloads and connection state information proxied over the unix domain
- * socket and fulfilled in the proxy process.
- *
- * The public client helper pieces are built as part of lws
- */
- #include <private-lib-core.h>
- static void
- lws_sspc_sul_retry_cb(lws_sorted_usec_list_t *sul)
- {
- lws_sspc_handle_t *h = lws_container_of(sul, lws_sspc_handle_t, sul_retry);
- static struct lws_client_connect_info i;
- /*
- * We may have started up before the system proxy, so be prepared with
- * a sul to retry at 1Hz
- */
- memset(&i, 0, sizeof i);
- i.context = h->context;
- if (h->context->ss_proxy_port) { /* tcp */
- i.address = h->context->ss_proxy_address;
- i.port = h->context->ss_proxy_port;
- i.iface = h->context->ss_proxy_bind;
- } else {
- if (h->context->ss_proxy_bind)
- i.address = h->context->ss_proxy_bind;
- else
- i.address = "[email protected]";
- }
- i.host = i.address;
- i.origin = i.address;
- i.method = "RAW";
- i.protocol = lws_sspc_protocols[0].name;
- i.local_protocol_name = lws_sspc_protocols[0].name;
- i.path = "";
- i.pwsi = &h->cwsi;
- i.opaque_user_data = (void *)h;
- if (!lws_client_connect_via_info(&i)) {
- lws_sul_schedule(h->context, 0, &h->sul_retry,
- lws_sspc_sul_retry_cb, LWS_US_PER_SEC);
- return;
- }
- lwsl_notice("%s: sspc ss wsi %p\n", __func__, h->cwsi);
- }
- static int
- lws_sspc_serialize_metadata(lws_sspc_metadata_t *md, uint8_t *p, uint8_t *end)
- {
- int n, txc;
- if (md->name[0] == '\0') {
- lwsl_info("%s: sending tx credit update %d\n", __func__,
- md->tx_cr_adjust);
- p[0] = LWSSS_SER_TXPRE_TXCR_UPDATE;
- lws_ser_wu16be(&p[1], 4);
- lws_ser_wu32be(&p[3], md->tx_cr_adjust);
- n = 7;
- } else {
- lwsl_info("%s: sending metadata\n", __func__);
- p[0] = LWSSS_SER_TXPRE_METADATA;
- txc = strlen(md->name);
- n = txc + 1 + md->len;
- if (n > 0xffff)
- /* we can't serialize this metadata in 16b length */
- return -1;
- if (n > lws_ptr_diff(end, &p[4]))
- /* we don't have space for this metadata */
- return -1;
- lws_ser_wu16be(&p[1], n);
- p[3] = txc;
- memcpy(&p[4], md->name, txc);
- memcpy(&p[4 + txc], &md[1], md->len);
- n = 4 + txc + md->len;
- }
- lws_dll2_remove(&md->list);
- lws_free(md);
- return n;
- }
- static int
- callback_sspc_client(struct lws *wsi, enum lws_callback_reasons reason,
- void *user, void *in, size_t len)
- {
- lws_sspc_handle_t *h = (lws_sspc_handle_t *)lws_get_opaque_user_data(wsi);
- uint8_t s[32], pkt[LWS_PRE + 2048], *p = pkt + LWS_PRE,
- *end = p + sizeof(pkt) - LWS_PRE;
- void *m = (void *)((uint8_t *)&h[1]);
- const uint8_t *cp;
- lws_usec_t us;
- int flags, n;
- switch (reason) {
- case LWS_CALLBACK_PROTOCOL_INIT:
- break;
- case LWS_CALLBACK_PROTOCOL_DESTROY:
- break;
- case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
- lwsl_warn("%s: CONNECTION_ERROR\n", __func__);
- lws_set_opaque_user_data(wsi, NULL);
- h->cwsi = NULL;
- lws_sul_schedule(h->context, 0, &h->sul_retry,
- lws_sspc_sul_retry_cb, LWS_US_PER_SEC);
- break;
- case LWS_CALLBACK_RAW_CONNECTED:
- if (!h)
- return -1;
- lwsl_info("%s: CONNECTED (%s)\n", __func__, h->ssi.streamtype);
- h->state = LPCSCLI_SENDING_INITIAL_TX;
- h->dsh = lws_dsh_create(NULL, (LWS_PRE + LWS_SS_MTU) * 160, 1);
- if (!h->dsh)
- return -1;
- lws_set_timeout(wsi, PENDING_TIMEOUT_AWAITING_CLIENT_HS_SEND, 3);
- lws_callback_on_writable(wsi);
- break;
- case LWS_CALLBACK_RAW_CLOSE:
- /*
- * our ss proxy Unix Domain socket has closed...
- */
- lwsl_notice("%s: LWS_CALLBACK_RAW_CLOSE: proxy conn down\n",
- __func__);
- if (h) {
- h->cwsi = NULL;
- /*
- * schedule a reconnect in 1s
- */
- lws_sul_schedule(h->context, 0, &h->sul_retry,
- lws_sspc_sul_retry_cb, LWS_US_PER_SEC);
- }
- break;
- case LWS_CALLBACK_RAW_RX:
- /*
- * ie, the proxy has sent us something
- */
- lwsl_info("%s: RAW_RX: rx %d\n", __func__, (int)len);
- if (!h || !h->cwsi) {
- lwsl_err("%s: rx with bad conn state\n", __func__);
- return -1;
- }
- n = lws_ss_deserialize_parse(&h->parser, lws_get_context(wsi),
- h->dsh, in, len, &h->state, h,
- (lws_ss_handle_t **)m, &h->ssi, 1);
- switch (n) {
- case LWSSSSRET_OK:
- break;
- case LWSSSSRET_DISCONNECT_ME:
- return -1;
- case LWSSSSRET_DESTROY_ME:
- lws_sspc_destroy(&h);
- return -1;
- }
- if (h->state == LPCSCLI_LOCAL_CONNECTED ||
- h->state == LPCSCLI_ONWARD_CONNECT)
- lws_set_timeout(wsi, 0, 0);
- break;
- case LWS_CALLBACK_RAW_WRITEABLE:
- /*
- * We can transmit something to the proxy...
- */
- if (!h)
- break;
- lwsl_debug("%s: WRITEABLE %p: (%s) state %d\n", __func__, wsi,
- h->ssi.streamtype, h->state);
- /*
- * Management of ss timeout can happen any time and doesn't
- * depend on wsi existence or state
- */
- n = 0;
- cp = s;
- if (h->pending_timeout_update) {
- s[0] = LWSSS_SER_TXPRE_TIMEOUT_UPDATE;
- s[1] = 0;
- s[2] = 4;
- /*
- * 0: use policy timeout value
- * 0xffffffff: cancel the timeout
- */
- lws_ser_wu32be(&s[3], h->timeout_ms);
- /* in case anything else to write */
- lws_callback_on_writable(h->cwsi);
- h->pending_timeout_update = 0;
- n = 7;
- goto do_write;
- }
- s[1] = 0;
- /*
- * This is the state of the link that connects us to the onward
- * proxy
- */
- switch (h->state) {
- case LPCSCLI_SENDING_INITIAL_TX:
- /*
- * We are negotating the opening of a particular
- * streamtype
- */
- n = strlen(h->ssi.streamtype) + 4;
- s[0] = LWSSS_SER_TXPRE_STREAMTYPE;
- lws_ser_wu16be(&s[1], n);
- lws_ser_wu32be(&s[3], h->txc.peer_tx_cr_est);
- //h->txcr_out = txc;
- lws_strncpy((char *)&s[7], h->ssi.streamtype, sizeof(s) - 7);
- n += 3;
- h->state = LPCSCLI_WAITING_CREATE_RESULT;
- break;
- case LPCSCLI_LOCAL_CONNECTED:
- // lwsl_notice("%s: LPCSCLI_LOCAL_CONNECTED\n", __func__);
- /*
- * Do we need to prioritize sending any metadata
- * changes?
- */
- if (h->metadata_owner.count) {
- lws_sspc_metadata_t *md = lws_container_of(
- lws_dll2_get_tail(&h->metadata_owner),
- lws_sspc_metadata_t, list);
- cp = p;
- n = lws_sspc_serialize_metadata(md, p, end);
- if (n < 0)
- goto metadata_hangup;
- lwsl_debug("%s: (local_conn) metadata\n", __func__);
- goto req_write_and_issue;
- }
- if (h->pending_writeable_len) {
- lwsl_debug("%s: (local_conn) PAYLOAD_LENGTH_HINT %u\n",
- __func__, (unsigned int)h->writeable_len);
- s[0] = LWSSS_SER_TXPRE_PAYLOAD_LENGTH_HINT;
- lws_ser_wu16be(&s[1], 4);
- lws_ser_wu32be(&s[3], h->writeable_len);
- h->pending_writeable_len = 0;
- n = 7;
- goto req_write_and_issue;
- }
- if (h->conn_req_state >= LWSSSPC_ONW_ONGOING) {
- lwsl_info("%s: conn_req_state %d\n", __func__,
- h->conn_req_state);
- break;
- }
- lwsl_info("%s: (local_conn) onward connect\n", __func__);
- h->conn_req_state = LWSSSPC_ONW_ONGOING;
- s[0] = LWSSS_SER_TXPRE_ONWARD_CONNECT;
- s[1] = 0;
- s[2] = 0;
- n = 3;
- break;
- case LPCSCLI_OPERATIONAL:
- lwsl_notice("%s: LPCSCLI_OPERATIONAL\n", __func__);
- /*
- *
- * - Do we need to prioritize sending any metadata
- * changes? (includes txcr updates)
- *
- * - Do we need to forward a hint about the payload
- * length?
- */
- if (h->metadata_owner.count) {
- lws_sspc_metadata_t *md = lws_container_of(
- lws_dll2_get_tail(&h->metadata_owner),
- lws_sspc_metadata_t, list);
- cp = p;
- n = lws_sspc_serialize_metadata(md, p, end);
- if (n < 0)
- goto metadata_hangup;
- goto req_write_and_issue;
- }
- if (h->pending_writeable_len) {
- lwsl_info("%s: PAYLOAD_LENGTH_HINT %u\n",
- __func__, (unsigned int)h->writeable_len);
- s[0] = LWSSS_SER_TXPRE_PAYLOAD_LENGTH_HINT;
- lws_ser_wu16be(&s[1], 4);
- lws_ser_wu32be(&s[3], h->writeable_len);
- h->pending_writeable_len = 0;
- n = 7;
- goto req_write_and_issue;
- }
- /* we can't write anything if we don't have credit */
- if (!h->ignore_txc && h->txc.tx_cr <= 0) {
- lwsl_info("%s: WRITEABLE / OPERATIONAL:"
- " lack credit (%d)\n", __func__,
- h->txc.tx_cr);
- // break;
- }
- len = sizeof(pkt) - LWS_PRE - 19;
- flags = 0;
- n = h->ssi.tx(m, h->ord++, pkt + LWS_PRE + 19, &len,
- &flags);
- if (n == LWSSSSRET_TX_DONT_SEND) {
- n = 0;
- break;
- }
- h->txc.tx_cr -= len;
- cp = p;
- n = len + 19;
- us = lws_now_usecs();
- p[0] = LWSSS_SER_TXPRE_TX_PAYLOAD;
- lws_ser_wu16be(&p[1], len + 19 - 3);
- lws_ser_wu32be(&p[3], flags);
- /* time spent here waiting to send this */
- lws_ser_wu32be(&p[7], us - h->us_earliest_write_req);
- /* ust that the client write happened */
- lws_ser_wu64be(&p[11], us);
- h->us_earliest_write_req = 0;
- if (flags & LWSSS_FLAG_EOM)
- if (h->rsidx + 1 < (int)LWS_ARRAY_SIZE(h->rideshare_ofs) &&
- h->rideshare_ofs[h->rsidx + 1])
- h->rsidx++;
- break;
- default:
- break;
- }
- do_write_nz:
- if (!n)
- break;
- do_write:
- n = lws_write(wsi, (uint8_t *)cp, n, LWS_WRITE_RAW);
- if (n < 0) {
- lwsl_notice("%s: WRITEABLE: %d\n", __func__, n);
- goto hangup;
- }
- break;
- default:
- break;
- }
- return lws_callback_http_dummy(wsi, reason, user, in, len);
- metadata_hangup:
- lwsl_err("%s: metadata too large\n", __func__);
- hangup:
- lwsl_warn("hangup\n");
- /* hang up on him */
- return -1;
- req_write_and_issue:
- /* in case anything else to write */
- lws_callback_on_writable(h->cwsi);
- goto do_write_nz;
- }
- const struct lws_protocols lws_sspc_protocols[] = {
- {
- "ssproxy-protocol",
- callback_sspc_client,
- 0,
- 2048, 2048, NULL, 0
- },
- { NULL, NULL, 0, 0, 0, NULL, 0 }
- };
- int
- lws_sspc_create(struct lws_context *context, int tsi, const lws_ss_info_t *ssi,
- void *opaque_user_data, lws_sspc_handle_t **ppss,
- struct lws_sequencer *seq_owner, const char **ppayload_fmt)
- {
- lws_sspc_handle_t *h;
- uint8_t *ua;
- char *p;
- lwsl_notice("%s: streamtype %s\n", __func__, ssi->streamtype);
- /* allocate the handle (including ssi), the user alloc,
- * and the streamname */
- h = malloc(sizeof(lws_sspc_handle_t) + ssi->user_alloc +
- strlen(ssi->streamtype) + 1);
- if (!h)
- return 1;
- memset(h, 0, sizeof(*h));
- memcpy(&h->ssi, ssi, sizeof(*ssi));
- ua = (uint8_t *)&h[1];
- memset(ua, 0, ssi->user_alloc);
- p = (char *)ua + ssi->user_alloc;
- memcpy(p, ssi->streamtype, strlen(ssi->streamtype) + 1);
- h->ssi.streamtype = (const char *)p;
- h->context = context;
- if (!ssi->manual_initial_tx_credit)
- h->txc.peer_tx_cr_est = 500000000;
- else
- h->txc.peer_tx_cr_est = ssi->manual_initial_tx_credit;
- if (!strcmp(ssi->streamtype, "_lws_smd"))
- h->ignore_txc = 1;
- lws_dll2_add_head(&h->client_list, &context->pt[tsi].ss_client_owner);
- /* fill in the things the real api does for the caller */
- *((void **)(ua + ssi->opaque_user_data_offset)) = opaque_user_data;
- *((void **)(ua + ssi->handle_offset)) = h;
- if (ppss)
- *ppss = h;
- /* try the actual connect */
- lws_sspc_sul_retry_cb(&h->sul_retry);
- return 0;
- }
- /* used on context destroy when iterating listed lws_ss on a pt */
- int
- lws_sspc_destroy_dll(struct lws_dll2 *d, void *user)
- {
- lws_sspc_handle_t *h = lws_container_of(d, lws_sspc_handle_t, client_list);
- lws_sspc_destroy(&h);
- return 0;
- }
- void
- lws_sspc_destroy(lws_sspc_handle_t **ph)
- {
- lws_sspc_handle_t *h;
- void *m;
- lwsl_debug("%s\n", __func__);
- if (!*ph)
- return;
- h = *ph;
- m = (void *)((uint8_t *)&h[1]);
- if (h->destroying)
- return;
- h->destroying = 1;
- lws_sul_cancel(&h->sul_retry);
- lws_dll2_remove(&h->client_list);
- if (h->dsh)
- lws_dsh_destroy(&h->dsh);
- if (h->cwsi) {
- struct lws *wsi = h->cwsi;
- h->cwsi = NULL;
- if (wsi)
- lws_set_timeout(wsi, 1, LWS_TO_KILL_SYNC);
- }
- /* clean out any pending metadata changes that didn't make it */
- lws_start_foreach_dll_safe(struct lws_dll2 *, d, d1,
- lws_dll2_get_head(&(*ph)->metadata_owner)) {
- lws_sspc_metadata_t *md =
- lws_container_of(d, lws_sspc_metadata_t, list);
- lws_dll2_remove(&md->list);
- lws_free(md);
- } lws_end_foreach_dll_safe(d, d1);
- h->ssi.state(m, NULL, LWSSSCS_DESTROYING, 0);
- *ph = NULL;
- free(h);
- }
- lws_ss_state_return_t
- lws_sspc_request_tx(lws_sspc_handle_t *h)
- {
- if (!h || !h->cwsi)
- return LWSSSSRET_OK;
- if (!h->us_earliest_write_req)
- h->us_earliest_write_req = lws_now_usecs();
- if (h->state == LPCSCLI_LOCAL_CONNECTED &&
- h->conn_req_state == LWSSSPC_ONW_NONE)
- h->conn_req_state = LWSSSPC_ONW_REQ;
- lws_callback_on_writable(h->cwsi);
- return LWSSSSRET_OK;
- }
- /*
- * Currently we fulfil the writeable part locally by just enabling POLLOUT on
- * the UDS link, without serialization footprint, which is reasonable as far as
- * it goes.
- *
- * But for the ..._len() variant, the expected payload length hint we are being
- * told is something that must be serialized to the onward peer, since either
- * that guy or someone upstream of him is the guy who will compose the framing
- * with it that actually goes out.
- *
- * This information is needed at the upstream guy before we have sent any
- * payload, eg, for http POST, he has to prepare the content-length in the
- * headers, before any payload. So we have to issue a serialization of the
- * length at this point.
- */
- lws_ss_state_return_t
- lws_sspc_request_tx_len(lws_sspc_handle_t *h, unsigned long len)
- {
- /*
- * for client conns, they cannot even complete creation of the handle
- * without the onwared connection to the proxy, it's not legal to start
- * using it until it's operation and has the onward connection (and has
- * called CREATED state)
- */
- if (!h)
- return LWSSSSRET_OK;
- lwsl_notice("%s: setting h %p writeable_len %u\n", __func__, h,
- (unsigned int)len);
- h->writeable_len = len;
- h->pending_writeable_len = 1;
- if (!h->us_earliest_write_req)
- h->us_earliest_write_req = lws_now_usecs();
- if (h->state == LPCSCLI_LOCAL_CONNECTED &&
- h->conn_req_state == LWSSSPC_ONW_NONE)
- h->conn_req_state = LWSSSPC_ONW_REQ;
- /*
- * We're going to use this up with serializing h->writeable_len... that
- * will request again.
- */
- lws_callback_on_writable(h->cwsi);
- return LWSSSSRET_OK;
- }
- int
- lws_sspc_client_connect(lws_sspc_handle_t *h)
- {
- if (!h || h->state == LPCSCLI_OPERATIONAL)
- return 0;
- assert(h->state == LPCSCLI_LOCAL_CONNECTED);
- if (h->state == LPCSCLI_LOCAL_CONNECTED &&
- h->conn_req_state == LWSSSPC_ONW_NONE)
- h->conn_req_state = LWSSSPC_ONW_REQ;
- if (h->cwsi)
- lws_callback_on_writable(h->cwsi);
- return 0;
- }
- struct lws_context *
- lws_sspc_get_context(struct lws_sspc_handle *h)
- {
- return h->context;
- }
- const char *
- lws_sspc_rideshare(struct lws_sspc_handle *h)
- {
- /*
- * ...the serialized RX rideshare name if any...
- */
- if (h->parser.rideshare[0]) {
- lwsl_info("%s: parser %s\n", __func__, h->parser.rideshare);
- return h->parser.rideshare;
- }
- /*
- * The tx rideshare index
- */
- if (h->rideshare_list[0]) {
- lwsl_info("%s: tx list %s\n", __func__,
- &h->rideshare_list[h->rideshare_ofs[h->rsidx]]);
- return &h->rideshare_list[h->rideshare_ofs[h->rsidx]];
- }
- /*
- * ... otherwise default to our stream type name
- */
- lwsl_info("%s: def %s\n", __func__, h->ssi.streamtype);
- return h->ssi.streamtype;
- }
- static int
- _lws_sspc_set_metadata(struct lws_sspc_handle *h, const char *name,
- const void *value, size_t len, int tx_cr_adjust)
- {
- lws_sspc_metadata_t *md;
- /*
- * Are we replacing a pending metadata of the same name? It's not
- * efficient to do this but user code can do what it likes... let's
- * optimize away the old one.
- *
- * Tx credit adjust always has name ""
- */
- lws_start_foreach_dll_safe(struct lws_dll2 *, d, d1,
- lws_dll2_get_head(&h->metadata_owner)) {
- md = lws_container_of(d, lws_sspc_metadata_t, list);
- if (!strcmp(name, md->name)) {
- lws_dll2_remove(&md->list);
- lws_free(md);
- break;
- }
- } lws_end_foreach_dll_safe(d, d1);
- /*
- * We have to stash the metadata and pass it to the proxy
- */
- md = lws_malloc(sizeof(*md) + len, "set metadata");
- if (!md) {
- lwsl_err("%s: OOM\n", __func__);
- return 1;
- }
- memset(md, 0, sizeof(*md));
- md->tx_cr_adjust = tx_cr_adjust;
- h->txc.peer_tx_cr_est += tx_cr_adjust;
- lws_strncpy(md->name, name, sizeof(md->name));
- md->len = len;
- if (len)
- memcpy(&md[1], value, len);
- lws_dll2_add_tail(&md->list, &h->metadata_owner);
- if (len) {
- lwsl_info("%s: set metadata %s\n", __func__, name);
- lwsl_hexdump_info(value, len);
- } else
- lwsl_info("%s: serializing tx cr adj %d\n", __func__,
- (int)tx_cr_adjust);
- if (h->cwsi)
- lws_callback_on_writable(h->cwsi);
- return 0;
- }
- int
- lws_sspc_set_metadata(struct lws_sspc_handle *h, const char *name,
- const void *value, size_t len)
- {
- return _lws_sspc_set_metadata(h, name, value, len, 0);
- }
- int
- lws_sspc_add_peer_tx_credit(struct lws_sspc_handle *h, int32_t bump)
- {
- lwsl_notice("%s: %d\n", __func__, bump);
- return _lws_sspc_set_metadata(h, "", NULL, 0, (int)bump);
- }
- int
- lws_sspc_get_est_peer_tx_credit(struct lws_sspc_handle *h)
- {
- return h->txc.peer_tx_cr_est;
- }
- void
- lws_sspc_start_timeout(struct lws_sspc_handle *h, unsigned int timeout_ms)
- {
- h->timeout_ms = (uint32_t)timeout_ms;
- h->pending_timeout_update = 1;
- lws_callback_on_writable(h->cwsi);
- }
- void
- lws_sspc_cancel_timeout(struct lws_sspc_handle *h)
- {
- lws_sspc_start_timeout(h, (unsigned int)-1);
- }
- void *
- lws_sspc_to_user_object(struct lws_sspc_handle *h)
- {
- return (void *)&h[1];
- }
- void
- lws_sspc_change_handlers(struct lws_sspc_handle *h,
- lws_ss_state_return_t (*rx)(void *userobj, const uint8_t *buf, size_t len, int flags),
- lws_ss_state_return_t (*tx)(void *userobj, lws_ss_tx_ordinal_t ord, uint8_t *buf,
- size_t *len, int *flags),
- lws_ss_state_return_t (*state)(void *userobj, void *h_src /* ss handle type */,
- lws_ss_constate_t state, lws_ss_tx_ordinal_t ack))
- {
- if (rx)
- h->ssi.rx = rx;
- if (tx)
- h->ssi.tx = tx;
- if (state)
- h->ssi.state = state;
- }
|