mirror of
https://github.com/reactos/reactos.git
synced 2025-01-11 00:30:58 +00:00
912ce51ae6
Sync with trunk head (r48826) svn path=/branches/cmake-bringup/; revision=48831
162 lines
4.5 KiB
C
162 lines
4.5 KiB
C
/*
|
||
* COPYRIGHT: See COPYING in the top level directory
|
||
* PROJECT: USB hub driver
|
||
* FILE: drivers/usb/cromwell/hub/misc.c
|
||
* PURPOSE: Misceallenous operations
|
||
*
|
||
* PROGRAMMERS: Herv<72> Poussineau (hpoussin@reactos.com),
|
||
*/
|
||
|
||
#define NDEBUG
|
||
#include "usbhub.h"
|
||
#include <stdarg.h>
|
||
|
||
NTSTATUS NTAPI
|
||
ForwardIrpAndWaitCompletion(
|
||
IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp,
|
||
IN PVOID Context)
|
||
{
|
||
if (Irp->PendingReturned)
|
||
KeSetEvent((PKEVENT)Context, IO_NO_INCREMENT, FALSE);
|
||
return STATUS_MORE_PROCESSING_REQUIRED;
|
||
}
|
||
|
||
NTSTATUS
|
||
ForwardIrpAndWait(
|
||
IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp)
|
||
{
|
||
KEVENT Event;
|
||
NTSTATUS Status;
|
||
|
||
KeInitializeEvent(&Event, NotificationEvent, FALSE);
|
||
IoCopyCurrentIrpStackLocationToNext(Irp);
|
||
|
||
IoSetCompletionRoutine(Irp, ForwardIrpAndWaitCompletion, &Event, TRUE, TRUE, TRUE);
|
||
|
||
Status = IoCallDriver(DeviceObject, Irp);
|
||
if (Status == STATUS_PENDING)
|
||
{
|
||
Status = KeWaitForSingleObject(&Event, Suspended, KernelMode, FALSE, NULL);
|
||
if (NT_SUCCESS(Status))
|
||
Status = Irp->IoStatus.Status;
|
||
}
|
||
|
||
return Status;
|
||
}
|
||
|
||
NTSTATUS NTAPI
|
||
ForwardIrpAndForget(
|
||
IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp)
|
||
{
|
||
PDEVICE_OBJECT LowerDevice = ((PHUB_DEVICE_EXTENSION)DeviceObject->DeviceExtension)->LowerDevice;
|
||
|
||
ASSERT(LowerDevice);
|
||
|
||
IoSkipCurrentIrpStackLocation(Irp);
|
||
return IoCallDriver(LowerDevice, Irp);
|
||
}
|
||
|
||
/* I really want PCSZ strings as last arguments because
|
||
* PnP ids are ANSI-encoded in PnP device string
|
||
* identification */
|
||
NTSTATUS
|
||
UsbhubInitMultiSzString(
|
||
OUT PUNICODE_STRING Destination,
|
||
... /* list of PCSZ */)
|
||
{
|
||
va_list args;
|
||
PCSZ Source;
|
||
ANSI_STRING AnsiString;
|
||
UNICODE_STRING UnicodeString;
|
||
ULONG DestinationSize = 0;
|
||
NTSTATUS Status = STATUS_SUCCESS;
|
||
|
||
ASSERT(Destination);
|
||
|
||
/* Calculate length needed for destination unicode string */
|
||
va_start(args, Destination);
|
||
Source = va_arg(args, PCSZ);
|
||
while (Source != NULL)
|
||
{
|
||
RtlInitAnsiString(&AnsiString, Source);
|
||
DestinationSize += RtlAnsiStringToUnicodeSize(&AnsiString)
|
||
+ sizeof(WCHAR) /* final NULL */;
|
||
Source = va_arg(args, PCSZ);
|
||
}
|
||
va_end(args);
|
||
if (DestinationSize == 0)
|
||
{
|
||
RtlInitUnicodeString(Destination, NULL);
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
/* Initialize destination string */
|
||
DestinationSize += sizeof(WCHAR); // final NULL
|
||
Destination->Buffer = (PWSTR)ExAllocatePoolWithTag(PagedPool, DestinationSize, USB_HUB_TAG);
|
||
if (!Destination->Buffer)
|
||
return STATUS_INSUFFICIENT_RESOURCES;
|
||
Destination->Length = 0;
|
||
Destination->MaximumLength = (USHORT)DestinationSize;
|
||
|
||
/* Copy arguments to destination string */
|
||
/* Use a temporary unicode string, which buffer is shared with
|
||
* destination string, to copy arguments */
|
||
UnicodeString.Length = Destination->Length;
|
||
UnicodeString.MaximumLength = Destination->MaximumLength;
|
||
UnicodeString.Buffer = Destination->Buffer;
|
||
va_start(args, Destination);
|
||
Source = va_arg(args, PCSZ);
|
||
while (Source != NULL)
|
||
{
|
||
RtlInitAnsiString(&AnsiString, Source);
|
||
Status = RtlAnsiStringToUnicodeString(&UnicodeString, &AnsiString, FALSE);
|
||
if (!NT_SUCCESS(Status))
|
||
{
|
||
ExFreePoolWithTag(Destination->Buffer, USB_HUB_TAG);
|
||
break;
|
||
}
|
||
Destination->Length += UnicodeString.Length + sizeof(WCHAR);
|
||
UnicodeString.MaximumLength -= UnicodeString.Length + sizeof(WCHAR);
|
||
UnicodeString.Buffer += UnicodeString.Length / sizeof(WCHAR) + 1;
|
||
UnicodeString.Length = 0;
|
||
Source = va_arg(args, PCSZ);
|
||
}
|
||
va_end(args);
|
||
if (NT_SUCCESS(Status))
|
||
{
|
||
/* Finish multi-sz string */
|
||
Destination->Buffer[Destination->Length / sizeof(WCHAR)] = L'\0';
|
||
Destination->Length += sizeof(WCHAR);
|
||
}
|
||
return Status;
|
||
}
|
||
|
||
NTSTATUS
|
||
UsbhubDuplicateUnicodeString(
|
||
OUT PUNICODE_STRING Destination,
|
||
IN PUNICODE_STRING Source,
|
||
IN POOL_TYPE PoolType)
|
||
{
|
||
ASSERT(Destination);
|
||
|
||
if (Source == NULL)
|
||
{
|
||
RtlInitUnicodeString(Destination, NULL);
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
Destination->Buffer = ExAllocatePool(PoolType, Source->MaximumLength);
|
||
if (Destination->Buffer == NULL)
|
||
{
|
||
return STATUS_INSUFFICIENT_RESOURCES;
|
||
}
|
||
|
||
Destination->MaximumLength = Source->MaximumLength;
|
||
Destination->Length = Source->Length;
|
||
RtlCopyMemory(Destination->Buffer, Source->Buffer, Source->MaximumLength);
|
||
|
||
return STATUS_SUCCESS;
|
||
}
|