xref: /aosp_15_r20/external/libwebsockets/plugins/ssh-base/telnet.c (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
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