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;
521 WINPR_ASSERT(data->serial);
522 WINPR_ASSERT(data->irp);
525 UINT 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);
530 data->irp->Discard(data->irp);
534 EnterCriticalSection(&data->serial->TerminatingIrpThreadsLock);
535 WINPR_ASSERT(data->irp->Complete);
536 error = data->irp->Complete(data->irp);
537 LeaveCriticalSection(&data->serial->TerminatingIrpThreadsLock);
540 if (error && data->serial->rdpcontext)
541 setChannelError(data->serial->rdpcontext, error,
"irp_thread_func reported an error");
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 ListDictionary_Lock(serial->IrpThreads);
588 ULONG_PTR* ids = NULL;
589 const size_t nbIds = ListDictionary_GetKeys(serial->IrpThreads, &ids);
591 for (
size_t i = 0; i < nbIds; i++)
593 ULONG_PTR
id = ids[i];
595 close_unterminated_irp_thread(serial->IrpThreads, serial->log,
id);
597 close_terminated_irp_thread(serial->IrpThreads, serial->log,
id);
601 ListDictionary_Unlock(serial->IrpThreads);
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;
630 ListDictionary_Lock(serial->IrpThreads);
631 previousIrpThread = ListDictionary_GetItemValue(serial->IrpThreads, (
void*)key);
632 ListDictionary_Unlock(serial->IrpThreads);
634 if (previousIrpThread)
637 WLog_Print(serial->log, WLOG_DEBUG,
638 "IRP recall: IRP with the CompletionId=%" PRIu32
" not yet completed!",
659 data = (IRP_THREAD_DATA*)calloc(1,
sizeof(IRP_THREAD_DATA));
663 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP_THREAD_DATA.");
667 data->serial = serial;
670 irpThread = CreateThread(NULL, 0, irp_thread_func, (
void*)data, CREATE_SUSPENDED, NULL);
672 if (irpThread == INVALID_HANDLE_VALUE)
674 WLog_Print(serial->log, WLOG_WARN,
"Could not allocate a new IRP thread.");
678 key = irp->CompletionId + 1ull;
680 ListDictionary_Lock(serial->IrpThreads);
681 if (ListDictionary_Count(serial->IrpThreads) >= MAX_IRP_THREADS)
683 WLog_Print(serial->log, WLOG_WARN,
684 "Number of IRP threads threshold reached: %" PRIuz
", keep on anyway",
685 ListDictionary_Count(serial->IrpThreads));
695 const BOOL added = ListDictionary_Add(serial->IrpThreads, (
void*)key, irpThread);
696 ListDictionary_Unlock(serial->IrpThreads);
700 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_Add failed!");
705 ResumeThread(irpThread);
710 (void)CloseHandle(irpThread);
711 irp->IoStatus = STATUS_NO_MEMORY;
712 WINPR_ASSERT(irp->Complete);
717static DWORD WINAPI serial_thread_func(LPVOID arg)
720 wMessage message = { 0 };
721 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)arg;
722 UINT error = CHANNEL_RC_OK;
724 WINPR_ASSERT(serial);
728 if (!MessageQueue_Wait(serial->MainIrpQueue))
730 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Wait failed!");
731 error = ERROR_INTERNAL_ERROR;
735 if (!MessageQueue_Peek(serial->MainIrpQueue, &message, TRUE))
737 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Peek failed!");
738 error = ERROR_INTERNAL_ERROR;
742 if (message.id == WMQ_QUIT)
745 irp = (IRP*)message.wParam;
748 create_irp_thread(serial, irp);
751 ListDictionary_Lock(serial->IrpThreads);
752 ListDictionary_Clear(serial->IrpThreads);
753 ListDictionary_Unlock(serial->IrpThreads);
755 if (error && serial->rdpcontext)
756 setChannelError(serial->rdpcontext, error,
"serial_thread_func reported an error");
767static UINT serial_irp_request(DEVICE* device, IRP* irp)
769 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
770 WINPR_ASSERT(irp != NULL);
771 WINPR_ASSERT(serial);
774 return CHANNEL_RC_OK;
781 if (!MessageQueue_Post(serial->MainIrpQueue, NULL, 0, (
void*)irp, NULL))
783 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_Post failed!");
784 return ERROR_INTERNAL_ERROR;
787 return CHANNEL_RC_OK;
795static UINT serial_free(DEVICE* device)
798 SERIAL_DEVICE* serial = (SERIAL_DEVICE*)device;
800 return CHANNEL_RC_OK;
802 WLog_Print(serial->log, WLOG_DEBUG,
"freeing");
803 if (serial->MainIrpQueue)
804 MessageQueue_PostQuit(serial->MainIrpQueue, 0);
806 if (serial->MainThread)
808 if (WaitForSingleObject(serial->MainThread, INFINITE) == WAIT_FAILED)
810 error = GetLastError();
811 WLog_Print(serial->log, WLOG_ERROR,
812 "WaitForSingleObject failed with error %" PRIu32
"!", error);
814 (void)CloseHandle(serial->MainThread);
818 (void)CloseHandle(serial->hComm);
821 Stream_Free(serial->device.data, TRUE);
822 MessageQueue_Free(serial->MainIrpQueue);
823 ListDictionary_Free(serial->IrpThreads);
824 DeleteCriticalSection(&serial->TerminatingIrpThreadsLock);
826 return CHANNEL_RC_OK;
829static void serial_message_free(
void* obj)
837 IRP* irp = (IRP*)msg->wParam;
840 WINPR_ASSERT(irp->Discard);
844static void irp_thread_close(
void* arg)
849 HANDLE thz = _GetCurrentThread();
851 WLog_WARN(TAG,
"closing self, ignoring...");
854 (void)TerminateThread(hdl, 0);
855 (void)WaitForSingleObject(hdl, INFINITE);
856 (void)CloseHandle(hdl);
870 SERIAL_DEVICE* serial = NULL;
871 UINT error = CHANNEL_RC_OK;
873 WINPR_ASSERT(pEntryPoints);
876 WINPR_ASSERT(device);
878 wLog* log = WLog_Get(TAG);
879 const char* name = device->device.Name;
880 const char* path = device->Path;
881 const char* driver = device->Driver;
883 if (!name || (name[0] ==
'*'))
886 WLog_Print(log, WLOG_WARN,
887 "Serial port autodetection not implemented, nothing will be redirected!");
888 return CHANNEL_RC_OK;
891 if ((name && name[0]) && (path && path[0]))
893 WLog_Print(log, WLOG_DEBUG,
"Defining %s as %s", name, path);
895 if (!DefineCommDevice(name , path ))
897 DWORD status = GetLastError();
898 WLog_Print(log, WLOG_ERROR,
"DefineCommDevice failed with %08" PRIx32, status);
899 return ERROR_INTERNAL_ERROR;
902 serial = (SERIAL_DEVICE*)calloc(1,
sizeof(SERIAL_DEVICE));
906 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
907 return CHANNEL_RC_NO_MEMORY;
911 serial->device.type = RDPDR_DTYP_SERIAL;
912 serial->device.name = name;
913 serial->device.IRPRequest = serial_irp_request;
914 serial->device.Free = serial_free;
915 serial->rdpcontext = pEntryPoints->rdpcontext;
917 serial->device.data = Stream_New(NULL, len + 1);
919 if (!serial->device.data)
921 WLog_Print(serial->log, WLOG_ERROR,
"calloc failed!");
922 error = CHANNEL_RC_NO_MEMORY;
926 for (
size_t i = 0; i <= len; i++)
927 Stream_Write_INT8(serial->device.data, name[i] < 0 ?
'_' : name[i]);
931 if (_stricmp(driver,
"Serial") == 0)
932 serial->ServerSerialDriverId = SerialDriverSerialSys;
933 else if (_stricmp(driver,
"SerCx") == 0)
934 serial->ServerSerialDriverId = SerialDriverSerCxSys;
935 else if (_stricmp(driver,
"SerCx2") == 0)
936 serial->ServerSerialDriverId = SerialDriverSerCx2Sys;
939 WLog_Print(serial->log, WLOG_WARN,
"Unknown server's serial driver: %s.", driver);
940 WLog_Print(serial->log, WLOG_WARN,
941 "Valid options are: 'Serial' (default), 'SerCx' and 'SerCx2'");
948 serial->ServerSerialDriverId = SerialDriverSerialSys;
951 if (device->Permissive != NULL)
953 if (_stricmp(device->Permissive,
"permissive") == 0)
955 serial->permissive = TRUE;
959 WLog_Print(serial->log, WLOG_WARN,
"Unknown flag: %s", device->Permissive);
964 WLog_Print(serial->log, WLOG_DEBUG,
"Server's serial driver: %s (id: %u)", driver,
965 serial->ServerSerialDriverId);
967 serial->MainIrpQueue = MessageQueue_New(NULL);
969 if (!serial->MainIrpQueue)
971 WLog_Print(serial->log, WLOG_ERROR,
"MessageQueue_New failed!");
972 error = CHANNEL_RC_NO_MEMORY;
977 wObject* obj = MessageQueue_Object(serial->MainIrpQueue);
983 serial->IrpThreads = ListDictionary_New(FALSE);
985 if (!serial->IrpThreads)
987 WLog_Print(serial->log, WLOG_ERROR,
"ListDictionary_New failed!");
988 error = CHANNEL_RC_NO_MEMORY;
993 wObject* obj = ListDictionary_ValueObject(serial->IrpThreads);
998 InitializeCriticalSection(&serial->TerminatingIrpThreadsLock);
1000 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &serial->device);
1001 if (error != CHANNEL_RC_OK)
1003 WLog_Print(serial->log, WLOG_ERROR,
1004 "EntryPoints->RegisterDevice failed with error %" PRIu32
"!", error);
1008 serial->MainThread = CreateThread(NULL, 0, serial_thread_func, serial, 0, NULL);
1009 if (!serial->MainThread)
1011 WLog_Print(serial->log, WLOG_ERROR,
"CreateThread failed!");
1012 error = ERROR_INTERNAL_ERROR;
1020 serial_free(&serial->device);
This struct contains function pointer to initialize/free objects.
OBJECT_FREE_FN fnObjectFree