/*++ Copyright (c) 1990 Microsoft Corporation Module Name: smbclass.c Abstract: SMBus Class Driver Author: Ken Reneris Environment: Notes: Revision History: 27-Feb-97 Pnp support - Bob Moore --*/ #include "smbc.h" ULONG SMBCDebug = SMB_ERRORS; // // Prototypes // NTSTATUS DriverEntry ( IN PDRIVER_OBJECT DriverObject, IN PUNICODE_STRING RegistryPath ); NTSTATUS SmbClassInitializeDevice ( IN ULONG MajorVersion, IN ULONG MinorVersion, IN PDRIVER_OBJECT DriverObject ); NTSTATUS SmbClassDeviceInitialize ( PSMB_CLASS SmbClass ); VOID SmbCUnload( IN PDRIVER_OBJECT DriverObject ); NTSTATUS SmbCInternalIoctl( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ); NTSTATUS SmbCPnpDispatch( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ); NTSTATUS SmbCPowerDispatch( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ); NTSTATUS SmbCForwardRequest( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ); #ifdef ALLOC_PRAGMA #pragma alloc_text(PAGE,DriverEntry) #pragma alloc_text(PAGE,SmbClassInitializeDevice) #pragma alloc_text(PAGE,SmbCUnload) #endif NTSTATUS DriverEntry ( IN PDRIVER_OBJECT DriverObject, IN PUNICODE_STRING RegistryPath ) { return STATUS_SUCCESS; } NTSTATUS SmbClassInitializeDevice ( IN ULONG MajorVersion, IN ULONG MinorVersion, IN PDRIVER_OBJECT DriverObject ) /*++ Routine Description: This function is called by the SM bus miniport driver/DriverEntry to perform class specific initialization Arguments: MajorVersion - Version # MinorVersion - Version # DriverObject - From miniport DriverEntry Return Value: Status --*/ { if (MajorVersion != SMB_CLASS_MAJOR_VERSION) { return STATUS_REVISION_MISMATCH; } // // Set up the device driver entry points. // DriverObject->DriverUnload = SmbCUnload; DriverObject->MajorFunction[IRP_MJ_POWER] = SmbCPowerDispatch; DriverObject->MajorFunction[IRP_MJ_PNP] = SmbCPnpDispatch; DriverObject->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] = SmbCInternalIoctl; DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = SmbCForwardRequest; DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = SmbCForwardRequest; // // Miniport will set up the AddDevice entry // return STATUS_SUCCESS; } VOID SmbCUnload( IN PDRIVER_OBJECT DriverObject ) { SmbPrint (SMB_NOTE, ("SmBCUnLoad: \n")); if (DriverObject->DeviceObject != NULL) { SmbPrint (SMB_ERROR, ("SmBCUnLoad: Unload called before all devices removed.\n")); } } NTSTATUS SmbCPowerDispatch( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ) /*++ Routine Description: This is the dispatch routine for power requests. Arguments: DeviceObject - Pointer to class device object. Irp - Pointer to the request packet. Return Value: Status is returned. --*/ { PIO_STACK_LOCATION irpStack; PSMBDATA SmbData; NTSTATUS status; SmbData = DeviceObject->DeviceExtension; // // What do we do with the irp? // PoStartNextPowerIrp( Irp ); if (SmbData->Class.LowerDeviceObject != NULL) { // // Forward the request along // IoSkipCurrentIrpStackLocation( Irp ); status = PoCallDriver( SmbData->Class.LowerDeviceObject, Irp ); } else { // // Complete the request with the current status // status = Irp->IoStatus.Status; IoCompleteRequest( Irp, IO_NO_INCREMENT ); } return status; } NTSTATUS SmbCInternalIoctl ( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ) /*++ Routine Description: This routine is the dispatch routine for internal IOCTLs. Arguments: DeviceObject - Pointer to class device object. Irp - Pointer to the request packet. Return Value: Status is returned. --*/ { PIO_STACK_LOCATION IrpSp; PSMB_REQUEST SmbReq; PSMBDATA Smb; NTSTATUS Status; // // Get a pointer to the current parameters for this request. The // information is contained in the current stack location. // Status = STATUS_NOT_SUPPORTED; IrpSp = IoGetCurrentIrpStackLocation(Irp); Smb = (PSMBDATA) DeviceObject->DeviceExtension; switch (IrpSp->Parameters.DeviceIoControl.IoControlCode) { case SMB_BUS_REQUEST: Irp->IoStatus.Information = 0; // // Verify bus request is valid // if (IrpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(SMB_REQUEST)) { // Invalid buffer length SmbPrint(SMB_NOTE, ("SmbCIoctl: Invalid bus_req length\n")); Status = STATUS_BUFFER_TOO_SMALL; Irp->IoStatus.Information = sizeof(SMB_REQUEST); break; } SmbReq = IrpSp->Parameters.DeviceIoControl.Type3InputBuffer; if (SmbReq->Protocol > SMB_MAXIMUM_PROTOCOL || SmbReq->Address > 0x7F || (SmbReq->Protocol == SMB_WRITE_BLOCK && SmbReq->BlockLength > SMB_MAX_DATA_SIZE)) { // Invalid param in request SmbPrint(SMB_NOTE, ("SmbCIoctl: Invalid bus_req\n")); break; } // // Mark request pending and queue it to the service queue // Status = STATUS_PENDING; Irp->IoStatus.Status = STATUS_PENDING; IoMarkIrpPending (Irp); SmbClassLockDevice (&Smb->Class); InsertTailList (&Smb->WorkQueue, &Irp->Tail.Overlay.ListEntry); // // Start IO if needed // SmbClassStartIo (Smb); SmbClassUnlockDevice (&Smb->Class); break; case SMB_REGISTER_ALARM_NOTIFY: // // Registry for alarm notifications // Irp->IoStatus.Information = 0; Status = SmbCRegisterAlarm (Smb, Irp); break; case SMB_DEREGISTER_ALARM_NOTIFY: // // Deregister for alarm notifications // Irp->IoStatus.Information = 0; Status = SmbCDeregisterAlarm (Smb, Irp); break; default: // Forard Irp or complete Irp without modifying it. return SmbCForwardRequest(DeviceObject, Irp); } if (Status != STATUS_PENDING) { Irp->IoStatus.Status = Status; IoCompleteRequest (Irp, IO_NO_INCREMENT); } return Status; } NTSTATUS SmbCForwardRequest( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ) /*++ Routine Description: This routine forwards the irp down the stack Arguments: DeviceObject - The target Irp - The request Return Value: NTSTATUS --*/ { NTSTATUS Status; PSMBDATA Smb = (PSMBDATA) DeviceObject->DeviceExtension; if (Smb->Class.LowerDeviceObject != NULL) { IoSkipCurrentIrpStackLocation( Irp ); Status = IoCallDriver( Smb->Class.LowerDeviceObject, Irp ); } else { Status = Irp->IoStatus.Status; IoCompleteRequest( Irp, IO_NO_INCREMENT ); } return Status; }