22#include <freerdp/config.h>
30#include <winpr/collections.h>
31#include <winpr/comm.h>
33#include <winpr/stream.h>
34#include <winpr/synch.h>
35#include <winpr/thread.h>
36#include <winpr/wlog.h>
37#include <winpr/assert.h>
39#include <freerdp/freerdp.h>
40#include <freerdp/channels/rdpdr.h>
41#include <freerdp/channels/log.h>
42#include <freerdp/utils/rdpdr_utils.h>
44#define TAG CHANNELS_TAG("serial.client")
46#define MAX_IRP_THREADS 5
52 SERIAL_DRIVER_ID ServerSerialDriverId;
57 wMessageQueue* MainIrpQueue;
60 wListDictionary* IrpThreads;
62 rdpContext* rdpcontext;
67 SERIAL_DEVICE* serial;
71static void close_terminated_irp_thread_handles(SERIAL_DEVICE* serial, BOOL forceClose);
72static NTSTATUS GetLastErrorToIoStatus(SERIAL_DEVICE* serial)
76 switch (GetLastError())
78 case ERROR_BAD_DEVICE:
79 return STATUS_INVALID_DEVICE_REQUEST;
81 case ERROR_CALL_NOT_IMPLEMENTED:
82 return STATUS_NOT_IMPLEMENTED;
85 return STATUS_CANCELLED;
87 case ERROR_INSUFFICIENT_BUFFER:
88 return STATUS_BUFFER_TOO_SMALL;
90 case ERROR_INVALID_DEVICE_OBJECT_PARAMETER:
91 return STATUS_INVALID_DEVICE_STATE;
93 case ERROR_INVALID_HANDLE:
94 return STATUS_INVALID_DEVICE_REQUEST;
96 case ERROR_INVALID_PARAMETER:
97 return STATUS_INVALID_PARAMETER;
100 return STATUS_IO_DEVICE_ERROR;
102 case ERROR_IO_PENDING:
103 return STATUS_PENDING;
105 case ERROR_NOT_SUPPORTED:
106 return STATUS_NOT_SUPPORTED;
109 return STATUS_TIMEOUT;
114 WLog_Print(serial->log, WLOG_DEBUG,
"unexpected last-error: 0x%08" PRIX32
"", GetLastError());
115 return STATUS_UNSUCCESSFUL;
118static UINT serial_process_irp_create(SERIAL_DEVICE* serial, IRP* irp)
120 DWORD DesiredAccess = 0;
121 DWORD SharedAccess = 0;
122 DWORD CreateDisposition = 0;
123 UINT32 PathLength = 0;
125 WINPR_ASSERT(serial);
128 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
129 return ERROR_INVALID_DATA;
131 Stream_Read_UINT32(irp->input, DesiredAccess);
132 Stream_Seek_UINT64(irp->input);
133 Stream_Seek_UINT32(irp->input);
134 Stream_Read_UINT32(irp->input, SharedAccess);
135 Stream_Read_UINT32(irp->input, CreateDisposition);
136 Stream_Seek_UINT32(irp->input);
137 Stream_Read_UINT32(irp->input, PathLength);
139 if (!Stream_SafeSeek(irp->input, PathLength))
140 return ERROR_INVALID_DATA;
142 WINPR_ASSERT(PathLength == 0);
159 WLog_Print(serial->log, WLOG_DEBUG,
160 "DesiredAccess: 0x%" PRIX32
", SharedAccess: 0x%" PRIX32
161 ", CreateDisposition: 0x%" PRIX32
"",
162 DesiredAccess, SharedAccess, CreateDisposition);
164 DesiredAccess = GENERIC_READ | GENERIC_WRITE;
166 CreateDisposition = OPEN_EXISTING;
168 serial->hComm = winpr_CreateFile(serial->device.name, DesiredAccess, SharedAccess,
170 CreateDisposition, 0,
173 if (!serial->hComm || (serial->hComm == INVALID_HANDLE_VALUE))
175 WLog_Print(serial->log, WLOG_WARN,
"CreateFile failure: %s last-error: 0x%08" PRIX32
"",
176 serial->device.name, GetLastError());
177 irp->IoStatus = STATUS_UNSUCCESSFUL;
181 _comm_setServerSerialDriver(serial->hComm, serial->ServerSerialDriverId);
182 _comm_set_permissive(serial->hComm, serial->permissive);
191 WINPR_ASSERT(irp->FileId == 0);
192 irp->FileId = irp->devman->id_sequence++;
193 irp->IoStatus = STATUS_SUCCESS;
194 WLog_Print(serial->log, WLOG_DEBUG,
"%s (DeviceId: %" PRIu32
", FileId: %" PRIu32
") created.",
195 serial->device.name, irp->device->id, irp->FileId);
198 DWORD BytesReturned = 0;
199 if (!CommDeviceIoControl(serial->hComm, IOCTL_SERIAL_RESET_DEVICE, NULL, 0, NULL, 0,
200 &BytesReturned, NULL))
205 Stream_Write_UINT32(irp->output, irp->FileId);
206 Stream_Write_UINT8(irp->output, 0);
207 return CHANNEL_RC_OK;
210static UINT serial_process_irp_close(SERIAL_DEVICE* serial, IRP* irp)
212 WINPR_ASSERT(serial);
215 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
216 return ERROR_INVALID_DATA;
218 Stream_Seek(irp->input, 32);
220 close_terminated_irp_thread_handles(serial, TRUE);
222 if (!CloseHandle(serial->hComm))
224 WLog_Print(serial->log, WLOG_WARN,
"CloseHandle failure: %s (%" PRIu32
") closed.",
225 serial->device.name, irp->device->id);
226 irp->IoStatus = STATUS_UNSUCCESSFUL;
230 WLog_Print(serial->log, WLOG_DEBUG,
"%s (DeviceId: %" PRIu32
", FileId: %" PRIu32
") closed.",
231 serial->device.name, irp->device->id, irp->FileId);
232 irp->IoStatus = STATUS_SUCCESS;
234 serial->hComm = NULL;
235 Stream_Zero(irp->output, 5);
236 return CHANNEL_RC_OK;
244static UINT serial_process_irp_read(SERIAL_DEVICE* serial, IRP* irp)
251 WINPR_ASSERT(serial);
254 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
255 return ERROR_INVALID_DATA;
257 Stream_Read_UINT32(irp->input, Length);
258 Stream_Read_UINT64(irp->input, Offset);
261 Stream_Seek(irp->input, 20);
262 buffer = (BYTE*)calloc(Length,
sizeof(BYTE));
266 irp->IoStatus = STATUS_NO_MEMORY;
273 WLog_Print(serial->log, WLOG_DEBUG,
"reading %" PRIu32
" bytes from %s", Length,
274 serial->device.name);
277 if (CommReadFile(serial->hComm, buffer, Length, &nbRead, NULL))
279 irp->IoStatus = STATUS_SUCCESS;
283 WLog_Print(serial->log, WLOG_DEBUG,
284 "read failure to %s, nbRead=%" PRIu32
", last-error: 0x%08" PRIX32
"",
285 serial->device.name, nbRead, GetLastError());
286 irp->IoStatus = GetLastErrorToIoStatus(serial);
289 WLog_Print(serial->log, WLOG_DEBUG,
"%" PRIu32
" bytes read from %s", nbRead,
290 serial->device.name);
292 Stream_Write_UINT32(irp->output, nbRead);
296 if (!Stream_EnsureRemainingCapacity(irp->output, nbRead))
298 WLog_Print(serial->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
300 return CHANNEL_RC_NO_MEMORY;
303 Stream_Write(irp->output, buffer, nbRead);
307 return CHANNEL_RC_OK;
310static UINT serial_process_irp_write(SERIAL_DEVICE* serial, IRP* irp)
316 WINPR_ASSERT(serial);
319 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
320 return ERROR_INVALID_DATA;
322 Stream_Read_UINT32(irp->input, Length);
323 Stream_Read_UINT64(irp->input, Offset);
326 if (!Stream_SafeSeek(irp->input, 20))
327 return ERROR_INVALID_DATA;
335 WLog_Print(serial->log, WLOG_DEBUG,
"writing %" PRIu32
" bytes to %s", Length,
336 serial->device.name);
338 const void* ptr = Stream_ConstPointer(irp->input);
339 if (!Stream_SafeSeek(irp->input, Length))
340 return ERROR_INVALID_DATA;
342 if (CommWriteFile(serial->hComm, ptr, Length, &nbWritten, NULL))
344 irp->IoStatus = STATUS_SUCCESS;
348 WLog_Print(serial->log, WLOG_DEBUG,
349 "write failure to %s, nbWritten=%" PRIu32
", last-error: 0x%08" PRIX32
"",
350 serial->device.name, nbWritten, GetLastError());
351 irp->IoStatus = GetLastErrorToIoStatus(serial);
354 WLog_Print(serial->log, WLOG_DEBUG,
"%" PRIu32
" bytes written to %s", nbWritten,
355 serial->device.name);
356 Stream_Write_UINT32(irp->output, nbWritten);
357 Stream_Write_UINT8(irp->output, 0);
358 return CHANNEL_RC_OK;
366static UINT serial_process_irp_device_control(SERIAL_DEVICE* serial, IRP* irp)
368 UINT32 IoControlCode = 0;
369 UINT32 InputBufferLength = 0;
370 BYTE* InputBuffer = NULL;
371 UINT32 OutputBufferLength = 0;
372 BYTE* OutputBuffer = NULL;
373 DWORD BytesReturned = 0;
375 WINPR_ASSERT(serial);
378 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
379 return ERROR_INVALID_DATA;
381 Stream_Read_UINT32(irp->input, OutputBufferLength);
382 Stream_Read_UINT32(irp->input, InputBufferLength);
383 Stream_Read_UINT32(irp->input, IoControlCode);
384 Stream_Seek(irp->input, 20);
386 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, InputBufferLength))
387 return ERROR_INVALID_DATA;
389 OutputBuffer = (BYTE*)calloc(OutputBufferLength,
sizeof(BYTE));
391 if (OutputBuffer == NULL)
393 irp->IoStatus = STATUS_NO_MEMORY;
397 InputBuffer = (BYTE*)calloc(InputBufferLength,
sizeof(BYTE));
399 if (InputBuffer == NULL)
401 irp->IoStatus = STATUS_NO_MEMORY;
405 Stream_Read(irp->input, InputBuffer, InputBufferLength);
406 WLog_Print(serial->log, WLOG_DEBUG,
407 "CommDeviceIoControl: CompletionId=%" PRIu32
", IoControlCode=[0x%" PRIX32
"] %s",
408 irp->CompletionId, IoControlCode, _comm_serial_ioctl_name(IoControlCode));
411 if (CommDeviceIoControl(serial->hComm, IoControlCode, InputBuffer, InputBufferLength,
412 OutputBuffer, OutputBufferLength, &BytesReturned, NULL))
417 irp->IoStatus = STATUS_SUCCESS;
421 WLog_Print(serial->log, WLOG_DEBUG,
422 "CommDeviceIoControl failure: IoControlCode=[0x%" PRIX32
423 "] %s, last-error: 0x%08" PRIX32
"",
424 IoControlCode, _comm_serial_ioctl_name(IoControlCode), GetLastError());
425 irp->IoStatus = GetLastErrorToIoStatus(serial);
432 WINPR_ASSERT(OutputBufferLength == BytesReturned);
433 Stream_Write_UINT32(irp->output, BytesReturned);
435 if (BytesReturned > 0)
437 if (!Stream_EnsureRemainingCapacity(irp->output, BytesReturned))
439 WLog_Print(serial->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
442 return CHANNEL_RC_NO_MEMORY;
445 Stream_Write(irp->output, OutputBuffer, BytesReturned);
458 return CHANNEL_RC_OK;
466static UINT serial_process_irp(SERIAL_DEVICE* serial, IRP* irp)
468 UINT error = CHANNEL_RC_OK;
470 WINPR_ASSERT(serial);
473 WLog_Print(serial->log, WLOG_DEBUG,
"IRP MajorFunction: %s, MinorFunction: 0x%08" PRIX32
"\n",
474 rdpdr_irp_string(irp->MajorFunction), irp->MinorFunction);
476 switch (irp->MajorFunction)
479 error = serial_process_irp_create(serial, irp);
483 error = serial_process_irp_close(serial, irp);
487 error = serial_process_irp_read(serial, irp);
491 error = serial_process_irp_write(serial, irp);
494 case IRP_MJ_DEVICE_CONTROL:
495 error = serial_process_irp_device_control(serial, irp);
499 irp->IoStatus = STATUS_NOT_SUPPORTED;
503 DWORD level = WLOG_TRACE;
507 WLog_Print(serial->log, level,
508 "[%s|0x%08" PRIx32
"] completed with %s [0x%08" PRIx32
"] (IoStatus %s [0x%08" PRIx32
510 rdpdr_irp_string(irp->MajorFunction), irp->MajorFunction, WTSErrorToString(error),
511 error, NtStatus2Tag(irp->IoStatus), WINPR_CXX_COMPAT_CAST(UINT32, irp->IoStatus));
516static DWORD WINAPI irp_thread_func(LPVOID arg)
518 IRP_THREAD_DATA* data = (IRP_THREAD_DATA*)arg;
522 WINPR_ASSERT(data->serial);
523 WINPR_ASSERT(data->irp);
526 if ((error = serial_process_irp(data->serial, data->irp)))
528 WLog_Print(data->serial->log, WLOG_ERROR,
529 "serial_process_irp failed with error %" PRIu32
"", error);
533 EnterCriticalSection(&data->serial->TerminatingIrpThreadsLock);
534 WINPR_ASSERT(data->irp->Complete);
535 error = data->irp->Complete(data->irp);
536 LeaveCriticalSection(&data->serial->TerminatingIrpThreadsLock);
539 if (error && data->serial->rdpcontext)
540 setChannelError(data->serial->rdpcontext, error,
"irp_thread_func reported an error");
543 data->irp->Discard(data->irp);
553static void close_unterminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
556 HANDLE self = _GetCurrentThread();
557 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
558 if (self == cirpThread)
559 WLog_Print(log, WLOG_DEBUG,
"Skipping termination of own IRP thread");
561 ListDictionary_Remove(list, (
void*)
id);
564static void close_terminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
568 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
571 const DWORD waitResult = WaitForSingleObject(cirpThread, 0);
573 if (waitResult == WAIT_OBJECT_0)
574 ListDictionary_Remove(list, (
void*)
id);
575 else if (waitResult != WAIT_TIMEOUT)
578 WLog_Print(log, WLOG_WARN,
"WaitForSingleObject, got an unexpected result=0x%" PRIX32
"\n",
583void close_terminated_irp_thread_handles(SERIAL_DEVICE* serial, BOOL forceClose)
585 WINPR_ASSERT(serial);
587 EnterCriticalSection(&serial->TerminatingIrpThreadsLock);
589 ULONG_PTR* ids = NULL;
590 const size_t nbIds = ListDictionary_GetKeys(serial->IrpThreads, &ids);
592 for (
size_t i = 0; i < nbIds; i++)
594 ULONG_PTR
id = ids[i];
596 close_unterminated_irp_thread(serial->IrpThreads, serial->log,
id);
598 close_terminated_irp_thread(serial->IrpThreads, serial->log,
id);
603 LeaveCriticalSection(&serial->TerminatingIrpThreadsLock);
606static void create_irp_thread(SERIAL_DEVICE* serial, IRP* irp)
608 IRP_THREAD_DATA* data = NULL;
609 HANDLE irpThread = NULL;
610 HANDLE previousIrpThread = NULL;
613 WINPR_ASSERT(serial);
616 close_terminated_irp_thread_handles(serial, FALSE);
628 key = irp->CompletionId + 1ull;
629 previousIrpThread = ListDictionary_GetItemValue(serial->IrpThreads, (
void*)key);
631 if (previousIrpThread)
634 WLog_Print(serial->log, WLOG_DEBUG,
635 "IRP recall: IRP with the CompletionId=%" PRIu32
" not yet completed!",
655 if (ListDictionary_Count(serial->IrpThreads) >= MAX_IRP_THREADS)
657 WLog_Print(serial->log, WLOG_WARN,
658 "Number of IRP threads threshold reached: %" PRIuz
", keep on anyway",
659 ListDictionary_Count(serial->IrpThreads));
669 data = (IRP_THREAD_DATA*)calloc(1,
sizeof(IRP_THREAD_DATA));
673 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP_THREAD_DATA.");
677 data->serial = serial;
680 irpThread = CreateThread(NULL, 0, irp_thread_func, (
void*)data, CREATE_SUSPENDED, NULL);
682 if (irpThread == INVALID_HANDLE_VALUE)
684 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP thread.");
688 key = irp->CompletionId + 1ull;
690 if (!ListDictionary_Add(serial->IrpThreads, (
void*)key, irpThread))
692 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_Add failed!");
696 ResumeThread(irpThread);
701 (void)CloseHandle(irpThread);
702 irp->IoStatus = STATUS_NO_MEMORY;
703 WINPR_ASSERT(irp->Complete);
708static DWORD WINAPI serial_thread_func(LPVOID arg)
711 wMessage message = { 0 };
712 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)arg;
713 UINT error = CHANNEL_RC_OK;
715 WINPR_ASSERT(serial);
719 if (!MessageQueue_Wait(serial->MainIrpQueue))
721 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Wait failed!");
722 error = ERROR_INTERNAL_ERROR;
726 if (!MessageQueue_Peek(serial->MainIrpQueue, &message, TRUE))
728 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Peek failed!");
729 error = ERROR_INTERNAL_ERROR;
733 if (message.id == WMQ_QUIT)
736 irp = (IRP*)message.wParam;
739 create_irp_thread(serial, irp);
742 ListDictionary_Clear(serial->IrpThreads);
743 if (error && serial->rdpcontext)
744 setChannelError(serial->rdpcontext, error,
"serial_thread_func reported an error");
755static UINT serial_irp_request(DEVICE* device, IRP* irp)
757 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
758 WINPR_ASSERT(irp != NULL);
759 WINPR_ASSERT(serial);
762 return CHANNEL_RC_OK;
769 if (!MessageQueue_Post(serial->MainIrpQueue, NULL, 0, (
void*)irp, NULL))
771 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Post failed!");
772 return ERROR_INTERNAL_ERROR;
775 return CHANNEL_RC_OK;
783static UINT serial_free(DEVICE* device)
786 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
788 return CHANNEL_RC_OK;
790 WLog_Print(serial->log, WLOG_DEBUG,
"freeing");
791 if (serial->MainIrpQueue)
792 MessageQueue_PostQuit(serial->MainIrpQueue, 0);
794 if (serial->MainThread)
796 if (WaitForSingleObject(serial->MainThread, INFINITE) == WAIT_FAILED)
798 error = GetLastError();
799 WLog_Print(serial->log, WLOG_ERROR,
800 "WaitForSingleObject failed with error %" PRIu32
"!", error);
802 (void)CloseHandle(serial->MainThread);
806 (void)CloseHandle(serial->hComm);
809 Stream_Free(serial->device.data, TRUE);
810 MessageQueue_Free(serial->MainIrpQueue);
811 ListDictionary_Free(serial->IrpThreads);
812 DeleteCriticalSection(&serial->TerminatingIrpThreadsLock);
814 return CHANNEL_RC_OK;
817static void serial_message_free(
void* obj)
825 IRP* irp = (IRP*)msg->wParam;
828 WINPR_ASSERT(irp->Discard);
832static void irp_thread_close(
void* arg)
837 HANDLE thz = _GetCurrentThread();
839 WLog_WARN(TAG,
"closing self, ignoring...");
842 (void)TerminateThread(hdl, 0);
843 (void)WaitForSingleObject(hdl, INFINITE);
844 (void)CloseHandle(hdl);
858 SERIAL_DEVICE* serial = NULL;
859 UINT error = CHANNEL_RC_OK;
861 WINPR_ASSERT(pEntryPoints);
864 WINPR_ASSERT(device);
866 wLog* log = WLog_Get(TAG);
867 const char* name = device->device.Name;
868 const char* path = device->Path;
869 const char* driver = device->Driver;
871 if (!name || (name[0] ==
'*'))
874 WLog_Print(log, WLOG_WARN,
875 "Serial port autodetection not implemented, nothing will be redirected!");
876 return CHANNEL_RC_OK;
879 if ((name && name[0]) && (path && path[0]))
881 WLog_Print(log, WLOG_DEBUG,
"Defining %s as %s", name, path);
883 if (!DefineCommDevice(name , path ))
885 DWORD status = GetLastError();
886 WLog_Print(log, WLOG_ERROR,
"DefineCommDevice failed with %08" PRIx32, status);
887 return ERROR_INTERNAL_ERROR;
890 serial = (SERIAL_DEVICE*)calloc(1,
sizeof(SERIAL_DEVICE));
894 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
895 return CHANNEL_RC_NO_MEMORY;
899 serial->device.type = RDPDR_DTYP_SERIAL;
900 serial->device.name = name;
901 serial->device.IRPRequest = serial_irp_request;
902 serial->device.Free = serial_free;
903 serial->rdpcontext = pEntryPoints->rdpcontext;
905 serial->device.data = Stream_New(NULL, len + 1);
907 if (!serial->device.data)
909 WLog_Print(serial->log, WLOG_ERROR,
"calloc failed!");
910 error = CHANNEL_RC_NO_MEMORY;
914 for (
size_t i = 0; i <= len; i++)
915 Stream_Write_INT8(serial->device.data, name[i] < 0 ?
'_' : name[i]);
919 if (_stricmp(driver,
"Serial") == 0)
920 serial->ServerSerialDriverId = SerialDriverSerialSys;
921 else if (_stricmp(driver,
"SerCx") == 0)
922 serial->ServerSerialDriverId = SerialDriverSerCxSys;
923 else if (_stricmp(driver,
"SerCx2") == 0)
924 serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
927 WLog_Print(serial->log, WLOG_WARN,
"Unknown server's serial driver: %s.", driver);
928 WLog_Print(serial->log, WLOG_WARN,
929 "Valid options are: 'Serial' (default), 'SerCx' and 'SerCx2'");
936 serial->ServerSerialDriverId = SerialDriverSerialSys;
939 if (device->Permissive != NULL)
941 if (_stricmp(device->Permissive,
"permissive") == 0)
943 serial->permissive = TRUE;
947 WLog_Print(serial->log, WLOG_WARN,
"Unknown flag: %s", device->Permissive);
952 WLog_Print(serial->log, WLOG_DEBUG,
"Server's serial driver: %s (id: %u)", driver,
953 serial->ServerSerialDriverId);
955 serial->MainIrpQueue = MessageQueue_New(NULL);
957 if (!serial->MainIrpQueue)
959 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_New failed!");
960 error = CHANNEL_RC_NO_MEMORY;
965 wObject* obj = MessageQueue_Object(serial->MainIrpQueue);
967 obj->fnObjectFree = serial_message_free;
971 serial->IrpThreads = ListDictionary_New(FALSE);
973 if (!serial->IrpThreads)
975 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_New failed!");
976 error = CHANNEL_RC_NO_MEMORY;
981 wObject* obj = ListDictionary_ValueObject(serial->IrpThreads);
983 obj->fnObjectFree = irp_thread_close;
986 InitializeCriticalSection(&serial->TerminatingIrpThreadsLock);
988 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &serial->device);
989 if (error != CHANNEL_RC_OK)
991 WLog_Print(serial->log, WLOG_ERROR,
992 "EntryPoints->RegisterDevice failed with error %" PRIu32
"!", error);
996 serial->MainThread = CreateThread(NULL, 0, serial_thread_func, serial, 0, NULL);
997 if (!serial->MainThread)
999 WLog_Print(serial->log, WLOG_ERROR,
"CreateThread failed!");
1000 error = ERROR_INTERNAL_ERROR;
1008 serial_free(&serial->device);
This struct contains function pointer to initialize/free objects.