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 error = data->irp->Complete(data->irp);
533 LeaveCriticalSection(&data->serial->TerminatingIrpThreadsLock);
536 if (error && data->serial->rdpcontext)
537 setChannelError(data->serial->rdpcontext, error,
"irp_thread_func reported an error");
540 data->irp->Discard(data->irp);
550static void close_unterminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
553 HANDLE self = _GetCurrentThread();
554 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
555 if (self == cirpThread)
556 WLog_Print(log, WLOG_DEBUG,
"Skipping termination of own IRP thread");
558 ListDictionary_Remove(list, (
void*)
id);
561static void close_terminated_irp_thread(wListDictionary* list, wLog* log, ULONG_PTR
id)
565 HANDLE cirpThread = ListDictionary_GetItemValue(list, (
void*)
id);
568 const DWORD waitResult = WaitForSingleObject(cirpThread, 0);
570 if (waitResult == WAIT_OBJECT_0)
571 ListDictionary_Remove(list, (
void*)
id);
572 else if (waitResult != WAIT_TIMEOUT)
575 WLog_Print(log, WLOG_WARN,
"WaitForSingleObject, got an unexpected result=0x%" PRIX32
"\n",
580void close_terminated_irp_thread_handles(SERIAL_DEVICE* serial, BOOL forceClose)
582 WINPR_ASSERT(serial);
584 EnterCriticalSection(&serial->TerminatingIrpThreadsLock);
586 ULONG_PTR* ids = NULL;
587 const size_t nbIds = ListDictionary_GetKeys(serial->IrpThreads, &ids);
589 for (
size_t i = 0; i < nbIds; i++)
591 ULONG_PTR
id = ids[i];
593 close_unterminated_irp_thread(serial->IrpThreads, serial->log,
id);
595 close_terminated_irp_thread(serial->IrpThreads, serial->log,
id);
600 LeaveCriticalSection(&serial->TerminatingIrpThreadsLock);
603static void create_irp_thread(SERIAL_DEVICE* serial, IRP* irp)
605 IRP_THREAD_DATA* data = NULL;
606 HANDLE irpThread = NULL;
607 HANDLE previousIrpThread = NULL;
610 WINPR_ASSERT(serial);
613 close_terminated_irp_thread_handles(serial, FALSE);
625 key = irp->CompletionId + 1ull;
626 previousIrpThread = ListDictionary_GetItemValue(serial->IrpThreads, (
void*)key);
628 if (previousIrpThread)
631 WLog_Print(serial->log, WLOG_DEBUG,
632 "IRP recall: IRP with the CompletionId=%" PRIu32
" not yet completed!",
652 if (ListDictionary_Count(serial->IrpThreads) >= MAX_IRP_THREADS)
654 WLog_Print(serial->log, WLOG_WARN,
655 "Number of IRP threads threshold reached: %" PRIuz
", keep on anyway",
656 ListDictionary_Count(serial->IrpThreads));
666 data = (IRP_THREAD_DATA*)calloc(1,
sizeof(IRP_THREAD_DATA));
670 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP_THREAD_DATA.");
674 data->serial = serial;
677 irpThread = CreateThread(NULL, 0, irp_thread_func, (
void*)data, CREATE_SUSPENDED, NULL);
679 if (irpThread == INVALID_HANDLE_VALUE)
681 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP thread.");
685 key = irp->CompletionId + 1ull;
687 if (!ListDictionary_Add(serial->IrpThreads, (
void*)key, irpThread))
689 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_Add failed!");
693 ResumeThread(irpThread);
698 (void)CloseHandle(irpThread);
699 irp->IoStatus = STATUS_NO_MEMORY;
704static DWORD WINAPI serial_thread_func(LPVOID arg)
707 wMessage message = { 0 };
708 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)arg;
709 UINT error = CHANNEL_RC_OK;
711 WINPR_ASSERT(serial);
715 if (!MessageQueue_Wait(serial->MainIrpQueue))
717 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Wait failed!");
718 error = ERROR_INTERNAL_ERROR;
722 if (!MessageQueue_Peek(serial->MainIrpQueue, &message, TRUE))
724 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Peek failed!");
725 error = ERROR_INTERNAL_ERROR;
729 if (message.id == WMQ_QUIT)
732 irp = (IRP*)message.wParam;
735 create_irp_thread(serial, irp);
738 ListDictionary_Clear(serial->IrpThreads);
739 if (error && serial->rdpcontext)
740 setChannelError(serial->rdpcontext, error,
"serial_thread_func reported an error");
751static UINT serial_irp_request(DEVICE* device, IRP* irp)
753 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
754 WINPR_ASSERT(irp != NULL);
755 WINPR_ASSERT(serial);
758 return CHANNEL_RC_OK;
765 if (!MessageQueue_Post(serial->MainIrpQueue, NULL, 0, (
void*)irp, NULL))
767 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Post failed!");
768 return ERROR_INTERNAL_ERROR;
771 return CHANNEL_RC_OK;
779static UINT serial_free(DEVICE* device)
782 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
784 return CHANNEL_RC_OK;
786 WLog_Print(serial->log, WLOG_DEBUG,
"freeing");
787 if (serial->MainIrpQueue)
788 MessageQueue_PostQuit(serial->MainIrpQueue, 0);
790 if (serial->MainThread)
792 if (WaitForSingleObject(serial->MainThread, INFINITE) == WAIT_FAILED)
794 error = GetLastError();
795 WLog_Print(serial->log, WLOG_ERROR,
796 "WaitForSingleObject failed with error %" PRIu32
"!", error);
798 (void)CloseHandle(serial->MainThread);
802 (void)CloseHandle(serial->hComm);
805 Stream_Free(serial->device.data, TRUE);
806 MessageQueue_Free(serial->MainIrpQueue);
807 ListDictionary_Free(serial->IrpThreads);
808 DeleteCriticalSection(&serial->TerminatingIrpThreadsLock);
810 return CHANNEL_RC_OK;
813static void serial_message_free(
void* obj)
821 IRP* irp = (IRP*)msg->wParam;
824 WINPR_ASSERT(irp->Discard);
828static void irp_thread_close(
void* arg)
833 HANDLE thz = _GetCurrentThread();
835 WLog_WARN(TAG,
"closing self, ignoring...");
838 (void)TerminateThread(hdl, 0);
839 (void)WaitForSingleObject(hdl, INFINITE);
840 (void)CloseHandle(hdl);
854 SERIAL_DEVICE* serial = NULL;
855 UINT error = CHANNEL_RC_OK;
857 WINPR_ASSERT(pEntryPoints);
860 WINPR_ASSERT(device);
862 wLog* log = WLog_Get(TAG);
863 const char* name = device->device.Name;
864 const char* path = device->Path;
865 const char* driver = device->Driver;
867 if (!name || (name[0] ==
'*'))
870 WLog_Print(log, WLOG_WARN,
871 "Serial port autodetection not implemented, nothing will be redirected!");
872 return CHANNEL_RC_OK;
875 if ((name && name[0]) && (path && path[0]))
877 WLog_Print(log, WLOG_DEBUG,
"Defining %s as %s", name, path);
879 if (!DefineCommDevice(name , path ))
881 DWORD status = GetLastError();
882 WLog_Print(log, WLOG_ERROR,
"DefineCommDevice failed with %08" PRIx32, status);
883 return ERROR_INTERNAL_ERROR;
886 serial = (SERIAL_DEVICE*)calloc(1,
sizeof(SERIAL_DEVICE));
890 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
891 return CHANNEL_RC_NO_MEMORY;
895 serial->device.type = RDPDR_DTYP_SERIAL;
896 serial->device.name = name;
897 serial->device.IRPRequest = serial_irp_request;
898 serial->device.Free = serial_free;
899 serial->rdpcontext = pEntryPoints->rdpcontext;
901 serial->device.data = Stream_New(NULL, len + 1);
903 if (!serial->device.data)
905 WLog_Print(serial->log, WLOG_ERROR,
"calloc failed!");
906 error = CHANNEL_RC_NO_MEMORY;
910 for (
size_t i = 0; i <= len; i++)
911 Stream_Write_INT8(serial->device.data, name[i] < 0 ?
'_' : name[i]);
915 if (_stricmp(driver,
"Serial") == 0)
916 serial->ServerSerialDriverId = SerialDriverSerialSys;
917 else if (_stricmp(driver,
"SerCx") == 0)
918 serial->ServerSerialDriverId = SerialDriverSerCxSys;
919 else if (_stricmp(driver,
"SerCx2") == 0)
920 serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
923 WLog_Print(serial->log, WLOG_WARN,
"Unknown server's serial driver: %s.", driver);
924 WLog_Print(serial->log, WLOG_WARN,
925 "Valid options are: 'Serial' (default), 'SerCx' and 'SerCx2'");
932 serial->ServerSerialDriverId = SerialDriverSerialSys;
935 if (device->Permissive != NULL)
937 if (_stricmp(device->Permissive,
"permissive") == 0)
939 serial->permissive = TRUE;
943 WLog_Print(serial->log, WLOG_WARN,
"Unknown flag: %s", device->Permissive);
948 WLog_Print(serial->log, WLOG_DEBUG,
"Server's serial driver: %s (id: %d)", driver,
949 serial->ServerSerialDriverId);
951 serial->MainIrpQueue = MessageQueue_New(NULL);
953 if (!serial->MainIrpQueue)
955 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_New failed!");
956 error = CHANNEL_RC_NO_MEMORY;
961 wObject* obj = MessageQueue_Object(serial->MainIrpQueue);
963 obj->fnObjectFree = serial_message_free;
967 serial->IrpThreads = ListDictionary_New(FALSE);
969 if (!serial->IrpThreads)
971 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_New failed!");
972 error = CHANNEL_RC_NO_MEMORY;
977 wObject* obj = ListDictionary_ValueObject(serial->IrpThreads);
979 obj->fnObjectFree = irp_thread_close;
982 InitializeCriticalSection(&serial->TerminatingIrpThreadsLock);
984 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &serial->device);
985 if (error != CHANNEL_RC_OK)
987 WLog_Print(serial->log, WLOG_ERROR,
988 "EntryPoints->RegisterDevice failed with error %" PRIu32
"!", error);
992 serial->MainThread = CreateThread(NULL, 0, serial_thread_func, serial, 0, NULL);
993 if (!serial->MainThread)
995 WLog_Print(serial->log, WLOG_ERROR,
"CreateThread failed!");
996 error = ERROR_INTERNAL_ERROR;
1004 serial_free(&serial->device);
This struct contains function pointer to initialize/free objects.