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;
169 CreateFile(serial->device.name, DesiredAccess, SharedAccess, NULL,
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);
197 DWORD BytesReturned = 0;
198 if (!CommDeviceIoControl(serial->hComm, IOCTL_SERIAL_RESET_DEVICE, NULL, 0, NULL, 0,
199 &BytesReturned, NULL))
203 Stream_Write_UINT32(irp->output, irp->FileId);
204 Stream_Write_UINT8(irp->output, 0);
205 return CHANNEL_RC_OK;
208static UINT serial_process_irp_close(SERIAL_DEVICE* serial, IRP* irp)
210 WINPR_ASSERT(serial);
213 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
214 return ERROR_INVALID_DATA;
216 Stream_Seek(irp->input, 32);
218 close_terminated_irp_thread_handles(serial, TRUE);
220 if (!CloseHandle(serial->hComm))
222 WLog_Print(serial->log, WLOG_WARN,
"CloseHandle failure: %s (%" PRIu32
") closed.",
223 serial->device.name, irp->device->id);
224 irp->IoStatus = STATUS_UNSUCCESSFUL;
228 WLog_Print(serial->log, WLOG_DEBUG,
"%s (DeviceId: %" PRIu32
", FileId: %" PRIu32
") closed.",
229 serial->device.name, irp->device->id, irp->FileId);
230 irp->IoStatus = STATUS_SUCCESS;
232 serial->hComm = NULL;
233 Stream_Zero(irp->output, 5);
234 return CHANNEL_RC_OK;
242static UINT serial_process_irp_read(SERIAL_DEVICE* serial, IRP* irp)
249 WINPR_ASSERT(serial);
252 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
253 return ERROR_INVALID_DATA;
255 Stream_Read_UINT32(irp->input, Length);
256 Stream_Read_UINT64(irp->input, Offset);
259 Stream_Seek(irp->input, 20);
260 buffer = (BYTE*)calloc(Length,
sizeof(BYTE));
264 irp->IoStatus = STATUS_NO_MEMORY;
271 WLog_Print(serial->log, WLOG_DEBUG,
"reading %" PRIu32
" bytes from %s", Length,
272 serial->device.name);
275 if (CommReadFile(serial->hComm, buffer, Length, &nbRead, NULL))
277 irp->IoStatus = STATUS_SUCCESS;
281 WLog_Print(serial->log, WLOG_DEBUG,
282 "read failure to %s, nbRead=%" PRIu32
", last-error: 0x%08" PRIX32
"",
283 serial->device.name, nbRead, GetLastError());
284 irp->IoStatus = GetLastErrorToIoStatus(serial);
287 WLog_Print(serial->log, WLOG_DEBUG,
"%" PRIu32
" bytes read from %s", nbRead,
288 serial->device.name);
290 Stream_Write_UINT32(irp->output, nbRead);
294 if (!Stream_EnsureRemainingCapacity(irp->output, nbRead))
296 WLog_Print(serial->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
298 return CHANNEL_RC_NO_MEMORY;
301 Stream_Write(irp->output, buffer, nbRead);
305 return CHANNEL_RC_OK;
308static UINT serial_process_irp_write(SERIAL_DEVICE* serial, IRP* irp)
314 WINPR_ASSERT(serial);
317 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
318 return ERROR_INVALID_DATA;
320 Stream_Read_UINT32(irp->input, Length);
321 Stream_Read_UINT64(irp->input, Offset);
324 if (!Stream_SafeSeek(irp->input, 20))
325 return ERROR_INVALID_DATA;
333 WLog_Print(serial->log, WLOG_DEBUG,
"writing %" PRIu32
" bytes to %s", Length,
334 serial->device.name);
336 const void* ptr = Stream_ConstPointer(irp->input);
337 if (!Stream_SafeSeek(irp->input, Length))
338 return ERROR_INVALID_DATA;
340 if (CommWriteFile(serial->hComm, ptr, Length, &nbWritten, NULL))
342 irp->IoStatus = STATUS_SUCCESS;
346 WLog_Print(serial->log, WLOG_DEBUG,
347 "write failure to %s, nbWritten=%" PRIu32
", last-error: 0x%08" PRIX32
"",
348 serial->device.name, nbWritten, GetLastError());
349 irp->IoStatus = GetLastErrorToIoStatus(serial);
352 WLog_Print(serial->log, WLOG_DEBUG,
"%" PRIu32
" bytes written to %s", nbWritten,
353 serial->device.name);
354 Stream_Write_UINT32(irp->output, nbWritten);
355 Stream_Write_UINT8(irp->output, 0);
356 return CHANNEL_RC_OK;
364static UINT serial_process_irp_device_control(SERIAL_DEVICE* serial, IRP* irp)
366 UINT32 IoControlCode = 0;
367 UINT32 InputBufferLength = 0;
368 BYTE* InputBuffer = NULL;
369 UINT32 OutputBufferLength = 0;
370 BYTE* OutputBuffer = NULL;
371 DWORD BytesReturned = 0;
373 WINPR_ASSERT(serial);
376 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, 32))
377 return ERROR_INVALID_DATA;
379 Stream_Read_UINT32(irp->input, OutputBufferLength);
380 Stream_Read_UINT32(irp->input, InputBufferLength);
381 Stream_Read_UINT32(irp->input, IoControlCode);
382 Stream_Seek(irp->input, 20);
384 if (!Stream_CheckAndLogRequiredLengthWLog(serial->log, irp->input, InputBufferLength))
385 return ERROR_INVALID_DATA;
387 OutputBuffer = (BYTE*)calloc(OutputBufferLength,
sizeof(BYTE));
389 if (OutputBuffer == NULL)
391 irp->IoStatus = STATUS_NO_MEMORY;
395 InputBuffer = (BYTE*)calloc(InputBufferLength,
sizeof(BYTE));
397 if (InputBuffer == NULL)
399 irp->IoStatus = STATUS_NO_MEMORY;
403 Stream_Read(irp->input, InputBuffer, InputBufferLength);
404 WLog_Print(serial->log, WLOG_DEBUG,
405 "CommDeviceIoControl: CompletionId=%" PRIu32
", IoControlCode=[0x%" PRIX32
"] %s",
406 irp->CompletionId, IoControlCode, _comm_serial_ioctl_name(IoControlCode));
409 if (CommDeviceIoControl(serial->hComm, IoControlCode, InputBuffer, InputBufferLength,
410 OutputBuffer, OutputBufferLength, &BytesReturned, NULL))
415 irp->IoStatus = STATUS_SUCCESS;
419 WLog_Print(serial->log, WLOG_DEBUG,
420 "CommDeviceIoControl failure: IoControlCode=[0x%" PRIX32
421 "] %s, last-error: 0x%08" PRIX32
"",
422 IoControlCode, _comm_serial_ioctl_name(IoControlCode), GetLastError());
423 irp->IoStatus = GetLastErrorToIoStatus(serial);
430 WINPR_ASSERT(OutputBufferLength == BytesReturned);
431 Stream_Write_UINT32(irp->output, BytesReturned);
433 if (BytesReturned > 0)
435 if (!Stream_EnsureRemainingCapacity(irp->output, BytesReturned))
437 WLog_Print(serial->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
440 return CHANNEL_RC_NO_MEMORY;
443 Stream_Write(irp->output, OutputBuffer, BytesReturned);
456 return CHANNEL_RC_OK;
464static UINT serial_process_irp(SERIAL_DEVICE* serial, IRP* irp)
466 UINT error = CHANNEL_RC_OK;
468 WINPR_ASSERT(serial);
471 WLog_Print(serial->log, WLOG_DEBUG,
"IRP MajorFunction: %s, MinorFunction: 0x%08" PRIX32
"\n",
472 rdpdr_irp_string(irp->MajorFunction), irp->MinorFunction);
474 switch (irp->MajorFunction)
477 error = serial_process_irp_create(serial, irp);
481 error = serial_process_irp_close(serial, irp);
485 error = serial_process_irp_read(serial, irp);
489 error = serial_process_irp_write(serial, irp);
492 case IRP_MJ_DEVICE_CONTROL:
493 error = serial_process_irp_device_control(serial, irp);
497 irp->IoStatus = STATUS_NOT_SUPPORTED;
501 DWORD level = WLOG_TRACE;
505 WLog_Print(serial->log, level,
506 "[%s|0x%08" PRIx32
"] completed with %s [0x%08" PRIx32
"] (IoStatus %s [0x%08" PRIx32
508 rdpdr_irp_string(irp->MajorFunction), irp->MajorFunction, WTSErrorToString(error),
509 error, NtStatus2Tag(irp->IoStatus), irp->IoStatus);
514static DWORD WINAPI irp_thread_func(LPVOID arg)
516 IRP_THREAD_DATA* data = (IRP_THREAD_DATA*)arg;
520 WINPR_ASSERT(data->serial);
521 WINPR_ASSERT(data->irp);
524 if ((error = serial_process_irp(data->serial, data->irp)))
526 WLog_Print(data->serial->log, WLOG_ERROR,
527 "serial_process_irp failed with error %" PRIu32
"", error);
531 EnterCriticalSection(&data->serial->TerminatingIrpThreadsLock);
532 WINPR_ASSERT(data->irp->Complete);
533 error = data->irp->Complete(data->irp);
534 LeaveCriticalSection(&data->serial->TerminatingIrpThreadsLock);
537 if (error && data->serial->rdpcontext)
538 setChannelError(data->serial->rdpcontext, error,
"irp_thread_func reported an error");
541 data->irp->Discard(data->irp);
551static void close_unterminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
554 HANDLE self = _GetCurrentThread();
555 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
556 if (self == cirpThread)
557 WLog_Print(log, WLOG_DEBUG,
"Skipping termination of own IRP thread");
559 ListDictionary_Remove(list, (
void*)
id);
562static void close_terminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
566 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
569 const DWORD waitResult = WaitForSingleObject(cirpThread, 0);
571 if (waitResult == WAIT_OBJECT_0)
572 ListDictionary_Remove(list, (
void*)
id);
573 else if (waitResult != WAIT_TIMEOUT)
576 WLog_Print(log, WLOG_WARN,
"WaitForSingleObject, got an unexpected result=0x%" PRIX32
"\n",
581void close_terminated_irp_thread_handles(SERIAL_DEVICE* serial, BOOL forceClose)
583 WINPR_ASSERT(serial);
585 EnterCriticalSection(&serial->TerminatingIrpThreadsLock);
587 ULONG_PTR* ids = NULL;
588 const size_t nbIds = ListDictionary_GetKeys(serial->IrpThreads, &ids);
590 for (
size_t i = 0; i < nbIds; i++)
592 ULONG_PTR
id = ids[i];
594 close_unterminated_irp_thread(serial->IrpThreads, serial->log,
id);
596 close_terminated_irp_thread(serial->IrpThreads, serial->log,
id);
601 LeaveCriticalSection(&serial->TerminatingIrpThreadsLock);
604static void create_irp_thread(SERIAL_DEVICE* serial, IRP* irp)
606 IRP_THREAD_DATA* data = NULL;
607 HANDLE irpThread = NULL;
608 HANDLE previousIrpThread = NULL;
611 WINPR_ASSERT(serial);
614 close_terminated_irp_thread_handles(serial, FALSE);
626 key = irp->CompletionId + 1ull;
627 previousIrpThread = ListDictionary_GetItemValue(serial->IrpThreads, (
void*)key);
629 if (previousIrpThread)
632 WLog_Print(serial->log, WLOG_DEBUG,
633 "IRP recall: IRP with the CompletionId=%" PRIu32
" not yet completed!",
653 if (ListDictionary_Count(serial->IrpThreads) >= MAX_IRP_THREADS)
655 WLog_Print(serial->log, WLOG_WARN,
656 "Number of IRP threads threshold reached: %" PRIuz
", keep on anyway",
657 ListDictionary_Count(serial->IrpThreads));
667 data = (IRP_THREAD_DATA*)calloc(1,
sizeof(IRP_THREAD_DATA));
671 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP_THREAD_DATA.");
675 data->serial = serial;
678 irpThread = CreateThread(NULL, 0, irp_thread_func, (
void*)data, CREATE_SUSPENDED, NULL);
680 if (irpThread == INVALID_HANDLE_VALUE)
682 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP thread.");
686 key = irp->CompletionId + 1ull;
688 if (!ListDictionary_Add(serial->IrpThreads, (
void*)key, irpThread))
690 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_Add failed!");
694 ResumeThread(irpThread);
699 (void)CloseHandle(irpThread);
700 irp->IoStatus = STATUS_NO_MEMORY;
701 WINPR_ASSERT(irp->Complete);
706static DWORD WINAPI serial_thread_func(LPVOID arg)
709 wMessage message = { 0 };
710 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)arg;
711 UINT error = CHANNEL_RC_OK;
713 WINPR_ASSERT(serial);
717 if (!MessageQueue_Wait(serial->MainIrpQueue))
719 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Wait failed!");
720 error = ERROR_INTERNAL_ERROR;
724 if (!MessageQueue_Peek(serial->MainIrpQueue, &message, TRUE))
726 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Peek failed!");
727 error = ERROR_INTERNAL_ERROR;
731 if (message.id == WMQ_QUIT)
734 irp = (IRP*)message.wParam;
737 create_irp_thread(serial, irp);
740 ListDictionary_Clear(serial->IrpThreads);
741 if (error && serial->rdpcontext)
742 setChannelError(serial->rdpcontext, error,
"serial_thread_func reported an error");
753static UINT serial_irp_request(DEVICE* device, IRP* irp)
755 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
756 WINPR_ASSERT(irp != NULL);
757 WINPR_ASSERT(serial);
760 return CHANNEL_RC_OK;
767 if (!MessageQueue_Post(serial->MainIrpQueue, NULL, 0, (
void*)irp, NULL))
769 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Post failed!");
770 return ERROR_INTERNAL_ERROR;
773 return CHANNEL_RC_OK;
781static UINT serial_free(DEVICE* device)
784 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
786 return CHANNEL_RC_OK;
788 WLog_Print(serial->log, WLOG_DEBUG,
"freeing");
789 if (serial->MainIrpQueue)
790 MessageQueue_PostQuit(serial->MainIrpQueue, 0);
792 if (serial->MainThread)
794 if (WaitForSingleObject(serial->MainThread, INFINITE) == WAIT_FAILED)
796 error = GetLastError();
797 WLog_Print(serial->log, WLOG_ERROR,
798 "WaitForSingleObject failed with error %" PRIu32
"!", error);
800 (void)CloseHandle(serial->MainThread);
804 (void)CloseHandle(serial->hComm);
807 Stream_Free(serial->device.data, TRUE);
808 MessageQueue_Free(serial->MainIrpQueue);
809 ListDictionary_Free(serial->IrpThreads);
810 DeleteCriticalSection(&serial->TerminatingIrpThreadsLock);
812 return CHANNEL_RC_OK;
815static void serial_message_free(
void* obj)
823 IRP* irp = (IRP*)msg->wParam;
826 WINPR_ASSERT(irp->Discard);
830static void irp_thread_close(
void* arg)
835 HANDLE thz = _GetCurrentThread();
837 WLog_WARN(TAG,
"closing self, ignoring...");
840 (void)TerminateThread(hdl, 0);
841 (void)WaitForSingleObject(hdl, INFINITE);
842 (void)CloseHandle(hdl);
856 SERIAL_DEVICE* serial = NULL;
857 UINT error = CHANNEL_RC_OK;
859 WINPR_ASSERT(pEntryPoints);
862 WINPR_ASSERT(device);
864 wLog* log = WLog_Get(TAG);
865 const char* name = device->device.Name;
866 const char* path = device->Path;
867 const char* driver = device->Driver;
869 if (!name || (name[0] ==
'*'))
872 WLog_Print(log, WLOG_WARN,
873 "Serial port autodetection not implemented, nothing will be redirected!");
874 return CHANNEL_RC_OK;
877 if ((name && name[0]) && (path && path[0]))
879 WLog_Print(log, WLOG_DEBUG,
"Defining %s as %s", name, path);
881 if (!DefineCommDevice(name , path ))
883 DWORD status = GetLastError();
884 WLog_Print(log, WLOG_ERROR,
"DefineCommDevice failed with %08" PRIx32, status);
885 return ERROR_INTERNAL_ERROR;
888 serial = (SERIAL_DEVICE*)calloc(1,
sizeof(SERIAL_DEVICE));
892 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
893 return CHANNEL_RC_NO_MEMORY;
897 serial->device.type = RDPDR_DTYP_SERIAL;
898 serial->device.name = name;
899 serial->device.IRPRequest = serial_irp_request;
900 serial->device.Free = serial_free;
901 serial->rdpcontext = pEntryPoints->rdpcontext;
903 serial->device.data = Stream_New(NULL, len + 1);
905 if (!serial->device.data)
907 WLog_Print(serial->log, WLOG_ERROR,
"calloc failed!");
908 error = CHANNEL_RC_NO_MEMORY;
912 for (
size_t i = 0; i <= len; i++)
913 Stream_Write_INT8(serial->device.data, name[i] < 0 ?
'_' : name[i]);
917 if (_stricmp(driver,
"Serial") == 0)
918 serial->ServerSerialDriverId = SerialDriverSerialSys;
919 else if (_stricmp(driver,
"SerCx") == 0)
920 serial->ServerSerialDriverId = SerialDriverSerCxSys;
921 else if (_stricmp(driver,
"SerCx2") == 0)
922 serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
925 WLog_Print(serial->log, WLOG_WARN,
"Unknown server's serial driver: %s.", driver);
926 WLog_Print(serial->log, WLOG_WARN,
927 "Valid options are: 'Serial' (default), 'SerCx' and 'SerCx2'");
934 serial->ServerSerialDriverId = SerialDriverSerialSys;
937 if (device->Permissive != NULL)
939 if (_stricmp(device->Permissive,
"permissive") == 0)
941 serial->permissive = TRUE;
945 WLog_Print(serial->log, WLOG_WARN,
"Unknown flag: %s", device->Permissive);
950 WLog_Print(serial->log, WLOG_DEBUG,
"Server's serial driver: %s (id: %d)", driver,
951 serial->ServerSerialDriverId);
953 serial->MainIrpQueue = MessageQueue_New(NULL);
955 if (!serial->MainIrpQueue)
957 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_New failed!");
958 error = CHANNEL_RC_NO_MEMORY;
963 wObject* obj = MessageQueue_Object(serial->MainIrpQueue);
965 obj->fnObjectFree = serial_message_free;
969 serial->IrpThreads = ListDictionary_New(FALSE);
971 if (!serial->IrpThreads)
973 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_New failed!");
974 error = CHANNEL_RC_NO_MEMORY;
979 wObject* obj = ListDictionary_ValueObject(serial->IrpThreads);
981 obj->fnObjectFree = irp_thread_close;
984 InitializeCriticalSection(&serial->TerminatingIrpThreadsLock);
986 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &serial->device);
987 if (error != CHANNEL_RC_OK)
989 WLog_Print(serial->log, WLOG_ERROR,
990 "EntryPoints->RegisterDevice failed with error %" PRIu32
"!", error);
994 serial->MainThread = CreateThread(NULL, 0, serial_thread_func, serial, 0, NULL);
995 if (!serial->MainThread)
997 WLog_Print(serial->log, WLOG_ERROR,
"CreateThread failed!");
998 error = ERROR_INTERNAL_ERROR;
1006 serial_free(&serial->device);
This struct contains function pointer to initialize/free objects.