ss-raw.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /*
  2. * libwebsockets - small server side websockets and web server implementation
  3. *
  4. * Copyright (C) 2019 - 2020 Andy Green <[email protected]>
  5. *
  6. * Permission is hereby granted, free of charge, to any person obtaining a copy
  7. * of this software and associated documentation files (the "Software"), to
  8. * deal in the Software without restriction, including without limitation the
  9. * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  10. * sell copies of the Software, and to permit persons to whom the Software is
  11. * furnished to do so, subject to the following conditions:
  12. *
  13. * The above copyright notice and this permission notice shall be included in
  14. * all copies or substantial portions of the Software.
  15. *
  16. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  21. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  22. * IN THE SOFTWARE.
  23. *
  24. * This is the glue that wires up raw-socket to Secure Streams.
  25. */
  26. #include <private-lib-core.h>
  27. int
  28. secstream_raw(struct lws *wsi, enum lws_callback_reasons reason, void *user,
  29. void *in, size_t len)
  30. {
  31. lws_ss_handle_t *h = (lws_ss_handle_t *)lws_get_opaque_user_data(wsi);
  32. uint8_t buf[LWS_PRE + 1520], *p = &buf[LWS_PRE],
  33. *end = &buf[sizeof(buf) - 1];
  34. lws_ss_state_return_t r;
  35. size_t buflen;
  36. int f = 0;
  37. switch (reason) {
  38. case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
  39. assert(h);
  40. assert(h->policy);
  41. lwsl_info("%s: h: %p, %s CLIENT_CONNECTION_ERROR: %s\n", __func__,
  42. h, h->policy->streamtype, in ? (char *)in : "(null)");
  43. r = lws_ss_event_helper(h, LWSSSCS_UNREACHABLE);
  44. if (r == LWSSSSRET_DESTROY_ME)
  45. return _lws_ss_handle_state_ret(r, wsi, &h);
  46. h->wsi = NULL;
  47. r = lws_ss_backoff(h);
  48. if (r != LWSSSSRET_OK)
  49. return _lws_ss_handle_state_ret(r, wsi, &h);
  50. break;
  51. case LWS_CALLBACK_RAW_CLOSE:
  52. if (!h)
  53. break;
  54. lws_sul_cancel(&h->sul_timeout);
  55. lwsl_info("%s: h: %p, %s LWS_CALLBACK_CLOSED_CLIENT_HTTP\n",
  56. __func__, h,
  57. h->policy ? h->policy->streamtype : "no policy");
  58. h->wsi = NULL;
  59. if (h->policy && !(h->policy->flags & LWSSSPOLF_OPPORTUNISTIC) &&
  60. #if defined(LWS_WITH_SERVER)
  61. !(h->info.flags & LWSSSINFLAGS_ACCEPTED) && /* not server */
  62. #endif
  63. !h->txn_ok && !wsi->a.context->being_destroyed) {
  64. r = lws_ss_backoff(h);
  65. if (r != LWSSSSRET_OK)
  66. return _lws_ss_handle_state_ret(r, wsi, &h);
  67. break;
  68. }
  69. /* wsi is going down anyway */
  70. r = lws_ss_event_helper(h, LWSSSCS_DISCONNECTED);
  71. if (r == LWSSSSRET_DESTROY_ME)
  72. return _lws_ss_handle_state_ret(r, wsi, &h);
  73. break;
  74. case LWS_CALLBACK_RAW_CONNECTED:
  75. lwsl_info("%s: RAW_CONNECTED\n", __func__);
  76. h->retry = 0;
  77. h->seqstate = SSSEQ_CONNECTED;
  78. lws_sul_cancel(&h->sul);
  79. r = lws_ss_event_helper(h, LWSSSCS_CONNECTED);
  80. if (r != LWSSSSRET_OK)
  81. return _lws_ss_handle_state_ret(r, wsi, &h);
  82. lws_validity_confirmed(wsi);
  83. break;
  84. case LWS_CALLBACK_RAW_ADOPT:
  85. lwsl_info("%s: RAW_ADOPT\n", __func__);
  86. break;
  87. /* chunks of chunked content, with header removed */
  88. case LWS_CALLBACK_RAW_RX:
  89. if (!h || !h->info.rx)
  90. return 0;
  91. r = h->info.rx(ss_to_userobj(h), (const uint8_t *)in, len, 0);
  92. if (r != LWSSSSRET_OK)
  93. return _lws_ss_handle_state_ret(r, wsi, &h);
  94. return 0; /* don't passthru */
  95. case LWS_CALLBACK_RAW_WRITEABLE:
  96. lwsl_info("%s: RAW_WRITEABLE\n", __func__);
  97. if (!h || !h->info.tx)
  98. return 0;
  99. buflen = lws_ptr_diff(end, p);
  100. r = h->info.tx(ss_to_userobj(h), h->txord++, p, &buflen, &f);
  101. if (r == LWSSSSRET_TX_DONT_SEND)
  102. return 0;
  103. if (r < 0)
  104. return _lws_ss_handle_state_ret(r, wsi, &h);
  105. /*
  106. * flags are ignored with raw, there are no protocol payload
  107. * boundaries, just an arbitrarily-fragmented bytestream
  108. */
  109. p += buflen;
  110. if (lws_write(wsi, buf + LWS_PRE, lws_ptr_diff(p, buf + LWS_PRE),
  111. LWS_WRITE_HTTP) != (int)lws_ptr_diff(p, buf + LWS_PRE)) {
  112. lwsl_err("%s: write failed\n", __func__);
  113. return -1;
  114. }
  115. lws_set_timeout(wsi, 0, 0);
  116. break;
  117. default:
  118. break;
  119. }
  120. return 0;
  121. }
  122. static int
  123. secstream_connect_munge_raw(lws_ss_handle_t *h, char *buf, size_t len,
  124. struct lws_client_connect_info *i,
  125. union lws_ss_contemp *ct)
  126. {
  127. i->method = "RAW";
  128. return 0;
  129. }
  130. const struct lws_protocols protocol_secstream_raw = {
  131. "lws-secstream-raw",
  132. secstream_raw,
  133. 0,
  134. 0,
  135. };
  136. const struct ss_pcols ss_pcol_raw = {
  137. "raw",
  138. "",
  139. &protocol_secstream_raw,
  140. secstream_connect_munge_raw,
  141. NULL
  142. };