1 /* -*- c-basic-offset: 8 -*-
2    rdesktop: A Remote Desktop Protocol client.
3    Protocol services - ISO layer
4    Copyright (C) Matthew Chapman 1999-2005
5 
6    This program is free software; you can redistribute it and/or modify
7    it under the terms of the GNU General Public License as published by
8    the Free Software Foundation; either version 2 of the License, or
9    (at your option) any later version.
10 
11    This program is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14    GNU General Public License for more details.
15 
16    You should have received a copy of the GNU General Public License along
17    with this program; if not, write to the Free Software Foundation, Inc.,
18    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 */
20 
21 #include "rdesktop.h"
22 
23 /* Send a self-contained ISO PDU */
24 static BOOL
iso_send_msg(RDPCLIENT * This,uint8 code)25 iso_send_msg(RDPCLIENT * This, uint8 code)
26 {
27 	STREAM s;
28 
29 	s = tcp_init(This, 11);
30 
31 	if(s == NULL)
32 		return False;
33 
34 	out_uint8(s, 3);	/* version */
35 	out_uint8(s, 0);	/* reserved */
36 	out_uint16_be(s, 11);	/* length */
37 
38 	out_uint8(s, 6);	/* hdrlen */
39 	out_uint8(s, code);
40 	out_uint16(s, 0);	/* dst_ref */
41 	out_uint16(s, 0);	/* src_ref */
42 	out_uint8(s, 0);	/* class */
43 
44 	s_mark_end(s);
45 	return tcp_send(This, s);
46 }
47 
48 static BOOL
iso_send_connection_request(RDPCLIENT * This,char * cookie)49 iso_send_connection_request(RDPCLIENT * This, char *cookie)
50 {
51 	STREAM s;
52 	int cookielen = (int)strlen(cookie);
53 	int length = 11 + cookielen;
54 
55 	s = tcp_init(This, length);
56 
57 	if(s == NULL)
58 		return False;
59 
60 	out_uint8(s, 3);	/* version */
61 	out_uint8(s, 0);	/* reserved */
62 	out_uint16_be(s, length);	/* length */
63 
64 	out_uint8(s, length - 5);	/* hdrlen */
65 	out_uint8(s, ISO_PDU_CR);
66 	out_uint16(s, 0);	/* dst_ref */
67 	out_uint16(s, 0);	/* src_ref */
68 	out_uint8(s, 0);	/* class */
69 
70 	out_uint8p(s, cookie, cookielen);
71 
72 	s_mark_end(s);
73 	return tcp_send(This, s);
74 }
75 
76 /* Receive a message on the ISO layer, return code */
77 static STREAM
iso_recv_msg(RDPCLIENT * This,uint8 * code,uint8 * rdpver)78 iso_recv_msg(RDPCLIENT * This, uint8 * code, uint8 * rdpver)
79 {
80 	STREAM s;
81 	uint16 length;
82 	uint8 version;
83 
84 	s = tcp_recv(This, NULL, 4);
85 	if (s == NULL)
86 		return NULL;
87 	in_uint8(s, version);
88 	if (rdpver != NULL)
89 		*rdpver = version;
90 	if (version == 3)
91 	{
92 		in_uint8s(s, 1);	/* pad */
93 		in_uint16_be(s, length);
94 	}
95 	else
96 	{
97 		in_uint8(s, length);
98 		if (length & 0x80)
99 		{
100 			length &= ~0x80;
101 			next_be(s, length);
102 		}
103 	}
104 	s = tcp_recv(This, s, length - 4);
105 	if (s == NULL)
106 		return NULL;
107 	if (version != 3)
108 		return s;
109 	in_uint8s(s, 1);	/* hdrlen */
110 	in_uint8(s, *code);
111 	if (*code == ISO_PDU_DT)
112 	{
113 		in_uint8s(s, 1);	/* eot */
114 		return s;
115 	}
116 	in_uint8s(s, 5);	/* dst_ref, src_ref, class */
117 	return s;
118 }
119 
120 /* Initialise ISO transport data packet */
121 STREAM
iso_init(RDPCLIENT * This,int length)122 iso_init(RDPCLIENT * This, int length)
123 {
124 	STREAM s;
125 
126 	s = tcp_init(This, length + 7);
127 
128 	if(s == NULL)
129 		return NULL;
130 
131 	s_push_layer(s, iso_hdr, 7);
132 
133 	return s;
134 }
135 
136 /* Send an ISO data PDU */
137 BOOL
iso_send(RDPCLIENT * This,STREAM s)138 iso_send(RDPCLIENT * This, STREAM s)
139 {
140 	uint16 length;
141 
142 	s_pop_layer(s, iso_hdr);
143 	length = (uint16)(s->end - s->p);
144 
145 	out_uint8(s, 3);	/* version */
146 	out_uint8(s, 0);	/* reserved */
147 	out_uint16_be(s, length);
148 
149 	out_uint8(s, 2);	/* hdrlen */
150 	out_uint8(s, ISO_PDU_DT);	/* code */
151 	out_uint8(s, 0x80);	/* eot */
152 
153 	return tcp_send(This, s);
154 }
155 
156 /* Receive ISO transport data packet */
157 STREAM
iso_recv(RDPCLIENT * This,uint8 * rdpver)158 iso_recv(RDPCLIENT * This, uint8 * rdpver)
159 {
160 	STREAM s;
161 	uint8 code = 0;
162 
163 	s = iso_recv_msg(This, &code, rdpver);
164 	if (s == NULL)
165 		return NULL;
166 	if (rdpver != NULL)
167 		if (*rdpver != 3)
168 			return s;
169 	if (code != ISO_PDU_DT)
170 	{
171 		error("expected DT, got 0x%x\n", code);
172 		return NULL;
173 	}
174 	return s;
175 }
176 
177 /* Establish a connection up to the ISO layer */
178 BOOL
iso_connect(RDPCLIENT * This,char * server,char * cookie)179 iso_connect(RDPCLIENT * This, char *server, char *cookie)
180 {
181 	uint8 code = 0;
182 
183 	if (!tcp_connect(This, server))
184 		return False;
185 
186 	if (!iso_send_connection_request(This, cookie))
187 		return False;
188 
189 	if (iso_recv_msg(This, &code, NULL) == NULL)
190 		return False;
191 
192 	if (code != ISO_PDU_CC)
193 	{
194 		error("expected CC, got 0x%x\n", code);
195 		tcp_disconnect(This);
196 		return False;
197 	}
198 
199 	return True;
200 }
201 
202 /* Establish a reconnection up to the ISO layer */
203 BOOL
iso_reconnect(RDPCLIENT * This,char * server,char * cookie)204 iso_reconnect(RDPCLIENT * This, char *server, char *cookie)
205 {
206 	uint8 code = 0;
207 
208 	if (!tcp_connect(This, server))
209 		return False;
210 
211 	if (!iso_send_connection_request(This, cookie)) // BUGBUG should we really pass the cookie here?
212 		return False;
213 
214 	if (iso_recv_msg(This, &code, NULL) == NULL)
215 		return False;
216 
217 	if (code != ISO_PDU_CC)
218 	{
219 		error("expected CC, got 0x%x\n", code);
220 		tcp_disconnect(This);
221 		return False;
222 	}
223 
224 	return True;
225 }
226 
227 /* Disconnect from the ISO layer */
228 BOOL
iso_disconnect(RDPCLIENT * This)229 iso_disconnect(RDPCLIENT * This)
230 {
231 	iso_send_msg(This, ISO_PDU_DR);
232 	return tcp_disconnect(This);
233 }
234 
235 /* reset the state to support reconnecting */
236 void
iso_reset_state(RDPCLIENT * This)237 iso_reset_state(RDPCLIENT * This)
238 {
239 	tcp_reset_state(This);
240 }
241