windows-nt/Source/XPSP1/NT/drivers/wdm/usb/driver/usbser/serioctl.c
2020-09-26 16:20:57 +08:00

2258 lines
94 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/***************************************************************************
Copyright (c) 1998 Microsoft Corporation
Module Name:
SERIOCTL.C
Abstract:
Routines to handle serial IOCTLs for Legacy USB Modem Driver.
Environment:
kernel mode only
Notes:
THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR
PURPOSE.
Copyright (c) 1998 Microsoft Corporation. All Rights Reserved.
Revision History:
12/27/97 : created
Authors:
Tom Green
****************************************************************************/
#include <wdm.h>
#include <ntddser.h>
#include <stdio.h>
#include <stdlib.h>
#include <usb.h>
#include <usbdrivr.h>
#include <usbdlib.h>
#include <usbcomm.h>
#ifdef WMI_SUPPORT
#include <wmilib.h>
#include <wmidata.h>
#include <wmistr.h>
#endif
#include "usbser.h"
#include "serioctl.h"
#include "utils.h"
#include "usbserpw.h"
#include "debugwdm.h"
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGEUSBS, SetBaudRate)
#pragma alloc_text(PAGEUSBS, GetBaudRate)
#pragma alloc_text(PAGEUSBS, SetLineControl)
#pragma alloc_text(PAGEUSBS, GetLineControl)
#pragma alloc_text(PAGEUSBS, SetTimeouts)
#pragma alloc_text(PAGEUSBS, GetTimeouts)
#pragma alloc_text(PAGEUSBS, SetChars)
#pragma alloc_text(PAGEUSBS, GetChars)
#pragma alloc_text(PAGEUSBS, SetClrDtr)
#pragma alloc_text(PAGEUSBS, ResetDevice)
#pragma alloc_text(PAGEUSBS, SetRts)
#pragma alloc_text(PAGEUSBS, ClrRts)
#pragma alloc_text(PAGEUSBS, SetBreak)
#pragma alloc_text(PAGEUSBS, SetQueueSize)
#pragma alloc_text(PAGEUSBS, GetWaitMask)
#pragma alloc_text(PAGEUSBS, SetWaitMask)
#pragma alloc_text(PAGEUSBS, WaitOnMask)
#pragma alloc_text(PAGEUSBS, ImmediateChar)
#pragma alloc_text(PAGEUSBS, Purge)
#pragma alloc_text(PAGEUSBS, GetHandflow)
#pragma alloc_text(PAGEUSBS, SetHandflow)
#pragma alloc_text(PAGEUSBS, GetModemStatus)
#pragma alloc_text(PAGEUSBS, GetDtrRts)
#pragma alloc_text(PAGEUSBS, GetCommStatus)
#pragma alloc_text(PAGEUSBS, GetProperties)
#pragma alloc_text(PAGEUSBS, LsrmstInsert)
#pragma alloc_text(PAGEUSBS, ConfigSize)
#pragma alloc_text(PAGEUSBS, GetStats)
#pragma alloc_text(PAGEUSBS, ClearStats)
#pragma alloc_text(PAGEUSBS, SerialGetProperties)
#endif // ALLOC_PRAGMA
LOCAL UCHAR StopBits[] =
{
STOP_BIT_1, // USB_COMM_STOPBITS_10
STOP_BITS_1_5, // USB_COMM_STOPBITS_15
STOP_BITS_2 // USB_COMM_STOPBITS_20
};
LOCAL UCHAR ParityType[] =
{
NO_PARITY, // USB_COMM_PARITY_NONE
ODD_PARITY, // USB_COMM_PARITY_ODD
EVEN_PARITY, // USB_COMM_PARITY_EVEN
MARK_PARITY, // USB_COMM_PARITY_MARK
SPACE_PARITY // USB_COMM_PARITY_SPACE
};
/************************************************************************/
/* SetBaudRate */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_BAUD_RATE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetBaudRate(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj)
{
PSERIAL_BAUD_RATE Br = (PSERIAL_BAUD_RATE) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
KIRQL OldIrql;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetBaudRate");
UsbSerSerialDump(USBSERTRACEIOC, (">SetBaudRate(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(SERIAL_BAUD_RATE)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->CurrentBaud = Br->BaudRate;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_TRACE3(("BaudRate (%08X)\n", Br->BaudRate));
NtStatus = SetLineControlAndBaud(PDevObj);
}
DEBUG_LOG_PATH("exit SetBaudRate");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetBaudRate %08X\n", NtStatus));
return NtStatus;
} // SetBaudRate
/************************************************************************/
/* GetBaudRate */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_BAUD_RATE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetBaudRate(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj)
{
PSERIAL_BAUD_RATE Br = (PSERIAL_BAUD_RATE) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetBaudRate");
UsbSerSerialDump(USBSERTRACEIOC, (">GetBaudRate(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_BAUD_RATE)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
GetLineControlAndBaud(PDevObj);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
Br->BaudRate = DeviceExtension->CurrentBaud;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIAL_BAUD_RATE);
}
DEBUG_LOG_PATH("exit GetBaudRate");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetBaudRate %08X\n", NtStatus));
return NtStatus;
} // GetBaudRate
/************************************************************************/
/* SetLineControl */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_LINE_CONTROL */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetLineControl(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj)
{
PSERIAL_LINE_CONTROL LineControl
= (PSERIAL_LINE_CONTROL) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetLineControl");
UsbSerSerialDump(USBSERTRACEIOC, (">SetLineControl(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(SERIAL_LINE_CONTROL)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->LineControl = *LineControl;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
// set line control for USB modem
NtStatus = SetLineControlAndBaud(PDevObj);
}
DEBUG_LOG_PATH("exit SetLineControl");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetLineControl %08X\n", NtStatus));
return NtStatus;
} // SetLineControl
/************************************************************************/
/* GetLineControl */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_LINE_CONTROL */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetLineControl(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj)
{
PSERIAL_LINE_CONTROL LineControl =
(PSERIAL_LINE_CONTROL) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetLineControl");
UsbSerSerialDump(USBSERTRACEIOC, (">GetLineControl(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_LINE_CONTROL))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
GetLineControlAndBaud(PDevObj);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*LineControl = DeviceExtension->LineControl;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIAL_LINE_CONTROL);
}
DEBUG_LOG_PATH("exit GetLineControl");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetLineControl %08X\n", NtStatus));
return NtStatus;
} // GetLineControl
/************************************************************************/
/* SetTimeouts */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_TIMEOUTS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetTimeouts(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_TIMEOUTS Timeouts =
(PSERIAL_TIMEOUTS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetTimeouts");
UsbSerSerialDump(USBSERTRACEIOC | USBSERTRACETM,
(">SetTimeouts(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(SERIAL_TIMEOUTS))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->Timeouts = *Timeouts;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
DEBUG_LOG_PATH("exit SetTimeouts");
UsbSerSerialDump(USBSERTRACEIOC | USBSERTRACETM, ("<SetTimeouts %08X\n",
NtStatus));
return NtStatus;
} // SetTimeouts
/************************************************************************/
/* GetTimeouts */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_TIMEOUTS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetTimeouts(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_TIMEOUTS Timeouts =
(PSERIAL_TIMEOUTS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetTimeouts");
UsbSerSerialDump(USBSERTRACEIOC | USBSERTRACETM,
(">GetTimeouts(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_TIMEOUTS))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*Timeouts = DeviceExtension->Timeouts;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIAL_TIMEOUTS);
}
DEBUG_LOG_PATH("exit GetTimeouts");
UsbSerSerialDump(USBSERTRACEIOC | USBSERTRACETM, ("<GetTimeouts %08X\n",
NtStatus));
return NtStatus;
} // GetTimeouts
/************************************************************************/
/* SetChars */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_CHARS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetChars(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_CHARS SpecialChars =
(PSERIAL_CHARS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetChars");
UsbSerSerialDump(USBSERTRACEIOC, (">SetChars(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(SERIAL_CHARS))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->SpecialChars = *SpecialChars;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
DEBUG_LOG_PATH("exit SetChars");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetChars %08X\n"));
return NtStatus;
} // SetChars
/************************************************************************/
/* GetChars */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_CHARS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetChars(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_CHARS SpecialChars =
(PSERIAL_CHARS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetChars");
UsbSerSerialDump(USBSERTRACEIOC, (">GetChars(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_CHARS))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*SpecialChars = DeviceExtension->SpecialChars;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIAL_CHARS);
}
DEBUG_LOG_PATH("exit GetChars");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetChars %08X\n", NtStatus));
return NtStatus;
} // GetChars
/************************************************************************/
/* SetClrDtr */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_DTR and IOCTL_SERIAL_CLR_DTR */
/* */
/* Arguments: */
/* */
/* PDevObj - pointer to device object */
/* Set - TRUE if setting DTR, FALSE if clearing DTR */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetClrDtr(IN PDEVICE_OBJECT PDevObj, IN BOOLEAN Set)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
USHORT State = 0;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetClrDtr");
UsbSerSerialDump(USBSERTRACEIOC, (">SetClrDtr\n"));
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
if(DeviceExtension->DTRRTSState & SERIAL_RTS_STATE)
State |= USB_COMM_RTS;
if (Set) {
DeviceExtension->DTRRTSState |= SERIAL_DTR_STATE;
State |= USB_COMM_DTR;
} else {
DeviceExtension->DTRRTSState &= ~SERIAL_DTR_STATE;
}
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
if(DeviceExtension->DTRRTSState & SERIAL_RTS_STATE)
State |= USB_COMM_RTS;
NtStatus = ClassVendorCommand(PDevObj, USB_COMM_SET_CONTROL_LINE_STATE,
State, DeviceExtension->CommInterface,
NULL, NULL, FALSE, USBSER_CLASS_COMMAND);
if(!NT_SUCCESS(NtStatus)) {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->DTRRTSState &= ~SERIAL_DTR_STATE;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
DEBUG_LOG_PATH("exit SetClrDtr");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetClrDtr %08X\n", NtStatus));
return NtStatus;
} // SetClrDtr
/************************************************************************/
/* ResetDevice */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_RESET_DEVICE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
ResetDevice(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter ResetDevice");
UsbSerSerialDump(USBSERTRACEIOC, (">ResetDevice(%08X)\n", Irp));
// get line control and baud rate info
GetLineControlAndBaud(PDevObj);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
// do device extension device specific stuff here
DeviceExtension->SupportedBauds = SERIAL_BAUD_300 | SERIAL_BAUD_600
| SERIAL_BAUD_1200 | SERIAL_BAUD_2400 | SERIAL_BAUD_4800
| SERIAL_BAUD_9600 | SERIAL_BAUD_19200 | SERIAL_BAUD_38400
| SERIAL_BAUD_57600 | SERIAL_BAUD_115200;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_LOG_PATH("exit ResetDevice");
UsbSerSerialDump(USBSERTRACEIOC, ("<ResetDevice %08X\n"));
return NtStatus;
} // ResetDevice
/************************************************************************/
/* SetRts */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_RTS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetRts(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
USHORT State = 0;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetRts");
UsbSerSerialDump(USBSERTRACEIOC, (">SetRts(%08X)\n", Irp));
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->DTRRTSState |= SERIAL_RTS_STATE;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_LOG_PATH("exit SetRts");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetRts %08X\n", NtStatus));
return NtStatus;
} // SetRts
/************************************************************************/
/* ClrRts */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_CLR_RTS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
ClrRts(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
USHORT State = 0;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter ClrRts");
UsbSerSerialDump(USBSERTRACEIOC, (">ClrRts(%08X)\n", Irp));
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->DTRRTSState &= ~SERIAL_RTS_STATE;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_LOG_PATH("exit ClrRts");
UsbSerSerialDump(USBSERTRACEIOC, ("<ClrRts %08X\n", NtStatus));
return NtStatus;
} // ClrRts
/************************************************************************/
/* SetBreak */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_BREAK_ON & IOCTL_SERIAL_SET_BREAK_OFF */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* PDevObj - pointer to device object */
/* Time - time to assert break in ms */
/* (0xFFFF - on / 0x0000 - off) */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetBreak(IN PIRP Irp, IN PDEVICE_OBJECT PDevObj, USHORT Time)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
USBSER_LOCKED_PAGED_CODE();
DEBUG_LOG_PATH("enter SetBreak");
UsbSerSerialDump(USBSERTRACEIOC, (">SetBreak(%08X)\n", Irp));
NtStatus = ClassVendorCommand(PDevObj, USB_COMM_SEND_BREAK, Time,
DeviceExtension->CommInterface, NULL,
NULL, FALSE, USBSER_CLASS_COMMAND);
DEBUG_LOG_PATH("exit SetBreak");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetBreak %08X\n", NtStatus));
return NtStatus;
} // SetBreak
/************************************************************************/
/* SetQueueSize */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_QUEUE_SIZE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetQueueSize(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
PULONG QueueSize = (PULONG) Irp->AssociatedIrp.SystemBuffer;
DEBUG_LOG_PATH("enter SetQueueSize");
UsbSerSerialDump(USBSERTRACEIOC, (">SetQueueSize(%08X)\n", Irp));
USBSER_LOCKED_PAGED_CODE();
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(ULONG))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
DEBUG_TRACE1(("SetQueueSize (%08X)\n", *QueueSize));
// we will go ahead and save this, but we don't care.
// DeviceExtension->RxQueueSize = *QueueSize;
}
DEBUG_LOG_PATH("exit SetQueueSize");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetQueueSize %08X\n", NtStatus));
return NtStatus;
} // SetQueueSize
/************************************************************************/
/* GetWaitMask */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_WAIT_MASK */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetWaitMask(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG WaitMask = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetWaitMask");
UsbSerSerialDump(USBSERTRACEIOC, (">GetWaitMask(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if(IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(ULONG))
{
NtStatus = STATUS_BUFFER_TOO_SMALL;
}
else
{
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*WaitMask = DeviceExtension->IsrWaitMask;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(ULONG);
}
DEBUG_LOG_PATH("exit GetWaitMask");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetWaitMask %08X\n"));
return NtStatus;
} // GetWaitMask
/************************************************************************/
/* SetWaitMask */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_WAIT_MASK */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetWaitMask(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG WaitMask = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetWaitMask");
UsbSerSerialDump(USBSERTRACEIOC, (">SetWaitMask(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
// make sure it's a valid request
if (*WaitMask & ~( SERIAL_EV_RXCHAR |
SERIAL_EV_RXFLAG |
SERIAL_EV_TXEMPTY |
SERIAL_EV_CTS |
SERIAL_EV_DSR |
SERIAL_EV_RLSD |
SERIAL_EV_BREAK |
SERIAL_EV_ERR |
SERIAL_EV_RING |
SERIAL_EV_PERR |
SERIAL_EV_RX80FULL |
SERIAL_EV_EVENT1 |
SERIAL_EV_EVENT2)) {
NtStatus = STATUS_INVALID_PARAMETER;
} else {
UsbSerCompletePendingWaitMasks(DeviceExtension);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->HistoryMask = 0;
DeviceExtension->IsrWaitMask = *WaitMask;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_TRACE3(("SetWaitMask (%08X)\n", *WaitMask));
}
}
DEBUG_LOG_PATH("exit SetWaitMask");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetWaitMask %08X\n", NtStatus));
return NtStatus;
} // SetWaitMask
/************************************************************************/
/* WaitOnMask */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_WAIT_ON_MASK */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
WaitOnMask(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG WaitMask = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter WaitOnMask");
UsbSerSerialDump(USBSERTRACEIOC, (">WaitOnMask(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
// if we have an event to report, just go ahead and return it
if (DeviceExtension->HistoryMask) {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*WaitMask = DeviceExtension->HistoryMask;
DeviceExtension->HistoryMask = 0;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(ULONG);
UsbSerSerialDump(USBSERCOMPEV,
("Completing maskirp(3) %08x\n",
*WaitMask));
DEBUG_TRACE3(("Signal Event (%08X)\n", *WaitMask));
} else {
KIRQL cancelIrql;
ACQUIRE_CANCEL_SPINLOCK(DeviceExtension, &cancelIrql);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
// just in case something comes in, we'll do a while loop
while (DeviceExtension->CurrentMaskIrp) {
PIRP pOldIrp;
DEBUG_TRACE3(("Completing previous mask\n"));
pOldIrp = DeviceExtension->CurrentMaskIrp;
DeviceExtension->CurrentMaskIrp = NULL;
pOldIrp->IoStatus.Status = STATUS_SUCCESS;
IoSetCancelRoutine(pOldIrp, NULL);
*WaitMask = 0;
UsbSerSerialDump(USBSERCOMPEV,
("Completing maskirp(4)\n"));
//
// Release locks, complete request, then
// reacquire the locks
//
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock,
OldIrql);
RELEASE_CANCEL_SPINLOCK(DeviceExtension, cancelIrql);
IoCompleteRequest(pOldIrp, IO_SERIAL_INCREMENT);
ACQUIRE_CANCEL_SPINLOCK(DeviceExtension, &cancelIrql);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock,
&OldIrql);
}
//
// Check to see if it needs to be cancelled
//
if (Irp->Cancel) {
Irp->IoStatus.Status = STATUS_CANCELLED;
Irp->IoStatus.Information = 0;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
RELEASE_CANCEL_SPINLOCK(DeviceExtension, cancelIrql);
} else {
IoSetCancelRoutine(Irp, UsbSerCancelWaitOnMask);
NtStatus = Irp->IoStatus.Status = STATUS_PENDING;
DeviceExtension->CurrentMaskIrp = Irp;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
RELEASE_CANCEL_SPINLOCK(DeviceExtension, cancelIrql);
IoMarkIrpPending(Irp);
}
}
}
DEBUG_LOG_PATH("exit WaitOnMask");
UsbSerSerialDump(USBSERTRACEIOC, ("<WaitOnMask %08X\n", NtStatus));
return NtStatus;
} // WaitOnMask
/************************************************************************/
/* ImmediateChar */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_IMMEDIATE_CHAR */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceObject - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
ImmediateChar(IN PIRP Irp, IN PDEVICE_OBJECT DeviceObject)
{
PUCHAR Char = (PUCHAR) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
USBSER_LOCKED_PAGED_CODE();
DEBUG_LOG_PATH("enter ImmediateChar");
UsbSerSerialDump(USBSERTRACEIOC, (">ImmediateChar(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(UCHAR)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
//
// We just treat this as a write since we have no internal
// data buffer.
//
IrpStack->Parameters.Write.Length = sizeof(UCHAR);
NtStatus = UsbSer_Write(DeviceObject, Irp);
}
DEBUG_LOG_PATH("exit ImmediateChar");
UsbSerSerialDump(USBSERTRACEIOC, ("<ImmediateChar, %08X\n", NtStatus));
return NtStatus;
} // ImmediateChar
/************************************************************************/
/* Purge */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_PURGE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
Purge(IN PDEVICE_OBJECT PDevObj, IN PIRP Irp,
IN PDEVICE_EXTENSION DeviceExtension)
{
ULONG Mask = *((PULONG) Irp->AssociatedIrp.SystemBuffer);
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
KIRQL OldIrql;
ULONG Count;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter Purge");
UsbSerSerialDump(USBSERTRACEIOC, (">Purge(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
// make sure purge request is valid
if ((!Mask) || (Mask & ( ~( SERIAL_PURGE_TXABORT |
SERIAL_PURGE_RXABORT |
SERIAL_PURGE_TXCLEAR |
SERIAL_PURGE_RXCLEAR)))) {
NtStatus = STATUS_INVALID_PARAMETER;
} else {
if (Mask & SERIAL_PURGE_RXCLEAR) {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
Count = DeviceExtension->CharsInReadBuff;
DeviceExtension->CharsInReadBuff = 0;
DeviceExtension->CurrentReadBuffPtr = 0;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
if(Count)
{
RestartRead(DeviceExtension);
}
}
if (Mask & SERIAL_PURGE_RXABORT) {
UsbSerKillAllReadsOrWrites(PDevObj, &DeviceExtension->ReadQueue,
&DeviceExtension->CurrentReadIrp);
}
if (Mask & SERIAL_PURGE_TXABORT) {
//
// DO NOTHING because USB owns the request. However, it may
// prove in practice that we will have to cancel the IRPs on behalf
// of the caller.
}
}
}
DEBUG_LOG_PATH("exit Purge");
UsbSerSerialDump(USBSERTRACEIOC, ("<Purge %08X\n", NtStatus));
return NtStatus;
} // Purge
/************************************************************************/
/* GetHandflow */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_HANDFLOW */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetHandflow(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_HANDFLOW HandFlow
= (PSERIAL_HANDFLOW) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetHandflow");
UsbSerSerialDump(USBSERTRACEIOC, (">GetHandFlow(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_HANDFLOW)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*HandFlow = DeviceExtension->HandFlow;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIAL_HANDFLOW);
}
DEBUG_LOG_PATH("exit GetHandflow");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetHandFlow %08X\n", NtStatus));
return NtStatus;
} // GetHandflow
/************************************************************************/
/* SetHandflow */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_SET_HANDFLOW */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetHandflow(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_HANDFLOW HandFlow
= (PSERIAL_HANDFLOW) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SetHandflow");
UsbSerSerialDump(USBSERTRACEIOC, (">SetHandFlow(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.InputBufferLength
< sizeof(SERIAL_HANDFLOW)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->HandFlow = *HandFlow;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_TRACE3(("ControlHandShake (%08X)\n",
DeviceExtension->HandFlow.ControlHandShake));
}
DEBUG_LOG_PATH("exit SetHandflow");
UsbSerSerialDump(USBSERTRACEIOC, ("<SetHandFlow %08X\n", NtStatus));
return NtStatus;
} // SetHandflow
/************************************************************************/
/* GetModemStatus */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_MODEMSTATUS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetModemStatus(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG ModemStatus = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
KIRQL OldIrql;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetModemStatus");
UsbSerSerialDump(USBSERTRACEIOC, (">GetModemStatus(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*ModemStatus = DeviceExtension->FakeModemStatus;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(ULONG);
DEBUG_TRACE3(("ModemStatus (%08X)\n", *ModemStatus));
}
DEBUG_LOG_PATH("exit GetModemStatus");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetModemStatus %08X\n", NtStatus));
return NtStatus;
} // GetModemStatus
/************************************************************************/
/* GetDtrRts */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_DTRRTS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetDtrRts(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG ModemControl = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
KIRQL OldIrql;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetDtrRts");
UsbSerSerialDump(USBSERTRACEIOC, (">GetDtrRts(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*ModemControl = DeviceExtension->DTRRTSState;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(ULONG);
}
DEBUG_LOG_PATH("exit GetDtrRts");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetDtrRts %08X\n", NtStatus));
return NtStatus;
} // GetDtrRts
/************************************************************************/
/* GetCommStatus */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_COMMSTATUS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetCommStatus(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIAL_STATUS SerialStatus
= (PSERIAL_STATUS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetCommStatus");
UsbSerSerialDump(USBSERTRACEIOC, (">GetCommStatus(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_STATUS)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
if (NT_SUCCESS(NtStatus)) {
RtlZeroMemory(SerialStatus, sizeof(SERIAL_STATUS));
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
SerialStatus->AmountInInQueue = DeviceExtension->CharsInReadBuff;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_TRACE2(("AmountInInQueue (%08X)\n", SerialStatus->AmountInInQueue));
SerialStatus->Errors = 0;
SerialStatus->EofReceived = FALSE;
SerialStatus->AmountInOutQueue = 0;
SerialStatus->WaitForImmediate = 0;
SerialStatus->HoldReasons = 0;
Irp->IoStatus.Information = sizeof(SERIAL_STATUS);
}
}
DEBUG_LOG_PATH("exit GetCommStatus");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetCommStatus %08X\n", NtStatus));
return NtStatus;
} // GetCommStatus
/************************************************************************/
/* GetProperties */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_PROPERTIES */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetProperties(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
USBSER_LOCKED_PAGED_CODE();
DEBUG_LOG_PATH("enter GetProperties");
UsbSerSerialDump(USBSERTRACEIOC, (">GetProperties(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIAL_COMMPROP)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
SerialGetProperties(DeviceExtension,
(PSERIAL_COMMPROP)Irp->AssociatedIrp.SystemBuffer);
Irp->IoStatus.Information = sizeof(SERIAL_COMMPROP);
}
DEBUG_LOG_PATH("exit GetProperties");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetProperties %08X\n", NtStatus));
return NtStatus;
} // GetProperties
/************************************************************************/
/* LsrmstInsert */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_LSRMST_INSERT */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
LsrmstInsert(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
USBSER_LOCKED_PAGED_CODE();
DEBUG_LOG_PATH("enter LsrmstInsert");
UsbSerSerialDump(USBSERTRACEIOC, (">LsrmstInsert(%08X)\n", Irp));
UsbSerSerialDump(USBSERTRACEIOC, ("<LsrmstInsert (%08X)\n",
STATUS_NOT_SUPPORTED));
return STATUS_NOT_SUPPORTED;
} // LsrmstInsert
/************************************************************************/
/* ConfigSize */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_CONFIG_SIZE */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
ConfigSize(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PULONG ConfigSize = (PULONG) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
PIO_STACK_LOCATION IrpStack;
USBSER_LOCKED_PAGED_CODE();
DEBUG_LOG_PATH("enter ConfigSize");
UsbSerSerialDump(USBSERTRACEIOC, (">ConfigSize(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(ULONG)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
*ConfigSize = 0;
Irp->IoStatus.Information = sizeof(ULONG);
}
DEBUG_LOG_PATH("exit ConfigSize");
UsbSerSerialDump(USBSERTRACEIOC, ("<ConfigSize %08X\n", NtStatus));
return NtStatus;
} // ConfigSize
/************************************************************************/
/* GetStats */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_GET_STATS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetStats(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
PSERIALPERF_STATS Stats
= (PSERIALPERF_STATS) Irp->AssociatedIrp.SystemBuffer;
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
PIO_STACK_LOCATION IrpStack;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter GetStats");
UsbSerSerialDump(USBSERTRACEIOC, (">GetStats(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
IrpStack = IoGetCurrentIrpStackLocation(Irp);
if (IrpStack->Parameters.DeviceIoControl.OutputBufferLength
< sizeof(SERIALPERF_STATS)) {
NtStatus = STATUS_BUFFER_TOO_SMALL;
} else {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
*Stats = DeviceExtension->PerfStats;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
Irp->IoStatus.Information = sizeof(SERIALPERF_STATS);
}
DEBUG_LOG_PATH("exit GetStats");
UsbSerSerialDump(USBSERTRACEIOC, ("<GetStats %08X\n", NtStatus));
return NtStatus;
} // GetStats
/************************************************************************/
/* ClearStats */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Handle IOCTL_SERIAL_CLEAR_STATS */
/* */
/* Arguments: */
/* */
/* Irp - pointer to an I/O Request Packet */
/* DeviceExtension - pointer to device extension */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
ClearStats(IN PIRP Irp, IN PDEVICE_EXTENSION DeviceExtension)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
KIRQL OldIrql;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter ClearStats");
UsbSerSerialDump(USBSERTRACEIOC, (">ClearStats(%08X)\n", Irp));
Irp->IoStatus.Information = 0;
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
RtlZeroMemory(&DeviceExtension->PerfStats,
sizeof(SERIALPERF_STATS));
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_LOG_PATH("exit ClearStats");
UsbSerSerialDump(USBSERTRACEIOC, ("<ClearStats %08X\n", NtStatus));
return NtStatus;
} // ClearStats
/************************************************************************/
/* SerialGetProperties */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Get serial device properties */
/* */
/* Arguments: */
/* */
/* DeviceExtension - pointer to device extension */
/* Properties - pointer to device properties to fill in */
/* */
/* Return Value: */
/* */
/* VOID */
/* */
/************************************************************************/
VOID
SerialGetProperties(IN PDEVICE_EXTENSION DeviceExtension,
IN PSERIAL_COMMPROP Properties)
{
KIRQL OldIrql;
USBSER_ALWAYS_LOCKED_CODE();
DEBUG_LOG_PATH("enter SerialGetProperties");
UsbSerSerialDump(USBSERTRACEIOC, (">SerialGetProperties\n"));
RtlZeroMemory(Properties, sizeof(SERIAL_COMMPROP));
Properties->PacketLength = sizeof(SERIAL_COMMPROP);
Properties->PacketVersion = 2;
Properties->ServiceMask = SERIAL_SP_SERIALCOMM;
Properties->MaxTxQueue = 0;
Properties->MaxRxQueue = 0;
Properties->MaxBaud = SERIAL_BAUD_USER;
Properties->ProvSubType = SERIAL_SP_MODEM;
Properties->ProvCapabilities = SERIAL_PCF_DTRDSR | SERIAL_PCF_CD
| SERIAL_PCF_PARITY_CHECK | SERIAL_PCF_TOTALTIMEOUTS
| SERIAL_PCF_INTTIMEOUTS;
Properties->SettableParams = SERIAL_SP_PARITY | SERIAL_SP_BAUD
| SERIAL_SP_DATABITS | SERIAL_SP_STOPBITS | SERIAL_SP_HANDSHAKING
| SERIAL_SP_PARITY_CHECK | SERIAL_SP_CARRIER_DETECT;
Properties->SettableData = SERIAL_DATABITS_7 | SERIAL_DATABITS_8;
Properties->SettableStopParity = SERIAL_STOPBITS_10 | SERIAL_PARITY_NONE
| SERIAL_PARITY_ODD | SERIAL_PARITY_EVEN | SERIAL_PARITY_MARK
| SERIAL_PARITY_SPACE;
Properties->CurrentTxQueue = 0;
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
Properties->CurrentRxQueue = DeviceExtension->RxQueueSize;
Properties->SettableBaud = DeviceExtension->SupportedBauds;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_LOG_PATH("exit SerialGetProperties");
UsbSerSerialDump(USBSERTRACEIOC, ("<SerialGetProperties\n"));
} // SerialGetProperties
/************************************************************************/
/* GetLineControlAndBaud */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* CDC command to get line control settings and baud */
/* */
/* Arguments: */
/* */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
GetLineControlAndBaud(IN PDEVICE_OBJECT PDevObj)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
USB_COMM_LINE_CODING LineCoding;
ULONG Size = sizeof(LineCoding);
KIRQL OldIrql;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
DEBUG_LOG_PATH("enter GetLineControlAndBaud");
NtStatus = ClassVendorCommand(PDevObj, USB_COMM_GET_LINE_CODING, 0,
DeviceExtension->CommInterface,
&LineCoding, &Size, TRUE,
USBSER_CLASS_COMMAND);
if (NT_SUCCESS(NtStatus)) {
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
DeviceExtension->CurrentBaud = LineCoding.DTERate;
DeviceExtension->LineControl.StopBits = StopBits[LineCoding.CharFormat];
DeviceExtension->LineControl.Parity = ParityType[LineCoding.ParityType];
DeviceExtension->LineControl.WordLength = LineCoding.DataBits;
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
DEBUG_TRACE3(("Baud (%08X) StopBits (%08X) DataBits (%08X)\n",
LineCoding.DTERate, LineCoding.CharFormat,
LineCoding.DataBits));
}
DEBUG_LOG_PATH("exit GetLineControlAndBaud");
return NtStatus;
} // GetLineControlAndBaud
/************************************************************************/
/* SetLineControlAndBaud */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* CDC command to set line control and baud */
/* */
/* Arguments: */
/* */
/* PDevObj - pointer to device object */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
SetLineControlAndBaud(IN PDEVICE_OBJECT PDevObj)
{
NTSTATUS NtStatus = STATUS_SUCCESS;
USB_COMM_LINE_CODING LineCoding;
ULONG Size = sizeof(LineCoding);
PSERIAL_LINE_CONTROL LineControl;
KIRQL OldIrql;
PDEVICE_EXTENSION DeviceExtension = PDevObj->DeviceExtension;
DEBUG_LOG_PATH("enter SetLineControlAndBaud");
// get pointer to line control in extension
LineControl = &DeviceExtension->LineControl;
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
// set up the line coding data structure
LineCoding.DTERate = DeviceExtension->CurrentBaud;
LineCoding.DataBits = LineControl->WordLength;
switch (DeviceExtension->LineControl.StopBits) {
case STOP_BIT_1:
LineCoding.CharFormat = USB_COMM_STOPBITS_10;
break;
case STOP_BITS_1_5:
LineCoding.CharFormat = USB_COMM_STOPBITS_15;
break;
case STOP_BITS_2:
LineCoding.CharFormat = USB_COMM_STOPBITS_20;
break;
default:
NtStatus = STATUS_INVALID_PARAMETER;
break;
}
switch (DeviceExtension->LineControl.Parity) {
case NO_PARITY:
LineCoding.ParityType = USB_COMM_PARITY_NONE;
break;
case ODD_PARITY:
LineCoding.ParityType = USB_COMM_PARITY_ODD;
break;
case EVEN_PARITY:
LineCoding.ParityType = USB_COMM_PARITY_EVEN;
break;
case MARK_PARITY:
LineCoding.ParityType = USB_COMM_PARITY_MARK;
break;
case SPACE_PARITY:
LineCoding.ParityType = USB_COMM_PARITY_SPACE;
break;
default:
NtStatus = STATUS_INVALID_PARAMETER;
break;
}
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
// the request must be valid, so send it down to the device
if (NT_SUCCESS(NtStatus)) {
NtStatus = ClassVendorCommand(PDevObj, USB_COMM_SET_LINE_CODING, 0,
DeviceExtension->CommInterface,
&LineCoding, &Size, FALSE,
USBSER_CLASS_COMMAND);
}
// let's go ahead and just grab this info again in case of an error
GetLineControlAndBaud(PDevObj);
DEBUG_LOG_PATH("exit SetLineControlAndBaud");
return NtStatus;
} // SetLineControlAndBaud
/************************************************************************/
/* NotifyCompletion */
/************************************************************************/
/* */
/* Routine Description: */
/* */
/* Notify completion routine. */
/* */
/* Arguments: */
/* */
/* DeviceObject - pointer to a device object */
/* Irp - pointer to Irp */
/* Context - pointer to driver defined context */
/* */
/* Return Value: */
/* */
/* NTSTATUS */
/* */
/************************************************************************/
NTSTATUS
NotifyCompletion(IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp, IN PVOID Context)
{
PDEVICE_EXTENSION DeviceExtension = (PDEVICE_EXTENSION) Context;
PURB Urb;
ULONG Count;
KIRQL OldIrql;
KIRQL cancelIrql;
PUSB_COMM_SERIAL_STATUS SerialState;
USHORT ModemStatus;
USHORT OldModemStatus;
PIRP CurrentMaskIrp = NULL;
BOOLEAN startRead = FALSE;
DEBUG_LOG_PATH("enter NotifyCompletion");
Urb = DeviceExtension->NotifyUrb;
Count = Urb->UrbBulkOrInterruptTransfer.TransferBufferLength;
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
SerialState = (PUSB_COMM_SERIAL_STATUS) DeviceExtension->NotificationBuff;
// see if it is our notification
if (SerialState->Notification == USB_COMM_SERIAL_STATE
&& Count == sizeof(USB_COMM_SERIAL_STATUS)) {
OldModemStatus = DeviceExtension->FakeModemStatus;
// cobble up a fake modem status
DeviceExtension->FakeModemStatus = SERIAL_MSR_CTS;
DeviceExtension->FakeLineStatus = 0;
DeviceExtension->HistoryMask = 0;
ModemStatus = SerialState->SerialState;
DeviceExtension->FakeLineStatus = 0;
DEBUG_TRACE1(("CDC Serial State (%08X)\n", ModemStatus));
if (ModemStatus & USB_COMM_DSR)
DeviceExtension->FakeModemStatus |= SERIAL_MSR_DSR;
if (ModemStatus & USB_COMM_DCD)
DeviceExtension->FakeModemStatus |= SERIAL_MSR_DCD;
if (ModemStatus & USB_COMM_RING)
DeviceExtension->FakeModemStatus |= SERIAL_MSR_RI;
// let's see what has changed in the status register
ModemStatus = OldModemStatus ^ DeviceExtension->FakeModemStatus;
if (ModemStatus & SERIAL_MSR_DSR)
DeviceExtension->HistoryMask |= SERIAL_EV_DSR;
if (ModemStatus & SERIAL_MSR_DCD)
DeviceExtension->HistoryMask |= SERIAL_EV_RLSD;
if (ModemStatus & SERIAL_MSR_RI)
DeviceExtension->HistoryMask |= SERIAL_EV_RING;
// see if we have any events we are waiting for
DeviceExtension->HistoryMask &= DeviceExtension->IsrWaitMask;
// update perf stats if we had any errors
if (ModemStatus & USB_COMM_FRAMING_ERROR) {
DeviceExtension->PerfStats.FrameErrorCount++;
DeviceExtension->FakeLineStatus |= SERIAL_LSR_FE;
}
if (ModemStatus & USB_COMM_OVERRUN) {
DeviceExtension->PerfStats.BufferOverrunErrorCount++;
DeviceExtension->FakeLineStatus |= SERIAL_LSR_OE;
}
if (ModemStatus & USB_COMM_PARITY_ERROR) {
DeviceExtension->PerfStats.ParityErrorCount++;
DeviceExtension->FakeLineStatus |= SERIAL_LSR_PE;
}
if (ModemStatus & USB_COMM_BREAK) {
DeviceExtension->FakeLineStatus |= SERIAL_LSR_BI;
}
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
ACQUIRE_CANCEL_SPINLOCK(DeviceExtension, &cancelIrql);
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
// let's see if we have any events to signal
CurrentMaskIrp = DeviceExtension->CurrentMaskIrp;
if (CurrentMaskIrp && DeviceExtension->HistoryMask) {
*(PULONG) (CurrentMaskIrp->AssociatedIrp.SystemBuffer) =
DeviceExtension->HistoryMask;
CurrentMaskIrp->IoStatus.Status = STATUS_SUCCESS;
CurrentMaskIrp->IoStatus.Information = sizeof(ULONG);
DeviceExtension->CurrentMaskIrp = NULL;
}
else
{
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
RELEASE_CANCEL_SPINLOCK(DeviceExtension, cancelIrql);
}
DEBUG_TRACE1(("Modem Status (%08X)\n", DeviceExtension->FakeModemStatus));
}
else
{
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
// complete the queued IRP if needed
if (CurrentMaskIrp && DeviceExtension->HistoryMask
&& Irp->IoStatus.Status == STATUS_SUCCESS)
{
//
// We should still be holding cancel spin lock because
// of above if()
UsbSerSerialDump(USBSERCOMPEV, ("Completing maskirp (4) %08X\n",
DeviceExtension->HistoryMask));
DeviceExtension->HistoryMask = 0;
IoSetCancelRoutine(CurrentMaskIrp, NULL);
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
RELEASE_CANCEL_SPINLOCK(DeviceExtension, cancelIrql);
IoCompleteRequest(CurrentMaskIrp, IO_NO_INCREMENT);
}
// check for Irp cancelled or error
if(Irp->IoStatus.Status == STATUS_CANCELLED)
{
goto NotifyCompletionErr;
}
else if(!NT_SUCCESS(Irp->IoStatus.Status))
{
UsbSerFetchBooleanLocked(&DeviceExtension->AcceptingRequests,
FALSE, &DeviceExtension->ControlLock);
goto NotifyCompletionErr;
}
ACQUIRE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, &OldIrql);
// kick off another notification request if we still need to
if (DeviceExtension->AcceptingRequests
&& (DeviceExtension->CurrentDevicePowerState == PowerDeviceD0))
{
// see if we have a work item queued already
if(DeviceExtension->IoWorkItem == NULL)
{
startRead = TRUE;
// kick off another read
DeviceExtension->IoWorkItem = IoAllocateWorkItem(DeviceExtension->PhysDeviceObject);
}
if(startRead && DeviceExtension->IoWorkItem)
{
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
IoQueueWorkItem(DeviceExtension->IoWorkItem,
USBSER_RestartNotifyReadWorkItem,
CriticalWorkQueue,
DeviceExtension);
}
else
{
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
} else {
RELEASE_SPINLOCK(DeviceExtension, &DeviceExtension->ControlLock, OldIrql);
}
NotifyCompletionErr:;
//
// Notify everyone if this is the last IRP and we aren't starting another read
//
if((InterlockedDecrement(&DeviceExtension->PendingNotifyCount) == 0))
{
UsbSerSerialDump(USBSERTRACERD, ("Notify pipe is empty\n"));
if(!startRead)
{
KeSetEvent(&DeviceExtension->PendingNotifyEvent, IO_NO_INCREMENT, FALSE);
}
}
DEBUG_LOG_PATH("exit NotifyCompletion");
return STATUS_MORE_PROCESSING_REQUIRED;
} // NotifyCompletion