FreeRDP
Loading...
Searching...
No Matches
peer.c
1
23#include <freerdp/config.h>
24
25#include "settings.h"
26
27#include <winpr/assert.h>
28#include <winpr/cast.h>
29#include <winpr/crt.h>
30#include <winpr/winsock.h>
31
32#include "info.h"
33#include "display.h"
34
35#include <freerdp/log.h>
36#include <freerdp/streamdump.h>
37#include <freerdp/redirection.h>
38#include <freerdp/crypto/certificate.h>
39
40#include "rdp.h"
41#include "peer.h"
42#include "multitransport.h"
43
44#define TAG FREERDP_TAG("core.peer")
45
46static state_run_t peer_recv_pdu(freerdp_peer* client, wStream* s);
47
48static HANDLE freerdp_peer_virtual_channel_open(freerdp_peer* client, const char* name,
49 UINT32 flags)
50{
51 WINPR_ASSERT(client);
52 WINPR_ASSERT(client->context);
53 WINPR_ASSERT(client->context->rdp);
54 WINPR_ASSERT(name);
55 rdpMcs* mcs = client->context->rdp->mcs;
56 WINPR_ASSERT(mcs);
57
58 if (flags & WTS_CHANNEL_OPTION_DYNAMIC)
59 return nullptr; /* not yet supported */
60
61 const size_t length = strnlen(name, 9);
62
63 if (length > 8)
64 return nullptr; /* SVC maximum name length is 8 */
65
66 BOOL joined = FALSE;
67 UINT32 index = 0;
68 rdpMcsChannel* mcsChannel = nullptr;
69 for (; index < mcs->channelCount; index++)
70 {
71 mcsChannel = &(mcs->channels[index]);
72
73 if (!mcsChannel->joined)
74 continue;
75
76 if (_strnicmp(name, mcsChannel->Name, length) == 0)
77 {
78 joined = TRUE;
79 break;
80 }
81 }
82
83 if (!joined)
84 return nullptr; /* channel is not joined */
85
86 rdpPeerChannel* peerChannel = (rdpPeerChannel*)mcsChannel->handle;
87
88 if (peerChannel)
89 {
90 /* channel is already open */
91 return (HANDLE)peerChannel;
92 }
93
94 WINPR_ASSERT(index <= UINT16_MAX);
95 peerChannel =
96 server_channel_common_new(client, (UINT16)index, mcsChannel->ChannelId, 128, nullptr, name);
97
98 if (peerChannel)
99 {
100 peerChannel->channelFlags = flags;
101 peerChannel->mcsChannel = mcsChannel;
102 mcsChannel->handle = (void*)peerChannel;
103 }
104
105 return (HANDLE)peerChannel;
106}
107
108static BOOL freerdp_peer_virtual_channel_close(WINPR_ATTR_UNUSED freerdp_peer* client,
109 HANDLE hChannel)
110{
111 rdpMcsChannel* mcsChannel = nullptr;
112 rdpPeerChannel* peerChannel = nullptr;
113
114 WINPR_ASSERT(client);
115
116 if (!hChannel)
117 return FALSE;
118
119 peerChannel = (rdpPeerChannel*)hChannel;
120 mcsChannel = peerChannel->mcsChannel;
121 WINPR_ASSERT(mcsChannel);
122 mcsChannel->handle = nullptr;
123 server_channel_common_free(peerChannel);
124 return TRUE;
125}
126
127static int freerdp_peer_virtual_channel_write(freerdp_peer* client, HANDLE hChannel,
128 const BYTE* buffer, UINT32 length)
129{
130 wStream* s = nullptr;
131 UINT32 flags = 0;
132 UINT32 chunkSize = 0;
133 UINT32 maxChunkSize = 0;
134 UINT32 totalLength = 0;
135 rdpPeerChannel* peerChannel = nullptr;
136 rdpMcsChannel* mcsChannel = nullptr;
137 rdpRdp* rdp = nullptr;
138
139 WINPR_ASSERT(client);
140 WINPR_ASSERT(client->context);
141
142 rdp = client->context->rdp;
143 WINPR_ASSERT(rdp);
144 WINPR_ASSERT(rdp->settings);
145
146 if (!hChannel)
147 return -1;
148
149 peerChannel = (rdpPeerChannel*)hChannel;
150 mcsChannel = peerChannel->mcsChannel;
151 WINPR_ASSERT(peerChannel);
152 WINPR_ASSERT(mcsChannel);
153 if (peerChannel->channelFlags & WTS_CHANNEL_OPTION_DYNAMIC)
154 return -1; /* not yet supported */
155
156 maxChunkSize = rdp->settings->VCChunkSize;
157 totalLength = length;
158 flags = CHANNEL_FLAG_FIRST;
159
160 while (length > 0)
161 {
162 UINT16 sec_flags = 0;
163 s = rdp_send_stream_init(rdp, &sec_flags);
164
165 if (!s)
166 return -1;
167
168 if (length > maxChunkSize)
169 {
170 chunkSize = rdp->settings->VCChunkSize;
171 }
172 else
173 {
174 chunkSize = length;
175 flags |= CHANNEL_FLAG_LAST;
176 }
177
178 if (mcsChannel->options & CHANNEL_OPTION_SHOW_PROTOCOL)
179 flags |= CHANNEL_FLAG_SHOW_PROTOCOL;
180
181 Stream_Write_UINT32(s, totalLength);
182 Stream_Write_UINT32(s, flags);
183
184 if (!Stream_EnsureRemainingCapacity(s, chunkSize))
185 {
186 Stream_Release(s);
187 return -1;
188 }
189
190 Stream_Write(s, buffer, chunkSize);
191
192 WINPR_ASSERT(peerChannel->channelId <= UINT16_MAX);
193 if (!rdp_send(rdp, s, (UINT16)peerChannel->channelId, sec_flags))
194 return -1;
195
196 buffer += chunkSize;
197 length -= chunkSize;
198 flags = 0;
199 }
200
201 return 1;
202}
203
204static void* freerdp_peer_virtual_channel_get_data(WINPR_ATTR_UNUSED freerdp_peer* client,
205 HANDLE hChannel)
206{
207 rdpPeerChannel* peerChannel = (rdpPeerChannel*)hChannel;
208
209 WINPR_ASSERT(client);
210 if (!hChannel)
211 return nullptr;
212
213 return peerChannel->extra;
214}
215
216static int freerdp_peer_virtual_channel_set_data(WINPR_ATTR_UNUSED freerdp_peer* client,
217 HANDLE hChannel, void* data)
218{
219 rdpPeerChannel* peerChannel = (rdpPeerChannel*)hChannel;
220
221 WINPR_ASSERT(client);
222 if (!hChannel)
223 return -1;
224
225 peerChannel->extra = data;
226 return 1;
227}
228
229static BOOL freerdp_peer_set_state(freerdp_peer* client, CONNECTION_STATE state)
230{
231 WINPR_ASSERT(client);
232 WINPR_ASSERT(client->context);
233 return rdp_server_transition_to_state(client->context->rdp, state);
234}
235
236static BOOL freerdp_peer_initialize(freerdp_peer* client)
237{
238 rdpRdp* rdp = nullptr;
239 rdpSettings* settings = nullptr;
240
241 WINPR_ASSERT(client);
242 WINPR_ASSERT(client->context);
243
244 rdp = client->context->rdp;
245 WINPR_ASSERT(rdp);
246
247 settings = rdp->settings;
248 WINPR_ASSERT(settings);
249
250 settings->ServerMode = TRUE;
251 settings->FrameAcknowledge = 0;
252 settings->LocalConnection = client->local;
253
254 const rdpCertificate* cert =
255 freerdp_settings_get_pointer(settings, FreeRDP_RdpServerCertificate);
256 if (!cert)
257 {
258 WLog_ERR(TAG, "Missing server certificate, can not continue.");
259 return FALSE;
260 }
261
262 if (freerdp_settings_get_bool(settings, FreeRDP_RdpSecurity))
263 {
264
265 if (!freerdp_certificate_is_rdp_security_compatible(cert))
266 {
267 if (!freerdp_settings_set_bool(settings, FreeRDP_RdpSecurity, FALSE))
268 return FALSE;
269 if (!freerdp_settings_set_bool(settings, FreeRDP_UseRdpSecurityLayer, FALSE))
270 return FALSE;
271 }
272 }
273
274 nego_set_RCG_supported(rdp->nego, settings->RemoteCredentialGuard);
275 nego_set_restricted_admin_mode_supported(rdp->nego, settings->RestrictedAdminModeSupported);
276
277 return (rdp_server_transition_to_state(rdp, CONNECTION_STATE_INITIAL));
278}
279
280#if defined(WITH_FREERDP_DEPRECATED)
281static BOOL freerdp_peer_get_fds(freerdp_peer* client, void** rfds, int* rcount)
282{
283 rdpTransport* transport = nullptr;
284 WINPR_ASSERT(client);
285 WINPR_ASSERT(client->context);
286 WINPR_ASSERT(client->context->rdp);
287
288 transport = client->context->rdp->transport;
289 WINPR_ASSERT(transport);
290 transport_get_fds(transport, rfds, rcount);
291 return TRUE;
292}
293#endif
294
295static HANDLE freerdp_peer_get_event_handle(freerdp_peer* client)
296{
297 HANDLE hEvent = nullptr;
298 rdpTransport* transport = nullptr;
299 WINPR_ASSERT(client);
300 WINPR_ASSERT(client->context);
301 WINPR_ASSERT(client->context->rdp);
302
303 transport = client->context->rdp->transport;
304 hEvent = transport_get_front_bio(transport);
305 return hEvent;
306}
307
308static DWORD freerdp_peer_get_event_handles(freerdp_peer* client, HANDLE* events, DWORD count)
309{
310 WINPR_ASSERT(client);
311 WINPR_ASSERT(client->context);
312 WINPR_ASSERT(client->context->rdp);
313 return transport_get_event_handles(client->context->rdp->transport, events, count);
314}
315
316static BOOL freerdp_peer_check_fds(freerdp_peer* peer)
317{
318 int status = 0;
319 rdpRdp* rdp = nullptr;
320
321 WINPR_ASSERT(peer);
322 WINPR_ASSERT(peer->context);
323
324 rdp = peer->context->rdp;
325 status = rdp_check_fds(rdp);
326
327 return (status >= 0);
328}
329
330static state_run_t peer_recv_data_pdu(freerdp_peer* client, wStream* s,
331 WINPR_ATTR_UNUSED UINT16 totalLength)
332{
333 BYTE type = 0;
334 UINT16 length = 0;
335 UINT32 share_id = 0;
336 BYTE compressed_type = 0;
337 UINT16 compressed_len = 0;
338 rdpUpdate* update = nullptr;
339
340 WINPR_ASSERT(s);
341 WINPR_ASSERT(client);
342 WINPR_ASSERT(client->context);
343 rdpRdp* rdp = client->context->rdp;
344 WINPR_ASSERT(rdp);
345 WINPR_ASSERT(rdp->mcs);
346
347 update = client->context->update;
348 WINPR_ASSERT(update);
349
350 if (!rdp_read_share_data_header(rdp, s, &length, &type, &share_id, &compressed_type,
351 &compressed_len))
352 return STATE_RUN_FAILED;
353
354#ifdef WITH_DEBUG_RDP
355 WLog_Print(rdp->log, WLOG_DEBUG, "recv %s Data PDU (0x%02" PRIX8 "), length: %" PRIu16 "",
356 data_pdu_type_to_string(type), type, length);
357#endif
358
359 switch (type)
360 {
361 case DATA_PDU_TYPE_SYNCHRONIZE:
362 if (!rdp_recv_client_synchronize_pdu(rdp, s))
363 return STATE_RUN_FAILED;
364
365 break;
366
367 case DATA_PDU_TYPE_CONTROL:
368 if (!rdp_server_accept_client_control_pdu(rdp, s))
369 return STATE_RUN_FAILED;
370
371 break;
372
373 case DATA_PDU_TYPE_INPUT:
374 if (!input_recv(rdp->input, s))
375 return STATE_RUN_FAILED;
376
377 break;
378
379 case DATA_PDU_TYPE_BITMAP_CACHE_PERSISTENT_LIST:
380 if (!rdp_server_accept_client_persistent_key_list_pdu(rdp, s))
381 return STATE_RUN_FAILED;
382 break;
383
384 case DATA_PDU_TYPE_FONT_LIST:
385 if (!rdp_server_accept_client_font_list_pdu(rdp, s))
386 return STATE_RUN_FAILED;
387
388 return STATE_RUN_CONTINUE; // State changed, trigger rerun
389
390 case DATA_PDU_TYPE_SHUTDOWN_REQUEST:
391 mcs_send_disconnect_provider_ultimatum(rdp->mcs,
392 Disconnect_Ultimatum_provider_initiated);
393 WLog_WARN(TAG, "disconnect provider ultimatum sent to peer, closing connection");
394 return STATE_RUN_QUIT_SESSION;
395
396 case DATA_PDU_TYPE_FRAME_ACKNOWLEDGE:
397 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
398 return STATE_RUN_FAILED;
399
400 Stream_Read_UINT32(s, client->ack_frame_id);
401 if (update->SurfaceFrameAcknowledge)
402 {
403 if (!update->SurfaceFrameAcknowledge(update->context, client->ack_frame_id))
404 return STATE_RUN_FAILED;
405 }
406 break;
407
408 case DATA_PDU_TYPE_REFRESH_RECT:
409 if (!update_read_refresh_rect(update, s))
410 return STATE_RUN_FAILED;
411
412 break;
413
414 case DATA_PDU_TYPE_SUPPRESS_OUTPUT:
415 if (!update_read_suppress_output(update, s))
416 return STATE_RUN_FAILED;
417
418 break;
419
420 default:
421 WLog_ERR(TAG, "Data PDU type %" PRIu8 "", type);
422 break;
423 }
424
425 return STATE_RUN_SUCCESS;
426}
427
428static state_run_t peer_recv_tpkt_pdu(freerdp_peer* client, wStream* s)
429{
430 state_run_t rc = STATE_RUN_SUCCESS;
431 UINT16 length = 0;
432 UINT16 pduType = 0;
433 UINT16 pduSource = 0;
434 UINT16 channelId = 0;
435 UINT16 securityFlags = 0;
436
437 WINPR_ASSERT(s);
438 WINPR_ASSERT(client);
439 WINPR_ASSERT(client->context);
440
441 rdpRdp* rdp = client->context->rdp;
442 WINPR_ASSERT(rdp);
443 WINPR_ASSERT(rdp->mcs);
444
445 rdpSettings* settings = client->context->settings;
446 WINPR_ASSERT(settings);
447
448 if (!rdp_read_header(rdp, s, &length, &channelId))
449 return STATE_RUN_FAILED;
450
451 rdp->inPackets++;
452 if (freerdp_shall_disconnect_context(rdp->context))
453 return STATE_RUN_SUCCESS;
454
455 if (rdp_get_state(rdp) <= CONNECTION_STATE_LICENSING)
456 {
457 return rdp_handle_message_channel(rdp, s, channelId, length);
458 }
459
460 if (!rdp_handle_optional_rdp_decryption(rdp, s, &length, &securityFlags))
461 return STATE_RUN_FAILED;
462
463 if (channelId == MCS_GLOBAL_CHANNEL_ID)
464 {
465 char buffer[256] = WINPR_C_ARRAY_INIT;
466 UINT16 pduLength = 0;
467 UINT16 remain = 0;
468 if (!rdp_read_share_control_header(rdp, s, &pduLength, &remain, &pduType, &pduSource))
469 return STATE_RUN_FAILED;
470
471 settings->PduSource = pduSource;
472
473 WLog_DBG(TAG, "Received %s", pdu_type_to_str(pduType, buffer, sizeof(buffer)));
474 switch (pduType)
475 {
476 case PDU_TYPE_DATA:
477 rc = peer_recv_data_pdu(client, s, pduLength);
478 break;
479
480 case PDU_TYPE_CONFIRM_ACTIVE:
481 if (!rdp_server_accept_confirm_active(rdp, s, pduLength))
482 return STATE_RUN_FAILED;
483
484 break;
485
486 case PDU_TYPE_FLOW_RESPONSE:
487 case PDU_TYPE_FLOW_STOP:
488 case PDU_TYPE_FLOW_TEST:
489 if (!Stream_SafeSeek(s, remain))
490 {
491 WLog_WARN(TAG, "Short PDU, need %" PRIu16 " bytes, got %" PRIuz, remain,
492 Stream_GetRemainingLength(s));
493 return STATE_RUN_FAILED;
494 }
495 break;
496
497 default:
498 WLog_ERR(TAG, "Client sent unknown pduType %" PRIu16 "", pduType);
499 return STATE_RUN_FAILED;
500 }
501 }
502 else if ((rdp->mcs->messageChannelId > 0) && (channelId == rdp->mcs->messageChannelId))
503 {
504 if (!settings->UseRdpSecurityLayer)
505 {
506 if (!rdp_read_security_header(rdp, s, &securityFlags, nullptr))
507 return STATE_RUN_FAILED;
508 }
509
510 return rdp_recv_message_channel_pdu(rdp, s, securityFlags);
511 }
512 else
513 {
514 if (!freerdp_channel_peer_process(client, s, channelId))
515 return STATE_RUN_FAILED;
516 }
517 if (!tpkt_ensure_stream_consumed(rdp->log, s, length))
518 return STATE_RUN_FAILED;
519
520 return rc;
521}
522
523static state_run_t peer_recv_handle_auto_detect(freerdp_peer* client, wStream* s)
524{
525 state_run_t ret = STATE_RUN_FAILED;
526 rdpRdp* rdp = nullptr;
527
528 WINPR_ASSERT(client);
529 WINPR_ASSERT(s);
530 WINPR_ASSERT(client->context);
531
532 rdp = client->context->rdp;
533 WINPR_ASSERT(rdp);
534
535 const rdpSettings* settings = client->context->settings;
536 WINPR_ASSERT(settings);
537
538 if (freerdp_settings_get_bool(settings, FreeRDP_NetworkAutoDetect))
539 {
540 switch (rdp_get_state(rdp))
541 {
542 case CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_REQUEST:
543 autodetect_on_connect_time_auto_detect_begin(rdp->autodetect);
544 switch (autodetect_get_state(rdp->autodetect))
545 {
546 case FREERDP_AUTODETECT_STATE_REQUEST:
547 ret = STATE_RUN_SUCCESS;
548 if (!rdp_server_transition_to_state(
549 rdp, CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_RESPONSE))
550 return STATE_RUN_FAILED;
551 break;
552 case FREERDP_AUTODETECT_STATE_COMPLETE:
553 ret = STATE_RUN_CONTINUE; /* Rerun in next state */
554 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_LICENSING))
555 return STATE_RUN_FAILED;
556 break;
557 default:
558 break;
559 }
560 break;
561 case CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_RESPONSE:
562 ret = peer_recv_pdu(client, s);
563 if (state_run_success(ret))
564 {
565 autodetect_on_connect_time_auto_detect_progress(rdp->autodetect);
566 switch (autodetect_get_state(rdp->autodetect))
567 {
568 case FREERDP_AUTODETECT_STATE_REQUEST:
569 ret = STATE_RUN_SUCCESS;
570 break;
571 case FREERDP_AUTODETECT_STATE_COMPLETE:
572 ret = STATE_RUN_CONTINUE; /* Rerun in next state */
573 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_LICENSING))
574 return STATE_RUN_FAILED;
575 break;
576 default:
577 break;
578 }
579 }
580 break;
581 default:
582 WINPR_ASSERT(FALSE);
583 break;
584 }
585 }
586 else
587 {
588 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_LICENSING))
589 return STATE_RUN_FAILED;
590
591 ret = STATE_RUN_CONTINUE; /* Rerun in next state */
592 }
593
594 return ret;
595}
596
597static state_run_t peer_recv_handle_licensing(freerdp_peer* client, wStream* s)
598{
599 state_run_t ret = STATE_RUN_FAILED;
600 rdpRdp* rdp = nullptr;
601 rdpSettings* settings = nullptr;
602
603 WINPR_ASSERT(client);
604 WINPR_ASSERT(s);
605 WINPR_ASSERT(client->context);
606
607 rdp = client->context->rdp;
608 WINPR_ASSERT(rdp);
609
610 settings = rdp->settings;
611 WINPR_ASSERT(settings);
612
613 switch (license_get_state(rdp->license))
614 {
615 case LICENSE_STATE_INITIAL:
616 {
617 const BOOL required =
618 freerdp_settings_get_bool(settings, FreeRDP_ServerLicenseRequired);
619
620 if (required)
621 {
622 if (!license_server_configure(rdp->license))
623 ret = STATE_RUN_FAILED;
624 else if (!license_server_send_request(rdp->license))
625 ret = STATE_RUN_FAILED;
626 else
627 ret = STATE_RUN_SUCCESS;
628 }
629 else
630 {
631 if (license_send_valid_client_error_packet(rdp))
632 ret = STATE_RUN_CONTINUE; /* Rerun in next state, might be capabilities */
633 }
634 }
635 break;
636 case LICENSE_STATE_COMPLETED:
637 ret = STATE_RUN_CONTINUE; /* Licensing completed, continue in next state */
638 break;
639 case LICENSE_STATE_ABORTED:
640 ret = STATE_RUN_FAILED;
641 break;
642 default:
643 ret = peer_recv_pdu(client, s);
644 break;
645 }
646
647 return ret;
648}
649
650static state_run_t peer_recv_fastpath_pdu(freerdp_peer* client, wStream* s)
651{
652 rdpRdp* rdp = nullptr;
653 UINT16 length = 0;
654 BOOL rc = 0;
655 rdpFastPath* fastpath = nullptr;
656
657 WINPR_ASSERT(s);
658 WINPR_ASSERT(client);
659 WINPR_ASSERT(client->context);
660
661 rdp = client->context->rdp;
662 WINPR_ASSERT(rdp);
663
664 fastpath = rdp->fastpath;
665 WINPR_ASSERT(fastpath);
666
667 rc = fastpath_read_header_rdp(fastpath, s, &length);
668
669 if (!rc || (length == 0))
670 {
671 WLog_ERR(TAG, "incorrect FastPath PDU header length %" PRIu16 "", length);
672 return STATE_RUN_FAILED;
673 }
674 if (!Stream_CheckAndLogRequiredLength(TAG, s, length))
675 return STATE_RUN_FAILED;
676
677 if (!fastpath_decrypt(fastpath, s, &length))
678 return STATE_RUN_FAILED;
679
680 rdp->inPackets++;
681
682 return fastpath_recv_inputs(fastpath, s);
683}
684
685state_run_t peer_recv_pdu(freerdp_peer* client, wStream* s)
686{
687 int rc = tpkt_verify_header(s);
688
689 if (rc > 0)
690 return peer_recv_tpkt_pdu(client, s);
691 else if (rc == 0)
692 return peer_recv_fastpath_pdu(client, s);
693 else
694 return STATE_RUN_FAILED;
695}
696
697static state_run_t peer_unexpected_client_message(rdpRdp* rdp, UINT32 flag)
698{
699 char buffer[1024] = WINPR_C_ARRAY_INIT;
700 WLog_WARN(TAG, "Unexpected client message in state %s, missing flag %s",
701 rdp_get_state_string(rdp), rdp_finalize_flags_to_str(flag, buffer, sizeof(buffer)));
702 return STATE_RUN_SUCCESS; /* we ignore this as per spec input PDU are already allowed */
703}
704
705state_run_t rdp_peer_handle_state_demand_active(freerdp_peer* client)
706{
707 state_run_t ret = STATE_RUN_FAILED;
708
709 WINPR_ASSERT(client);
710 WINPR_ASSERT(client->context);
711
712 rdpRdp* rdp = client->context->rdp;
713 WINPR_ASSERT(rdp);
714
715 if (client->Capabilities && !client->Capabilities(client))
716 {
717 WLog_ERR(TAG, "[%s] freerdp_peer::Capabilities() callback failed",
718 rdp_get_state_string(rdp));
719 }
720 else if (!rdp_send_demand_active(rdp))
721 {
722 WLog_ERR(TAG, "[%s] rdp_send_demand_active() fail", rdp_get_state_string(rdp));
723 }
724 else
725 {
726 if (!rdp_server_transition_to_state(rdp,
727 CONNECTION_STATE_CAPABILITIES_EXCHANGE_MONITOR_LAYOUT))
728 return STATE_RUN_FAILED;
729 ret = STATE_RUN_CONTINUE;
730 }
731 return ret;
732}
733
740static state_run_t rdp_peer_handle_state_active(freerdp_peer* client)
741{
742 state_run_t ret = STATE_RUN_FAILED;
743
744 WINPR_ASSERT(client);
745 WINPR_ASSERT(client->context);
746
747 if (!client->connected)
748 {
753 IFCALLRET(client->PostConnect, client->connected, client);
754 }
755 if (!client->connected)
756 {
757 WLog_ERR(TAG, "PostConnect for peer %p failed", WINPR_CXX_COMPAT_CAST(const void*, client));
758 ret = STATE_RUN_FAILED;
759 }
760 else if (!client->activated)
761 {
762 BOOL activated = TRUE;
763
764 /* Set client->activated TRUE before calling the Activate callback.
765 * the Activate callback might reset the client->activated flag even if it returns success
766 * (e.g. deactivate/reactivate sequence) */
767 client->activated = TRUE;
768 IFCALLRET(client->Activate, activated, client);
769
770 if (!activated)
771 {
772 WLog_ERR(TAG, "Activate for peer %p failed",
773 WINPR_CXX_COMPAT_CAST(const void*, client));
774 ret = STATE_RUN_FAILED;
775 }
776 else
777 ret = STATE_RUN_SUCCESS;
778 }
779 else
780 ret = STATE_RUN_ACTIVE;
781 return ret;
782}
783
784static state_run_t peer_recv_callback_internal(WINPR_ATTR_UNUSED rdpTransport* transport,
785 wStream* s, void* extra)
786{
787 freerdp_peer* client = (freerdp_peer*)extra;
788 state_run_t ret = STATE_RUN_FAILED;
789
790 WINPR_ASSERT(transport);
791 WINPR_ASSERT(client);
792 WINPR_ASSERT(client->context);
793
794 rdpRdp* rdp = client->context->rdp;
795 WINPR_ASSERT(rdp);
796
797 rdpSettings* settings = client->context->settings;
798 WINPR_ASSERT(settings);
799
800 if (client->ReachedState)
801 {
802 if (!client->ReachedState(client, rdp_get_state(rdp)))
803 return STATE_RUN_FAILED;
804 }
805 switch (rdp_get_state(rdp))
806 {
807 case CONNECTION_STATE_INITIAL:
808 if (!freerdp_settings_enforce_consistency(settings))
809 ret = STATE_RUN_FAILED;
810 else if (rdp_server_transition_to_state(rdp, CONNECTION_STATE_NEGO))
811 ret = STATE_RUN_CONTINUE;
812 break;
813
814 case CONNECTION_STATE_NEGO:
815 if (!rdp_server_accept_nego(rdp, s))
816 {
817 WLog_ERR(TAG, "%s - rdp_server_accept_nego() fail", rdp_get_state_string(rdp));
818 }
819 else
820 {
821 const UINT32 SelectedProtocol = nego_get_selected_protocol(rdp->nego);
822
823 settings->RdstlsSecurity = (SelectedProtocol & PROTOCOL_RDSTLS) != 0;
824 settings->NlaSecurity = (SelectedProtocol & PROTOCOL_HYBRID) != 0;
825 settings->TlsSecurity = (SelectedProtocol & PROTOCOL_SSL) != 0;
826 settings->RdpSecurity = (SelectedProtocol == PROTOCOL_RDP) != 0;
827
828 client->authenticated = FALSE;
829 if (SelectedProtocol & PROTOCOL_HYBRID)
830 {
832 (SEC_WINNT_AUTH_IDENTITY_INFO*)nego_get_identity(rdp->nego);
833 if (sspi_CopyAuthIdentity(&client->identity, identity) >= 0)
834 {
835 client->authenticated =
836 IFCALLRESULT(TRUE, client->Logon, client, &client->identity, TRUE);
837 }
838 nego_free_nla(rdp->nego);
839 }
840 else
841 {
842 client->authenticated =
843 IFCALLRESULT(TRUE, client->Logon, client, &client->identity, FALSE);
844 }
845 if (!client->authenticated)
846 ret = STATE_RUN_FAILED;
847 else
848 {
849 if (rdp_server_transition_to_state(rdp, CONNECTION_STATE_MCS_CREATE_REQUEST))
850 ret = STATE_RUN_SUCCESS;
851 }
852 }
853 break;
854
855 case CONNECTION_STATE_NLA:
856 WINPR_ASSERT(FALSE); // TODO
857 break;
858
859 case CONNECTION_STATE_MCS_CREATE_REQUEST:
860 if (!rdp_server_accept_mcs_connect_initial(rdp, s))
861 {
862 WLog_ERR(TAG,
863 "%s - "
864 "rdp_server_accept_mcs_connect_initial() fail",
865 rdp_get_state_string(rdp));
866 }
867 else
868 ret = STATE_RUN_SUCCESS;
869
870 break;
871
872 case CONNECTION_STATE_MCS_ERECT_DOMAIN:
873 if (!rdp_server_accept_mcs_erect_domain_request(rdp, s))
874 {
875 WLog_ERR(TAG,
876 "%s - "
877 "rdp_server_accept_mcs_erect_domain_request() fail",
878 rdp_get_state_string(rdp));
879 }
880 else
881 ret = STATE_RUN_SUCCESS;
882
883 break;
884
885 case CONNECTION_STATE_MCS_ATTACH_USER:
886 if (!rdp_server_accept_mcs_attach_user_request(rdp, s))
887 {
888 WLog_ERR(TAG,
889 "%s - "
890 "rdp_server_accept_mcs_attach_user_request() fail",
891 rdp_get_state_string(rdp));
892 }
893 else
894 ret = STATE_RUN_SUCCESS;
895
896 break;
897
898 case CONNECTION_STATE_MCS_CHANNEL_JOIN_REQUEST:
899 if (!rdp_server_accept_mcs_channel_join_request(rdp, s))
900 {
901 WLog_ERR(TAG,
902 "%s - "
903 "rdp_server_accept_mcs_channel_join_request() fail",
904 rdp_get_state_string(rdp));
905 }
906 else
907 ret = STATE_RUN_SUCCESS;
908 break;
909
910 case CONNECTION_STATE_RDP_SECURITY_COMMENCEMENT:
911 ret = STATE_RUN_SUCCESS;
912
913 if (!rdp_server_establish_keys(rdp, s))
914 {
915 WLog_ERR(TAG,
916 "%s - "
917 "rdp_server_establish_keys() fail",
918 rdp_get_state_string(rdp));
919 ret = STATE_RUN_FAILED;
920 }
921
922 if (state_run_success(ret))
923 {
924 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_SECURE_SETTINGS_EXCHANGE))
925 ret = STATE_RUN_FAILED;
926 else if (Stream_GetRemainingLength(s) > 0)
927 ret = STATE_RUN_CONTINUE; /* Rerun function */
928 }
929 break;
930
931 case CONNECTION_STATE_SECURE_SETTINGS_EXCHANGE:
932 if (rdp_recv_client_info(rdp, s))
933 {
934 if (rdp_server_transition_to_state(
935 rdp, CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_REQUEST))
936 ret = STATE_RUN_CONTINUE;
937 }
938 break;
939
940 case CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_REQUEST:
941 case CONNECTION_STATE_CONNECT_TIME_AUTO_DETECT_RESPONSE:
942 ret = peer_recv_handle_auto_detect(client, s);
943 break;
944
945 case CONNECTION_STATE_LICENSING:
946 ret = peer_recv_handle_licensing(client, s);
947 if (ret == STATE_RUN_CONTINUE)
948 {
949 if (!rdp_server_transition_to_state(
950 rdp, CONNECTION_STATE_MULTITRANSPORT_BOOTSTRAPPING_REQUEST))
951 ret = STATE_RUN_FAILED;
952 }
953 break;
954
955 case CONNECTION_STATE_MULTITRANSPORT_BOOTSTRAPPING_REQUEST:
956 if (settings->SupportMultitransport &&
957 ((settings->MultitransportFlags & INITIATE_REQUEST_PROTOCOL_UDPFECR) != 0))
958 {
959 /* only UDP reliable for now, nobody does lossy UDP (MS-RDPUDP only) these days */
960 ret = multitransport_server_request(rdp->multitransport,
961 INITIATE_REQUEST_PROTOCOL_UDPFECR);
962 switch (ret)
963 {
964 case STATE_RUN_SUCCESS:
965 if (!rdp_server_transition_to_state(
966 rdp, CONNECTION_STATE_MULTITRANSPORT_BOOTSTRAPPING_RESPONSE))
967 ret = STATE_RUN_FAILED;
968 break;
969 case STATE_RUN_CONTINUE:
970 /* mismatch on the supported kind of UDP transports */
971 if (!rdp_server_transition_to_state(
972 rdp, CONNECTION_STATE_CAPABILITIES_EXCHANGE_DEMAND_ACTIVE))
973 ret = STATE_RUN_FAILED;
974 break;
975 default:
976 break;
977 }
978 }
979 else
980 {
981 if (rdp_server_transition_to_state(
982 rdp, CONNECTION_STATE_CAPABILITIES_EXCHANGE_DEMAND_ACTIVE))
983 ret = STATE_RUN_CONTINUE; /* Rerun, initialize next state */
984 }
985 break;
986 case CONNECTION_STATE_MULTITRANSPORT_BOOTSTRAPPING_RESPONSE:
987 ret = peer_recv_pdu(client, s);
988 break;
989
990 case CONNECTION_STATE_CAPABILITIES_EXCHANGE_DEMAND_ACTIVE:
991 ret = rdp_peer_handle_state_demand_active(client);
992 break;
993
994 case CONNECTION_STATE_CAPABILITIES_EXCHANGE_MONITOR_LAYOUT:
995 if (freerdp_settings_get_bool(settings, FreeRDP_SupportMonitorLayoutPdu))
996 {
997 MONITOR_DEF* monitors = nullptr;
998
999 if (client->AdjustMonitorsLayout)
1000 {
1001 if (!client->AdjustMonitorsLayout(client))
1002 return STATE_RUN_FAILED;
1003 }
1004
1005 /* client supports the monitorLayout PDU, let's send him the monitors if any */
1006 ret = STATE_RUN_SUCCESS;
1007 if (freerdp_settings_get_uint32(settings, FreeRDP_MonitorCount) == 0)
1008 {
1009 const UINT32 w = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
1010 const UINT32 h = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
1011 const rdpMonitor primary = { .x = 0,
1012 .y = 0,
1013 .width = WINPR_ASSERTING_INT_CAST(int32_t, w),
1014 .height = WINPR_ASSERTING_INT_CAST(int32_t, h),
1015 .is_primary = TRUE,
1016 .orig_screen = 0,
1017 .attributes = { .physicalWidth = w,
1018 .physicalHeight = h,
1019 .orientation =
1020 ORIENTATION_LANDSCAPE,
1021 .desktopScaleFactor = 100,
1022 .deviceScaleFactor = 100 } };
1023 if (!freerdp_settings_set_pointer_array(settings, FreeRDP_MonitorDefArray, 0,
1024 &primary))
1025 ret = STATE_RUN_FAILED;
1026 else if (!freerdp_settings_set_uint32(settings, FreeRDP_MonitorCount, 1))
1027 ret = STATE_RUN_FAILED;
1028 }
1029 if (state_run_failed(ret))
1030 {
1031 }
1032 else if (!display_convert_rdp_monitor_to_monitor_def(
1033 settings->MonitorCount, settings->MonitorDefArray, &monitors))
1034 {
1035 ret = STATE_RUN_FAILED;
1036 }
1037 else if (!freerdp_display_send_monitor_layout(rdp->context, settings->MonitorCount,
1038 monitors))
1039 {
1040 ret = STATE_RUN_FAILED;
1041 }
1042 else
1043 ret = STATE_RUN_SUCCESS;
1044 free(monitors);
1045
1046 const size_t len = Stream_GetRemainingLength(s);
1047 if (!state_run_failed(ret) && (len > 0))
1048 ret = STATE_RUN_CONTINUE;
1049 }
1050 else
1051 {
1052 const size_t len = Stream_GetRemainingLength(s);
1053 if (len > 0)
1054 ret = STATE_RUN_CONTINUE;
1055 else
1056 ret = STATE_RUN_SUCCESS;
1057 }
1058 if (state_run_success(ret))
1059 {
1060 if (!rdp_server_transition_to_state(
1061 rdp, CONNECTION_STATE_CAPABILITIES_EXCHANGE_CONFIRM_ACTIVE))
1062 ret = STATE_RUN_FAILED;
1063 }
1064 break;
1065
1066 case CONNECTION_STATE_CAPABILITIES_EXCHANGE_CONFIRM_ACTIVE:
1071 ret = peer_recv_pdu(client, s);
1072 break;
1073
1074 case CONNECTION_STATE_FINALIZATION_SYNC:
1075 ret = peer_recv_pdu(client, s);
1076 if (rdp_finalize_is_flag_set(rdp, FINALIZE_CS_SYNCHRONIZE_PDU))
1077 {
1078 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_FINALIZATION_COOPERATE))
1079 ret = STATE_RUN_FAILED;
1080 }
1081 else
1082 ret = peer_unexpected_client_message(rdp, FINALIZE_CS_SYNCHRONIZE_PDU);
1083 break;
1084 case CONNECTION_STATE_FINALIZATION_COOPERATE:
1085 ret = peer_recv_pdu(client, s);
1086 if (rdp_finalize_is_flag_set(rdp, FINALIZE_CS_CONTROL_COOPERATE_PDU))
1087 {
1088 if (!rdp_server_transition_to_state(rdp,
1089 CONNECTION_STATE_FINALIZATION_REQUEST_CONTROL))
1090 ret = STATE_RUN_FAILED;
1091 }
1092 else
1093 ret = peer_unexpected_client_message(rdp, FINALIZE_CS_CONTROL_COOPERATE_PDU);
1094 break;
1095 case CONNECTION_STATE_FINALIZATION_REQUEST_CONTROL:
1096 ret = peer_recv_pdu(client, s);
1097 if (rdp_finalize_is_flag_set(rdp, FINALIZE_CS_CONTROL_REQUEST_PDU))
1098 {
1099 if (!rdp_send_server_control_granted_pdu(rdp))
1100 ret = STATE_RUN_FAILED;
1101 else if (!rdp_server_transition_to_state(
1102 rdp, CONNECTION_STATE_FINALIZATION_PERSISTENT_KEY_LIST))
1103 ret = STATE_RUN_FAILED;
1104 }
1105 else
1106 ret = peer_unexpected_client_message(rdp, FINALIZE_CS_CONTROL_REQUEST_PDU);
1107 break;
1108 case CONNECTION_STATE_FINALIZATION_PERSISTENT_KEY_LIST:
1109 if (freerdp_settings_get_bool(settings, FreeRDP_BitmapCachePersistEnabled) &&
1110 !rdp_finalize_is_flag_set(rdp, FINALIZE_DEACTIVATE_REACTIVATE))
1111 {
1112 ret = peer_recv_pdu(client, s);
1113
1114 if (rdp_finalize_is_flag_set(rdp, FINALIZE_CS_PERSISTENT_KEY_LIST_PDU))
1115 {
1116 if (!rdp_server_transition_to_state(rdp,
1117 CONNECTION_STATE_FINALIZATION_FONT_LIST))
1118 ret = STATE_RUN_FAILED;
1119 }
1120 else
1121 ret = peer_unexpected_client_message(rdp,
1122 CONNECTION_STATE_FINALIZATION_FONT_LIST);
1123 }
1124 else
1125 {
1126 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_FINALIZATION_FONT_LIST))
1127 ret = STATE_RUN_FAILED;
1128 else
1129 ret = STATE_RUN_CONTINUE;
1130 }
1131 break;
1132 case CONNECTION_STATE_FINALIZATION_FONT_LIST:
1133 ret = peer_recv_pdu(client, s);
1134 if (state_run_success(ret))
1135 {
1136 if (rdp_finalize_is_flag_set(rdp, FINALIZE_CS_FONT_LIST_PDU))
1137 {
1138 if (!rdp_server_transition_to_state(rdp, CONNECTION_STATE_ACTIVE))
1139 ret = STATE_RUN_FAILED;
1140 else
1141 {
1142 update_reset_state(rdp->update);
1143 ret = STATE_RUN_CONTINUE;
1144 }
1145 }
1146 else
1147 ret = peer_unexpected_client_message(rdp, FINALIZE_CS_FONT_LIST_PDU);
1148 }
1149 break;
1150
1151 case CONNECTION_STATE_ACTIVE:
1152 ret = rdp_peer_handle_state_active(client);
1153 if (ret >= STATE_RUN_ACTIVE)
1154 ret = peer_recv_pdu(client, s);
1155 break;
1156
1157 /* States that must not happen in server state machine */
1158 case CONNECTION_STATE_FINALIZATION_CLIENT_SYNC:
1159 case CONNECTION_STATE_FINALIZATION_CLIENT_COOPERATE:
1160 case CONNECTION_STATE_FINALIZATION_CLIENT_GRANTED_CONTROL:
1161 case CONNECTION_STATE_FINALIZATION_CLIENT_FONT_MAP:
1162 default:
1163 WLog_ERR(TAG, "%s state %" PRId32, rdp_get_state_string(rdp),
1164 WINPR_CXX_COMPAT_CAST(int32_t, rdp_get_state(rdp)));
1165 break;
1166 }
1167
1168 return ret;
1169}
1170
1171static state_run_t peer_recv_callback(rdpTransport* transport, wStream* s, void* extra)
1172{
1173 char buffer[64] = WINPR_C_ARRAY_INIT;
1174 state_run_t rc = STATE_RUN_FAILED;
1175 const size_t start = Stream_GetPosition(s);
1176 const rdpContext* context = transport_get_context(transport);
1177 DWORD level = WLOG_TRACE;
1178 static wLog* log = nullptr;
1179 if (!log)
1180 log = WLog_Get(TAG);
1181
1182 WINPR_ASSERT(context);
1183 do
1184 {
1185 const rdpRdp* rdp = context->rdp;
1186 const char* old = rdp_get_state_string(rdp);
1187
1188 if (rc == STATE_RUN_TRY_AGAIN)
1189 {
1190 if (!Stream_SetPosition(s, start))
1191 return STATE_RUN_FAILED;
1192 }
1193 rc = peer_recv_callback_internal(transport, s, extra);
1194
1195 const size_t len = Stream_GetRemainingLength(s);
1196 if ((len > 0) && !state_run_continue(rc))
1197 level = WLOG_WARN;
1198 WLog_Print(log, level,
1199 "(server)[%s -> %s] current return %s [%" PRIuz " bytes not processed]", old,
1200 rdp_get_state_string(rdp), state_run_result_string(rc, buffer, sizeof(buffer)),
1201 len);
1202 } while (state_run_continue(rc));
1203
1204 return rc;
1205}
1206
1207static BOOL freerdp_peer_close(freerdp_peer* client)
1208{
1209 UINT32 SelectedProtocol = 0;
1210 rdpContext* context = nullptr;
1211
1212 WINPR_ASSERT(client);
1213
1214 context = client->context;
1215 WINPR_ASSERT(context);
1216 WINPR_ASSERT(context->settings);
1217 WINPR_ASSERT(context->rdp);
1218
1222 SelectedProtocol = nego_get_selected_protocol(context->rdp->nego);
1223
1224 if (SelectedProtocol & PROTOCOL_FAILED_NEGO)
1225 return TRUE;
1226
1232 if (!rdp_send_deactivate_all(context->rdp))
1233 return FALSE;
1234
1235 if (freerdp_settings_get_bool(context->settings, FreeRDP_SupportErrorInfoPdu))
1236 {
1237 if (!rdp_send_error_info(context->rdp))
1238 return FALSE;
1239 }
1240
1241 return mcs_send_disconnect_provider_ultimatum(context->rdp->mcs,
1242 Disconnect_Ultimatum_provider_initiated);
1243}
1244
1245static void freerdp_peer_disconnect(freerdp_peer* client)
1246{
1247 rdpTransport* transport = nullptr;
1248 WINPR_ASSERT(client);
1249
1250 transport = freerdp_get_transport(client->context);
1251 transport_disconnect(transport);
1252}
1253
1254static BOOL freerdp_peer_send_channel_data(freerdp_peer* client, UINT16 channelId, const BYTE* data,
1255 size_t size)
1256{
1257 WINPR_ASSERT(client);
1258 WINPR_ASSERT(client->context);
1259 WINPR_ASSERT(client->context->rdp);
1260 return rdp_send_channel_data(client->context->rdp, channelId, data, size);
1261}
1262
1263static BOOL freerdp_peer_send_server_redirection_pdu(freerdp_peer* peer,
1264 const rdpRedirection* redirection)
1265{
1266 BOOL rc = FALSE;
1267 WINPR_ASSERT(peer);
1268 WINPR_ASSERT(peer->context);
1269
1270 UINT16 sec_flags = 0;
1271 wStream* s = rdp_send_stream_pdu_init(peer->context->rdp, &sec_flags);
1272 if (!s)
1273 return FALSE;
1274 if (!rdp_write_enhanced_security_redirection_packet(s, redirection))
1275 goto fail;
1276 if (!rdp_send_pdu(peer->context->rdp, s, PDU_TYPE_SERVER_REDIRECTION, 0, sec_flags))
1277 goto fail;
1278 rc = rdp_reset_runtime_settings(peer->context->rdp);
1279fail:
1280 Stream_Release(s);
1281 return rc;
1282}
1283
1284static BOOL freerdp_peer_send_channel_packet(freerdp_peer* client, UINT16 channelId,
1285 size_t totalSize, UINT32 flags, const BYTE* data,
1286 size_t chunkSize)
1287{
1288 WINPR_ASSERT(client);
1289 WINPR_ASSERT(client->context);
1290 WINPR_ASSERT(client->context->rdp);
1291 return rdp_channel_send_packet(client->context->rdp, channelId, totalSize, flags, data,
1292 chunkSize);
1293}
1294
1295static BOOL freerdp_peer_is_write_blocked(freerdp_peer* peer)
1296{
1297 rdpTransport* transport = nullptr;
1298 WINPR_ASSERT(peer);
1299 WINPR_ASSERT(peer->context);
1300 WINPR_ASSERT(peer->context->rdp);
1301 WINPR_ASSERT(peer->context->rdp->transport);
1302 transport = peer->context->rdp->transport;
1303 return transport_is_write_blocked(transport);
1304}
1305
1306static int freerdp_peer_drain_output_buffer(freerdp_peer* peer)
1307{
1308 rdpTransport* transport = nullptr;
1309 WINPR_ASSERT(peer);
1310 WINPR_ASSERT(peer->context);
1311 WINPR_ASSERT(peer->context->rdp);
1312 WINPR_ASSERT(peer->context->rdp->transport);
1313 transport = peer->context->rdp->transport;
1314 return transport_drain_output_buffer(transport);
1315}
1316
1317static BOOL freerdp_peer_has_more_to_read(freerdp_peer* peer)
1318{
1319 WINPR_ASSERT(peer);
1320 WINPR_ASSERT(peer->context);
1321 WINPR_ASSERT(peer->context->rdp);
1322 return transport_have_more_bytes_to_read(peer->context->rdp->transport);
1323}
1324
1325static LicenseCallbackResult freerdp_peer_nolicense(freerdp_peer* peer,
1326 WINPR_ATTR_UNUSED wStream* s)
1327{
1328 rdpRdp* rdp = nullptr;
1329
1330 WINPR_ASSERT(peer);
1331 WINPR_ASSERT(peer->context);
1332
1333 rdp = peer->context->rdp;
1334
1335 if (!license_send_valid_client_error_packet(rdp))
1336 {
1337 WLog_ERR(TAG, "freerdp_peer_nolicense: license_send_valid_client_error_packet() failed");
1338 return LICENSE_CB_ABORT;
1339 }
1340
1341 return LICENSE_CB_COMPLETED;
1342}
1343
1344BOOL freerdp_peer_context_new(freerdp_peer* client)
1345{
1346 return freerdp_peer_context_new_ex(client, nullptr);
1347}
1348
1349void freerdp_peer_context_free(freerdp_peer* client)
1350{
1351 if (!client)
1352 return;
1353
1354 IFCALL(client->ContextFree, client, client->context);
1355
1356 if (client->context)
1357 {
1358 rdpContext* ctx = client->context;
1359
1360 (void)CloseHandle(ctx->channelErrorEvent);
1361 ctx->channelErrorEvent = nullptr;
1362 free(ctx->errorDescription);
1363 ctx->errorDescription = nullptr;
1364 rdp_free(ctx->rdp);
1365 ctx->rdp = nullptr;
1366 metrics_free(ctx->metrics);
1367 ctx->metrics = nullptr;
1368 stream_dump_free(ctx->dump);
1369 ctx->dump = nullptr;
1370 free(ctx);
1371 }
1372 client->context = nullptr;
1373}
1374
1375static const char* os_major_type_to_string(UINT16 osMajorType)
1376{
1377 switch (osMajorType)
1378 {
1379 case OSMAJORTYPE_UNSPECIFIED:
1380 return "Unspecified platform";
1381 case OSMAJORTYPE_WINDOWS:
1382 return "Windows platform";
1383 case OSMAJORTYPE_OS2:
1384 return "OS/2 platform";
1385 case OSMAJORTYPE_MACINTOSH:
1386 return "Macintosh platform";
1387 case OSMAJORTYPE_UNIX:
1388 return "UNIX platform";
1389 case OSMAJORTYPE_IOS:
1390 return "iOS platform";
1391 case OSMAJORTYPE_OSX:
1392 return "OS X platform";
1393 case OSMAJORTYPE_ANDROID:
1394 return "Android platform";
1395 case OSMAJORTYPE_CHROME_OS:
1396 return "Chrome OS platform";
1397 default:
1398 break;
1399 }
1400
1401 return "Unknown platform";
1402}
1403
1404const char* freerdp_peer_os_major_type_string(freerdp_peer* client)
1405{
1406 WINPR_ASSERT(client);
1407
1408 rdpContext* context = client->context;
1409 WINPR_ASSERT(context);
1410 WINPR_ASSERT(context->settings);
1411
1412 const UINT32 osMajorType = freerdp_settings_get_uint32(context->settings, FreeRDP_OsMajorType);
1413 WINPR_ASSERT(osMajorType <= UINT16_MAX);
1414 return os_major_type_to_string((UINT16)osMajorType);
1415}
1416
1417static const char* os_minor_type_to_string(UINT16 osMinorType)
1418{
1419 switch (osMinorType)
1420 {
1421 case OSMINORTYPE_UNSPECIFIED:
1422 return "Unspecified version";
1423 case OSMINORTYPE_WINDOWS_31X:
1424 return "Windows 3.1x";
1425 case OSMINORTYPE_WINDOWS_95:
1426 return "Windows 95";
1427 case OSMINORTYPE_WINDOWS_NT:
1428 return "Windows NT";
1429 case OSMINORTYPE_OS2_V21:
1430 return "OS/2 2.1";
1431 case OSMINORTYPE_POWER_PC:
1432 return "PowerPC";
1433 case OSMINORTYPE_MACINTOSH:
1434 return "Macintosh";
1435 case OSMINORTYPE_NATIVE_XSERVER:
1436 return "Native X Server";
1437 case OSMINORTYPE_PSEUDO_XSERVER:
1438 return "Pseudo X Server";
1439 case OSMINORTYPE_WINDOWS_RT:
1440 return "Windows RT";
1441 default:
1442 break;
1443 }
1444
1445 return "Unknown version";
1446}
1447
1448const char* freerdp_peer_os_minor_type_string(freerdp_peer* client)
1449{
1450 WINPR_ASSERT(client);
1451
1452 rdpContext* context = client->context;
1453 WINPR_ASSERT(context);
1454 WINPR_ASSERT(context->settings);
1455
1456 const UINT32 osMinorType = freerdp_settings_get_uint32(context->settings, FreeRDP_OsMinorType);
1457 WINPR_ASSERT(osMinorType <= UINT16_MAX);
1458 return os_minor_type_to_string((UINT16)osMinorType);
1459}
1460
1461freerdp_peer* freerdp_peer_new(int sockfd)
1462{
1463 freerdp_peer* client = (freerdp_peer*)calloc(1, sizeof(freerdp_peer));
1464
1465 if (!client)
1466 return nullptr;
1467
1468 if (sockfd >= 0)
1469 {
1470 (void)freerdp_tcp_set_nodelay(WLog_Get(TAG), WLOG_DEBUG, sockfd);
1471 }
1472
1473 client->sockfd = sockfd;
1474 client->ContextSize = sizeof(rdpContext);
1475 client->Initialize = freerdp_peer_initialize;
1476#if defined(WITH_FREERDP_DEPRECATED)
1477 client->GetFileDescriptor = freerdp_peer_get_fds;
1478#endif
1479 client->GetEventHandle = freerdp_peer_get_event_handle;
1480 client->GetEventHandles = freerdp_peer_get_event_handles;
1481 client->CheckFileDescriptor = freerdp_peer_check_fds;
1482 client->Close = freerdp_peer_close;
1483 client->Disconnect = freerdp_peer_disconnect;
1484 client->SendChannelData = freerdp_peer_send_channel_data;
1485 client->SendChannelPacket = freerdp_peer_send_channel_packet;
1486 client->SendServerRedirection = freerdp_peer_send_server_redirection_pdu;
1487 client->IsWriteBlocked = freerdp_peer_is_write_blocked;
1488 client->DrainOutputBuffer = freerdp_peer_drain_output_buffer;
1489 client->HasMoreToRead = freerdp_peer_has_more_to_read;
1490 client->VirtualChannelOpen = freerdp_peer_virtual_channel_open;
1491 client->VirtualChannelClose = freerdp_peer_virtual_channel_close;
1492 client->VirtualChannelWrite = freerdp_peer_virtual_channel_write;
1493 client->VirtualChannelRead = nullptr; /* must be defined by server application */
1494 client->VirtualChannelGetData = freerdp_peer_virtual_channel_get_data;
1495 client->VirtualChannelSetData = freerdp_peer_virtual_channel_set_data;
1496 client->SetState = freerdp_peer_set_state;
1497
1498 return client;
1499}
1500
1501void freerdp_peer_free(freerdp_peer* client)
1502{
1503 if (!client)
1504 return;
1505
1506 sspi_FreeAuthIdentity(&client->identity);
1507 if (client->sockfd >= 0)
1508 closesocket((SOCKET)client->sockfd);
1509 free(client);
1510}
1511
1512static BOOL freerdp_peer_transport_setup(freerdp_peer* client)
1513{
1514 rdpRdp* rdp = nullptr;
1515
1516 WINPR_ASSERT(client);
1517 WINPR_ASSERT(client->context);
1518
1519 rdp = client->context->rdp;
1520 WINPR_ASSERT(rdp);
1521
1522 if (!transport_attach(rdp->transport, client->sockfd))
1523 return FALSE;
1524 client->sockfd = -1;
1525
1526 if (!transport_set_recv_callbacks(rdp->transport, peer_recv_callback, client))
1527 return FALSE;
1528
1529 if (!transport_set_blocking_mode(rdp->transport, FALSE))
1530 return FALSE;
1531
1532 return TRUE;
1533}
1534
1535BOOL freerdp_peer_context_new_ex(freerdp_peer* client, const rdpSettings* settings)
1536{
1537 rdpRdp* rdp = nullptr;
1538 rdpContext* context = nullptr;
1539 BOOL ret = TRUE;
1540
1541 if (!client)
1542 return FALSE;
1543
1544 WINPR_ASSERT(client->ContextSize >= sizeof(rdpContext));
1545 if (!(context = (rdpContext*)calloc(1, client->ContextSize)))
1546 goto fail;
1547
1548 client->context = context;
1549 context->peer = client;
1550 context->ServerMode = TRUE;
1551 context->log = WLog_Get(TAG);
1552 if (!context->log)
1553 goto fail;
1554
1555 if (settings)
1556 {
1557 context->settings = freerdp_settings_clone(settings);
1558 if (!context->settings)
1559 goto fail;
1560 }
1561
1562 context->dump = stream_dump_new();
1563 if (!context->dump)
1564 goto fail;
1565 if (!(context->metrics = metrics_new(context)))
1566 goto fail;
1567
1568 if (!(rdp = rdp_new(context)))
1569 goto fail;
1570
1571 rdp_log_build_warnings(rdp);
1572
1573#if defined(WITH_FREERDP_DEPRECATED)
1574 client->update = rdp->update;
1575 client->settings = rdp->settings;
1576 client->autodetect = rdp->autodetect;
1577#endif
1578 context->rdp = rdp;
1579 context->input = rdp->input;
1580 context->update = rdp->update;
1581 context->settings = rdp->settings;
1582 context->autodetect = rdp->autodetect;
1583 update_register_server_callbacks(rdp->update);
1584 autodetect_register_server_callbacks(rdp->autodetect);
1585
1586 if (!(context->channelErrorEvent = CreateEvent(nullptr, TRUE, FALSE, nullptr)))
1587 {
1588 WLog_ERR(TAG, "CreateEvent failed!");
1589 goto fail;
1590 }
1591
1592 if (!(context->errorDescription = calloc(1, 500)))
1593 {
1594 WLog_ERR(TAG, "calloc failed!");
1595 goto fail;
1596 }
1597
1598 if (!freerdp_peer_transport_setup(client))
1599 goto fail;
1600
1601 client->IsWriteBlocked = freerdp_peer_is_write_blocked;
1602 client->DrainOutputBuffer = freerdp_peer_drain_output_buffer;
1603 client->HasMoreToRead = freerdp_peer_has_more_to_read;
1604 client->LicenseCallback = freerdp_peer_nolicense;
1605 IFCALLRET(client->ContextNew, ret, client, client->context);
1606
1607 if (!ret)
1608 goto fail;
1609 return TRUE;
1610
1611fail:
1612 WLog_ERR(TAG, "ContextNew callback failed");
1613 freerdp_peer_context_free(client);
1614 return FALSE;
1615}
WINPR_ATTR_NODISCARD FREERDP_API const void * freerdp_settings_get_pointer(const rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a immutable pointer settings value.
FREERDP_API rdpSettings * freerdp_settings_clone(const rdpSettings *settings)
Creates a deep copy of settings.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_bool(rdpSettings *settings, FreeRDP_Settings_Keys_Bool id, BOOL val)
Sets a BOOL settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.