1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker *
4*1c60b9acSAndroid Build Coastguard Worker * Copyright (C) 2010 - 2019 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker *
6*1c60b9acSAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker *
13*1c60b9acSAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker *
16*1c60b9acSAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker */
24*1c60b9acSAndroid Build Coastguard Worker
25*1c60b9acSAndroid Build Coastguard Worker #include "libwebsockets.h"
26*1c60b9acSAndroid Build Coastguard Worker #include "lws-ssh.h"
27*1c60b9acSAndroid Build Coastguard Worker
28*1c60b9acSAndroid Build Coastguard Worker #include <string.h>
29*1c60b9acSAndroid Build Coastguard Worker
30*1c60b9acSAndroid Build Coastguard Worker struct per_vhost_data__telnet {
31*1c60b9acSAndroid Build Coastguard Worker struct lws_context *context;
32*1c60b9acSAndroid Build Coastguard Worker struct lws_vhost *vhost;
33*1c60b9acSAndroid Build Coastguard Worker const struct lws_protocols *protocol;
34*1c60b9acSAndroid Build Coastguard Worker struct per_session_data__telnet *live_pss_list;
35*1c60b9acSAndroid Build Coastguard Worker const struct lws_ssh_ops *ops;
36*1c60b9acSAndroid Build Coastguard Worker };
37*1c60b9acSAndroid Build Coastguard Worker
38*1c60b9acSAndroid Build Coastguard Worker struct per_session_data__telnet {
39*1c60b9acSAndroid Build Coastguard Worker struct per_session_data__telnet *next;
40*1c60b9acSAndroid Build Coastguard Worker struct per_vhost_data__telnet *vhd;
41*1c60b9acSAndroid Build Coastguard Worker uint32_t rx_tail;
42*1c60b9acSAndroid Build Coastguard Worker void *priv;
43*1c60b9acSAndroid Build Coastguard Worker
44*1c60b9acSAndroid Build Coastguard Worker uint32_t initial:1;
45*1c60b9acSAndroid Build Coastguard Worker
46*1c60b9acSAndroid Build Coastguard Worker char state;
47*1c60b9acSAndroid Build Coastguard Worker uint8_t cmd;
48*1c60b9acSAndroid Build Coastguard Worker };
49*1c60b9acSAndroid Build Coastguard Worker
50*1c60b9acSAndroid Build Coastguard Worker enum {
51*1c60b9acSAndroid Build Coastguard Worker LTS_BINARY_XMIT,
52*1c60b9acSAndroid Build Coastguard Worker LTS_ECHO,
53*1c60b9acSAndroid Build Coastguard Worker LTS_SUPPRESS_GA,
54*1c60b9acSAndroid Build Coastguard Worker
55*1c60b9acSAndroid Build Coastguard Worker
56*1c60b9acSAndroid Build Coastguard Worker LTSC_SUBOPT_END = 240,
57*1c60b9acSAndroid Build Coastguard Worker LTSC_BREAK = 243,
58*1c60b9acSAndroid Build Coastguard Worker LTSC_SUBOPT_START = 250,
59*1c60b9acSAndroid Build Coastguard Worker LTSC_WILL = 251,
60*1c60b9acSAndroid Build Coastguard Worker LTSC_WONT,
61*1c60b9acSAndroid Build Coastguard Worker LTSC_DO,
62*1c60b9acSAndroid Build Coastguard Worker LTSC_DONT,
63*1c60b9acSAndroid Build Coastguard Worker LTSC_IAC,
64*1c60b9acSAndroid Build Coastguard Worker
65*1c60b9acSAndroid Build Coastguard Worker LTST_WAIT_IAC = 0,
66*1c60b9acSAndroid Build Coastguard Worker LTST_GOT_IAC,
67*1c60b9acSAndroid Build Coastguard Worker LTST_WAIT_OPT,
68*1c60b9acSAndroid Build Coastguard Worker };
69*1c60b9acSAndroid Build Coastguard Worker
70*1c60b9acSAndroid Build Coastguard Worker static int
telnet_ld(struct per_session_data__telnet * pss,uint8_t c)71*1c60b9acSAndroid Build Coastguard Worker telnet_ld(struct per_session_data__telnet *pss, uint8_t c)
72*1c60b9acSAndroid Build Coastguard Worker {
73*1c60b9acSAndroid Build Coastguard Worker switch (pss->state) {
74*1c60b9acSAndroid Build Coastguard Worker case LTST_WAIT_IAC:
75*1c60b9acSAndroid Build Coastguard Worker if (c == LTSC_IAC) {
76*1c60b9acSAndroid Build Coastguard Worker pss->state = LTST_GOT_IAC;
77*1c60b9acSAndroid Build Coastguard Worker return 0;
78*1c60b9acSAndroid Build Coastguard Worker }
79*1c60b9acSAndroid Build Coastguard Worker return 1;
80*1c60b9acSAndroid Build Coastguard Worker
81*1c60b9acSAndroid Build Coastguard Worker case LTST_GOT_IAC:
82*1c60b9acSAndroid Build Coastguard Worker pss->state = LTST_WAIT_IAC;
83*1c60b9acSAndroid Build Coastguard Worker
84*1c60b9acSAndroid Build Coastguard Worker switch (c) {
85*1c60b9acSAndroid Build Coastguard Worker case LTSC_BREAK:
86*1c60b9acSAndroid Build Coastguard Worker return 0;
87*1c60b9acSAndroid Build Coastguard Worker case LTSC_WILL:
88*1c60b9acSAndroid Build Coastguard Worker case LTSC_WONT:
89*1c60b9acSAndroid Build Coastguard Worker case LTSC_DO:
90*1c60b9acSAndroid Build Coastguard Worker case LTSC_DONT:
91*1c60b9acSAndroid Build Coastguard Worker pss->cmd = c;
92*1c60b9acSAndroid Build Coastguard Worker pss->state = LTST_WAIT_OPT;
93*1c60b9acSAndroid Build Coastguard Worker return 0;
94*1c60b9acSAndroid Build Coastguard Worker case LTSC_IAC:
95*1c60b9acSAndroid Build Coastguard Worker return 1; /* double IAC */
96*1c60b9acSAndroid Build Coastguard Worker }
97*1c60b9acSAndroid Build Coastguard Worker return 0; /* ignore unknown */
98*1c60b9acSAndroid Build Coastguard Worker
99*1c60b9acSAndroid Build Coastguard Worker case LTST_WAIT_OPT:
100*1c60b9acSAndroid Build Coastguard Worker lwsl_notice(" tld: cmd %d: opt %d\n", pss->cmd, c);
101*1c60b9acSAndroid Build Coastguard Worker pss->state = LTST_WAIT_IAC;
102*1c60b9acSAndroid Build Coastguard Worker return 0;
103*1c60b9acSAndroid Build Coastguard Worker }
104*1c60b9acSAndroid Build Coastguard Worker
105*1c60b9acSAndroid Build Coastguard Worker return 0;
106*1c60b9acSAndroid Build Coastguard Worker }
107*1c60b9acSAndroid Build Coastguard Worker
108*1c60b9acSAndroid Build Coastguard Worker static uint8_t init[] = {
109*1c60b9acSAndroid Build Coastguard Worker LTSC_IAC, LTSC_WILL, 3,
110*1c60b9acSAndroid Build Coastguard Worker LTSC_IAC, LTSC_WILL, 1,
111*1c60b9acSAndroid Build Coastguard Worker LTSC_IAC, LTSC_DONT, 1,
112*1c60b9acSAndroid Build Coastguard Worker LTSC_IAC, LTSC_DO, 0
113*1c60b9acSAndroid Build Coastguard Worker };
114*1c60b9acSAndroid Build Coastguard Worker
115*1c60b9acSAndroid Build Coastguard Worker static int
lws_callback_raw_telnet(struct lws * wsi,enum lws_callback_reasons reason,void * user,void * in,size_t len)116*1c60b9acSAndroid Build Coastguard Worker lws_callback_raw_telnet(struct lws *wsi, enum lws_callback_reasons reason,
117*1c60b9acSAndroid Build Coastguard Worker void *user, void *in, size_t len)
118*1c60b9acSAndroid Build Coastguard Worker {
119*1c60b9acSAndroid Build Coastguard Worker struct per_session_data__telnet *pss =
120*1c60b9acSAndroid Build Coastguard Worker (struct per_session_data__telnet *)user, **p;
121*1c60b9acSAndroid Build Coastguard Worker struct per_vhost_data__telnet *vhd =
122*1c60b9acSAndroid Build Coastguard Worker (struct per_vhost_data__telnet *)
123*1c60b9acSAndroid Build Coastguard Worker lws_protocol_vh_priv_get(lws_get_vhost(wsi),
124*1c60b9acSAndroid Build Coastguard Worker lws_get_protocol(wsi));
125*1c60b9acSAndroid Build Coastguard Worker const struct lws_protocol_vhost_options *pvo =
126*1c60b9acSAndroid Build Coastguard Worker (const struct lws_protocol_vhost_options *)in;
127*1c60b9acSAndroid Build Coastguard Worker int n, m;
128*1c60b9acSAndroid Build Coastguard Worker uint8_t buf[LWS_PRE + 800], *pu = in;
129*1c60b9acSAndroid Build Coastguard Worker
130*1c60b9acSAndroid Build Coastguard Worker switch ((int)reason) {
131*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_PROTOCOL_INIT:
132*1c60b9acSAndroid Build Coastguard Worker vhd = lws_protocol_vh_priv_zalloc(lws_get_vhost(wsi),
133*1c60b9acSAndroid Build Coastguard Worker lws_get_protocol(wsi),
134*1c60b9acSAndroid Build Coastguard Worker sizeof(struct per_vhost_data__telnet));
135*1c60b9acSAndroid Build Coastguard Worker vhd->context = lws_get_context(wsi);
136*1c60b9acSAndroid Build Coastguard Worker vhd->protocol = lws_get_protocol(wsi);
137*1c60b9acSAndroid Build Coastguard Worker vhd->vhost = lws_get_vhost(wsi);
138*1c60b9acSAndroid Build Coastguard Worker
139*1c60b9acSAndroid Build Coastguard Worker while (pvo) {
140*1c60b9acSAndroid Build Coastguard Worker if (!strcmp(pvo->name, "ops"))
141*1c60b9acSAndroid Build Coastguard Worker vhd->ops = (const struct lws_ssh_ops *)pvo->value;
142*1c60b9acSAndroid Build Coastguard Worker
143*1c60b9acSAndroid Build Coastguard Worker pvo = pvo->next;
144*1c60b9acSAndroid Build Coastguard Worker }
145*1c60b9acSAndroid Build Coastguard Worker
146*1c60b9acSAndroid Build Coastguard Worker if (!vhd->ops) {
147*1c60b9acSAndroid Build Coastguard Worker lwsl_err("telnet pvo \"ops\" is mandatory\n");
148*1c60b9acSAndroid Build Coastguard Worker return -1;
149*1c60b9acSAndroid Build Coastguard Worker }
150*1c60b9acSAndroid Build Coastguard Worker break;
151*1c60b9acSAndroid Build Coastguard Worker
152*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_RAW_ADOPT:
153*1c60b9acSAndroid Build Coastguard Worker pss->next = vhd->live_pss_list;
154*1c60b9acSAndroid Build Coastguard Worker vhd->live_pss_list = pss;
155*1c60b9acSAndroid Build Coastguard Worker pss->vhd = vhd;
156*1c60b9acSAndroid Build Coastguard Worker pss->state = LTST_WAIT_IAC;
157*1c60b9acSAndroid Build Coastguard Worker pss->initial = 0;
158*1c60b9acSAndroid Build Coastguard Worker if (vhd->ops->channel_create)
159*1c60b9acSAndroid Build Coastguard Worker vhd->ops->channel_create(wsi, &pss->priv);
160*1c60b9acSAndroid Build Coastguard Worker lws_callback_on_writable(wsi);
161*1c60b9acSAndroid Build Coastguard Worker break;
162*1c60b9acSAndroid Build Coastguard Worker
163*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_RAW_CLOSE:
164*1c60b9acSAndroid Build Coastguard Worker p = &vhd->live_pss_list;
165*1c60b9acSAndroid Build Coastguard Worker
166*1c60b9acSAndroid Build Coastguard Worker while (*p) {
167*1c60b9acSAndroid Build Coastguard Worker if ((*p) == pss) {
168*1c60b9acSAndroid Build Coastguard Worker if (vhd->ops->channel_destroy)
169*1c60b9acSAndroid Build Coastguard Worker vhd->ops->channel_destroy(pss->priv);
170*1c60b9acSAndroid Build Coastguard Worker *p = pss->next;
171*1c60b9acSAndroid Build Coastguard Worker continue;
172*1c60b9acSAndroid Build Coastguard Worker }
173*1c60b9acSAndroid Build Coastguard Worker p = &((*p)->next);
174*1c60b9acSAndroid Build Coastguard Worker }
175*1c60b9acSAndroid Build Coastguard Worker break;
176*1c60b9acSAndroid Build Coastguard Worker
177*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_RAW_RX:
178*1c60b9acSAndroid Build Coastguard Worker n = 0;
179*1c60b9acSAndroid Build Coastguard Worker
180*1c60b9acSAndroid Build Coastguard Worker /* this stuff is coming in telnet line discipline, we
181*1c60b9acSAndroid Build Coastguard Worker * have to strip IACs and process IAC repeats */
182*1c60b9acSAndroid Build Coastguard Worker
183*1c60b9acSAndroid Build Coastguard Worker while (len--) {
184*1c60b9acSAndroid Build Coastguard Worker if (telnet_ld(pss, *pu))
185*1c60b9acSAndroid Build Coastguard Worker buf[n++] = *pu++;
186*1c60b9acSAndroid Build Coastguard Worker else
187*1c60b9acSAndroid Build Coastguard Worker pu++;
188*1c60b9acSAndroid Build Coastguard Worker
189*1c60b9acSAndroid Build Coastguard Worker if (n > 100 || !len)
190*1c60b9acSAndroid Build Coastguard Worker pss->vhd->ops->rx(pss->priv, wsi, buf, (uint32_t)n);
191*1c60b9acSAndroid Build Coastguard Worker }
192*1c60b9acSAndroid Build Coastguard Worker break;
193*1c60b9acSAndroid Build Coastguard Worker
194*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_RAW_WRITEABLE:
195*1c60b9acSAndroid Build Coastguard Worker n = 0;
196*1c60b9acSAndroid Build Coastguard Worker if (!pss->initial) {
197*1c60b9acSAndroid Build Coastguard Worker memcpy(buf + LWS_PRE, init, sizeof(init));
198*1c60b9acSAndroid Build Coastguard Worker
199*1c60b9acSAndroid Build Coastguard Worker n = sizeof(init);
200*1c60b9acSAndroid Build Coastguard Worker pss->initial = 1;
201*1c60b9acSAndroid Build Coastguard Worker } else {
202*1c60b9acSAndroid Build Coastguard Worker /* bring any waiting tx into second half of buffer
203*1c60b9acSAndroid Build Coastguard Worker * restrict how much we can send to 1/4 of the buffer,
204*1c60b9acSAndroid Build Coastguard Worker * because we have to apply telnet line discipline...
205*1c60b9acSAndroid Build Coastguard Worker * in the worst case of all 0xff, doubling the size
206*1c60b9acSAndroid Build Coastguard Worker */
207*1c60b9acSAndroid Build Coastguard Worker pu = buf + LWS_PRE + 400;
208*1c60b9acSAndroid Build Coastguard Worker m = (int)pss->vhd->ops->tx(pss->priv, LWS_STDOUT, pu,
209*1c60b9acSAndroid Build Coastguard Worker (size_t)((int)sizeof(buf) - LWS_PRE - n - 401) / 2);
210*1c60b9acSAndroid Build Coastguard Worker
211*1c60b9acSAndroid Build Coastguard Worker /*
212*1c60b9acSAndroid Build Coastguard Worker * apply telnet line discipline and copy into place
213*1c60b9acSAndroid Build Coastguard Worker * in output buffer
214*1c60b9acSAndroid Build Coastguard Worker */
215*1c60b9acSAndroid Build Coastguard Worker while (m--) {
216*1c60b9acSAndroid Build Coastguard Worker if (*pu == 0xff)
217*1c60b9acSAndroid Build Coastguard Worker buf[LWS_PRE + n++] = 0xff;
218*1c60b9acSAndroid Build Coastguard Worker buf[LWS_PRE + n++] = *pu++;
219*1c60b9acSAndroid Build Coastguard Worker }
220*1c60b9acSAndroid Build Coastguard Worker }
221*1c60b9acSAndroid Build Coastguard Worker if (n > 0) {
222*1c60b9acSAndroid Build Coastguard Worker m = lws_write(wsi, (unsigned char *)buf + LWS_PRE, (unsigned int)n,
223*1c60b9acSAndroid Build Coastguard Worker LWS_WRITE_HTTP);
224*1c60b9acSAndroid Build Coastguard Worker if (m < 0) {
225*1c60b9acSAndroid Build Coastguard Worker lwsl_err("ERROR %d writing to di socket\n", m);
226*1c60b9acSAndroid Build Coastguard Worker return -1;
227*1c60b9acSAndroid Build Coastguard Worker }
228*1c60b9acSAndroid Build Coastguard Worker }
229*1c60b9acSAndroid Build Coastguard Worker
230*1c60b9acSAndroid Build Coastguard Worker if (vhd->ops->tx_waiting(&pss->priv))
231*1c60b9acSAndroid Build Coastguard Worker lws_callback_on_writable(wsi);
232*1c60b9acSAndroid Build Coastguard Worker break;
233*1c60b9acSAndroid Build Coastguard Worker
234*1c60b9acSAndroid Build Coastguard Worker case LWS_CALLBACK_SSH_UART_SET_RXFLOW:
235*1c60b9acSAndroid Build Coastguard Worker /*
236*1c60b9acSAndroid Build Coastguard Worker * this is sent to set rxflow state on any connections that
237*1c60b9acSAndroid Build Coastguard Worker * sink on a particular uart. The uart index affected is in len
238*1c60b9acSAndroid Build Coastguard Worker *
239*1c60b9acSAndroid Build Coastguard Worker * More than one protocol may sink to the same uart, and the
240*1c60b9acSAndroid Build Coastguard Worker * protocol may select the uart itself, eg, in the URL used
241*1c60b9acSAndroid Build Coastguard Worker * to set up the connection.
242*1c60b9acSAndroid Build Coastguard Worker */
243*1c60b9acSAndroid Build Coastguard Worker lws_rx_flow_control(wsi, len & 1);
244*1c60b9acSAndroid Build Coastguard Worker break;
245*1c60b9acSAndroid Build Coastguard Worker
246*1c60b9acSAndroid Build Coastguard Worker default:
247*1c60b9acSAndroid Build Coastguard Worker break;
248*1c60b9acSAndroid Build Coastguard Worker }
249*1c60b9acSAndroid Build Coastguard Worker
250*1c60b9acSAndroid Build Coastguard Worker return 0;
251*1c60b9acSAndroid Build Coastguard Worker }
252*1c60b9acSAndroid Build Coastguard Worker
253*1c60b9acSAndroid Build Coastguard Worker const struct lws_protocols protocols_telnet[] = {
254*1c60b9acSAndroid Build Coastguard Worker {
255*1c60b9acSAndroid Build Coastguard Worker "lws-telnetd-base",
256*1c60b9acSAndroid Build Coastguard Worker lws_callback_raw_telnet,
257*1c60b9acSAndroid Build Coastguard Worker sizeof(struct per_session_data__telnet),
258*1c60b9acSAndroid Build Coastguard Worker 1024, 0, NULL, 900
259*1c60b9acSAndroid Build Coastguard Worker },
260*1c60b9acSAndroid Build Coastguard Worker { NULL, NULL, 0, 0, 0, NULL, 0 } /* terminator */
261*1c60b9acSAndroid Build Coastguard Worker };
262*1c60b9acSAndroid Build Coastguard Worker
263*1c60b9acSAndroid Build Coastguard Worker
264