Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

386 lines
8.4 KiB

/*++
Copyright (c) 1990 Microsoft Corporation
Module Name:
smbcsrv.c
Abstract:
SMBus class driver service functions
Author:
Ken Reneris
Environment:
Notes:
Revision History:
--*/
#include "smbc.h"
VOID
SmbCCheckAlarmDelete (
IN PSMBDATA Smb,
IN PSMB_ALARM SmbAlarm
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE,SmbCCheckAlarmDelete)
#pragma alloc_text(PAGE,SmbCRegisterAlarm)
#pragma alloc_text(PAGE,SmbCDeregisterAlarm)
#endif
UCHAR gHexDigits [] = "0123456789ABCDEF";
NTSTATUS
SmbCRunAlarmMethodCompletionRoutine (
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN PVOID Context
)
{
SmbPrint(SMB_ALARMS, ("SmbCRunAlarmMethodCompletionRoutine: Done running Control Method. Status=0x%08x\n", Irp->IoStatus.Status));
ExFreePool (Irp->AssociatedIrp.SystemBuffer);
IoFreeIrp(Irp);
return STATUS_MORE_PROCESSING_REQUIRED;
}
VOID
SmbCRunAlarmMethod (
IN PSMB_CLASS SmbClass,
IN UCHAR Address,
IN USHORT Data
)
/*++
Routine Description:
Run _Rxx for the alarm
--*/
{
PIRP irp;
PIO_STACK_LOCATION irpSp;
PACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER inputBuffer;
SmbPrint(SMB_ALARMS, ("SmbCRunAlarmMethod: Running Control method _R%02x\n", Address));
inputBuffer = ExAllocatePoolWithTag (
NonPagedPool,
sizeof (ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER),
'AbmS'
);
RtlZeroMemory( inputBuffer, sizeof(ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER) );
inputBuffer->Signature = ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER_SIGNATURE;
inputBuffer->MethodNameAsUlong = '00Q_';
inputBuffer->MethodName[2] = gHexDigits[ Address / 16];
inputBuffer->MethodName[3] = gHexDigits[ Address % 16];
inputBuffer->IntegerArgument = Data;
irp = IoAllocateIrp (SmbClass->LowerDeviceObject->StackSize, FALSE);
if (!irp) {
return;
}
irp->AssociatedIrp.SystemBuffer = inputBuffer;
ASSERT ((IOCTL_ACPI_ASYNC_EVAL_METHOD & 0x3) == METHOD_BUFFERED);
irp->Flags = IRP_BUFFERED_IO | IRP_INPUT_OPERATION;
irp->IoStatus.Status = STATUS_NOT_SUPPORTED;
irpSp = IoGetNextIrpStackLocation( irp );
irpSp->MajorFunction = IRP_MJ_DEVICE_CONTROL;
irpSp->Parameters.DeviceIoControl.OutputBufferLength = 0;
irpSp->Parameters.DeviceIoControl.InputBufferLength = sizeof(ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER);
irpSp->Parameters.DeviceIoControl.IoControlCode = IOCTL_ACPI_ASYNC_EVAL_METHOD;
irp->UserBuffer = NULL;
IoSetCompletionRoutine(
irp,
SmbCRunAlarmMethodCompletionRoutine,
NULL, // No Context This just frees the IRP
TRUE,
TRUE,
TRUE
);
IoCallDriver(SmbClass->LowerDeviceObject, irp);
}
VOID
SmbClassAlarm (
IN PSMB_CLASS SmbClass,
IN UCHAR Address,
IN USHORT Data
)
/*++
Routine Description:
Miniport has an alarm input
--*/
{
PSMBDATA Smb;
PSMB_ALARM SmbAlarm;
PLIST_ENTRY Entry, NextEntry;
BOOLEAN AlarmRegistered = FALSE;
Smb = CONTAINING_RECORD (SmbClass, SMBDATA, Class);
ASSERT_DEVICE_LOCKED (Smb);
Entry = Smb->Alarms.Flink;
while (Entry != &Smb->Alarms) {
SmbAlarm = CONTAINING_RECORD (Entry, SMB_ALARM, Link);
//
// If notification is for this address, issue it
//
if (Address >= SmbAlarm->MinAddress && Address <= SmbAlarm->MaxAddress) {
//
// A driver has registered for this notification. Don't call the BIOS.
//
AlarmRegistered = TRUE;
//
// Raise reference count before calling notifcation function
//
SmbAlarm->Reference += 1;
ASSERT (SmbAlarm->Reference != 0);
SmbClassUnlockDevice (SmbClass);
//
// Issue notification
//
SmbAlarm->NotifyFunction (SmbAlarm->NotifyContext, Address, Data);
//
// Continue
//
SmbClassLockDevice (SmbClass);
SmbAlarm->Reference -= 1;
}
//
// Get next entry
//
NextEntry = Entry->Flink;
//
// If entry is pending delete, hand it to deleting thread
//
if (SmbAlarm->Flag & SMBC_ALARM_DELETE_PENDING) {
SmbCCheckAlarmDelete (Smb, SmbAlarm);
}
//
// Move on
//
Entry = NextEntry;
}
//
// If no one registered for this alarm, call the _Rxx control method
//
if (!AlarmRegistered) {
SmbCRunAlarmMethod (SmbClass, Address, Data);
}
}
VOID
SmbCCheckAlarmDelete (
IN PSMBDATA Smb,
IN PSMB_ALARM SmbAlarm
)
{
//
// If alarm structure is referenced, wait somemore
//
if (SmbAlarm->Reference) {
return ;
}
//
// Time to free it. Remove it from the notification list, clear
// the pending flag and set the event to let waiting threads know
// that some entry was removed
//
RemoveEntryList (&SmbAlarm->Link);
SmbAlarm->Flag &= ~SMBC_ALARM_DELETE_PENDING;
KeSetEvent (&Smb->AlarmEvent, 0, FALSE);
}
NTSTATUS
SmbCRegisterAlarm (
PSMBDATA Smb,
PIRP Irp
)
/*++
Routine Description:
Called to register for an alarm event
--*/
{
PVOID LockPtr;
PSMB_ALARM SmbAlarm, *Result;
PSMB_REGISTER_ALARM RegAlarm;
PIO_STACK_LOCATION IrpSp;
PAGED_CODE();
IrpSp = IoGetCurrentIrpStackLocation(Irp);
if (ExGetPreviousMode() != KernelMode ||
IrpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(SMB_REGISTER_ALARM) ||
IrpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(PSMB_ALARM) ) {
return STATUS_INVALID_PARAMETER;
}
RegAlarm = IrpSp->Parameters.DeviceIoControl.Type3InputBuffer;
SmbAlarm = ExAllocatePoolWithTag (
NonPagedPool,
sizeof (SMB_ALARM),
'AbmS'
);
if (!SmbAlarm) {
return STATUS_INSUFFICIENT_RESOURCES;
}
SmbAlarm->Flag = 0;
SmbAlarm->Reference = 0;
SmbAlarm->MinAddress = RegAlarm->MinAddress;
SmbAlarm->MaxAddress = RegAlarm->MaxAddress;
SmbAlarm->NotifyFunction = RegAlarm->NotifyFunction;
SmbAlarm->NotifyContext = RegAlarm->NotifyContext;
//
// Add it to the alarm notification list
//
LockPtr = MmLockPagableCodeSection(SmbCRegisterAlarm);
SmbClassLockDevice (&Smb->Class);
InsertTailList (&Smb->Alarms, &SmbAlarm->Link);
SmbClassUnlockDevice (&Smb->Class);
MmUnlockPagableImageSection(LockPtr);
//
// Return value caller needs to deregister with
//
Result = (PSMB_ALARM *) Irp->UserBuffer;
*Result = SmbAlarm;
Irp->IoStatus.Information = sizeof(PSMB_ALARM);
return STATUS_SUCCESS;
}
NTSTATUS
SmbCDeregisterAlarm (
PSMBDATA Smb,
PIRP Irp
)
/*++
Routine Description:
Called to register for an alarm event
--*/
{
PVOID LockPtr;
PSMB_ALARM SmbAlarm;
PIO_STACK_LOCATION IrpSp;
PAGED_CODE();
IrpSp = IoGetCurrentIrpStackLocation(Irp);
if (ExGetPreviousMode() != KernelMode ||
IrpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(PSMB_ALARM) ) {
return STATUS_INVALID_PARAMETER;
}
SmbAlarm = * (PSMB_ALARM *) IrpSp->Parameters.DeviceIoControl.Type3InputBuffer;
LockPtr = MmLockPagableCodeSection(SmbCDeregisterAlarm);
SmbClassLockDevice (&Smb->Class);
//
// Flag alarm structure as delete pending
//
SmbAlarm->Flag |= SMBC_ALARM_DELETE_PENDING;
//
// While delete is pending wait
//
while (SmbAlarm->Flag & SMBC_ALARM_DELETE_PENDING) {
//
// Issue bogus alarm to generate freeing
//
KeResetEvent (&Smb->AlarmEvent);
SmbClassAlarm (&Smb->Class, 0xFF, 0);
//
// Wait for alarm structure to get freed, then check if it
// was ours
//
SmbClassUnlockDevice (&Smb->Class);
KeWaitForSingleObject (
&Smb->AlarmEvent,
Executive,
KernelMode,
FALSE,
NULL
);
SmbClassLockDevice (&Smb->Class);
}
//
// It's been removed, free the memory
//
SmbClassUnlockDevice (&Smb->Class);
MmUnlockPagableImageSection(LockPtr);
ExFreePool (SmbAlarm);
return STATUS_SUCCESS;
}