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.
 
 
 
 
 
 

9648 lines
274 KiB

/*++
Copyright (c) 1990-2000 Microsoft Corporation
Module Name:
videoprt.c
Abstract:
This is the NT Video port driver.
Author:
Andre Vachon (andreva) 18-Dec-1991
Environment:
kernel mode only
Notes:
This module is a driver which implements OS dependant functions on the
behalf of the video drivers
Revision History:
--*/
#define INITGUID
#include "videoprt.h"
NTSTATUS
DriverEntry(
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT,DriverEntry)
#pragma alloc_text(PAGE,pVideoPortCreateDeviceName)
#pragma alloc_text(PAGE,pVideoPortDispatch)
#pragma alloc_text(PAGE,VideoPortFreeDeviceBase)
#pragma alloc_text(PAGE,pVideoPortFreeDeviceBase)
#pragma alloc_text(PAGE,pVideoPortGetDeviceBase)
#pragma alloc_text(PAGE,VideoPortGetDeviceBase)
#pragma alloc_text(PAGE,pVideoPortGetDeviceDataRegistry)
#pragma alloc_text(PAGE,VideoPortGetDeviceData)
#pragma alloc_text(PAGE,pVideoPortGetRegistryCallback)
#pragma alloc_text(PAGE,VideoPortGetRegistryParameters)
#pragma alloc_text(PAGE,pVPInit)
#pragma alloc_text(PAGE,VpInitializeBusCallback)
#pragma alloc_text(PAGE,VpDriverUnload)
#pragma alloc_text(PAGE,VpAddDevice)
#pragma alloc_text(PAGE,VpCreateDevice)
#pragma alloc_text(PAGE,VideoPortInitialize)
#pragma alloc_text(PAGE,UpdateRegValue)
#pragma alloc_text(PAGE,VideoPortLegacyFindAdapter)
#pragma alloc_text(PAGE,VideoPortFindAdapter2)
#pragma alloc_text(PAGE,VideoPortFindAdapter)
#pragma alloc_text(PAGE,pVideoPortMapToNtStatus)
#pragma alloc_text(PAGE,pVideoPortMapUserPhysicalMem)
#pragma alloc_text(PAGE,VideoPortMapMemory)
#pragma alloc_text(PAGE,VideoPortMapBankedMemory)
#pragma alloc_text(PAGE,VideoPortAllocateBuffer)
#pragma alloc_text(PAGE,VideoPortReleaseBuffer)
#pragma alloc_text(PAGE,VideoPortScanRom)
#pragma alloc_text(PAGE,VideoPortSetRegistryParameters)
#pragma alloc_text(PAGE,VideoPortUnmapMemory)
#pragma alloc_text(PAGE,VpEnableDisplay)
#pragma alloc_text(PAGE,VpWin32kCallout)
#pragma alloc_text(PAGE,VpAllowFindAdapter)
#pragma alloc_text(PAGE,VpTranslateBusAddress)
#if DBG
#pragma alloc_text(PAGE,BuildRequirements)
#pragma alloc_text(PAGE,DumpRequirements)
#pragma alloc_text(PAGE,DumpResourceList)
#pragma alloc_text(PAGE,DumpUnicodeString)
#endif
#pragma alloc_text(PAGE,VideoPortCreateSecondaryDisplay)
#pragma alloc_text(PAGE,VideoPortQueryServices)
#pragma alloc_text(PAGE,VpInterfaceDefaultReference)
#pragma alloc_text(PAGE,VpInterfaceDefaultDereference)
#pragma alloc_text(PAGE,VpEnableAdapterInterface)
#pragma alloc_text(PAGE,VpDisableAdapterInterface)
//
// VideoPortQueryPerformanceCounter() cannot be pageable.
//
#endif
NTSTATUS
DriverEntry(
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath
)
/*++
Routine Description:
Temporary entry point needed to initialize the video port driver.
Arguments:
DriverObject - Pointer to the driver object created by the system.
Return Value:
STATUS_SUCCESS
--*/
{
UNREFERENCED_PARAMETER(DriverObject);
ASSERT(0);
//
//
//
// WARNING !!!
//
// This function is never called because we are loaded as a DLL by other video drivers !
//
//
//
//
//
//
// We always return STATUS_SUCCESS because otherwise no video miniport
// driver will be able to call us.
//
return STATUS_SUCCESS;
} // end DriverEntry()
NTSTATUS
pVideoPortCreateDeviceName(
PWSTR DeviceString,
ULONG DeviceNumber,
PUNICODE_STRING UnicodeString,
PWCHAR UnicodeBuffer
)
/*++
Routine Description:
Helper function that does string manipulation to create a device name
--*/
{
WCHAR ntNumberBuffer[STRING_LENGTH];
UNICODE_STRING ntNumberUnicodeString;
//
// Create the name buffer
//
UnicodeString->Buffer = UnicodeBuffer;
UnicodeString->Length = 0;
UnicodeString->MaximumLength = STRING_LENGTH;
//
// Create the miniport driver object name.
//
ntNumberUnicodeString.Buffer = ntNumberBuffer;
ntNumberUnicodeString.Length = 0;
ntNumberUnicodeString.MaximumLength = STRING_LENGTH;
if (NT_SUCCESS(RtlIntegerToUnicodeString(DeviceNumber,
10,
&ntNumberUnicodeString))) {
if (NT_SUCCESS(RtlAppendUnicodeToString(UnicodeString,
DeviceString))) {
if (NT_SUCCESS(RtlAppendUnicodeStringToString(UnicodeString,
&ntNumberUnicodeString))) {
UnicodeString->MaximumLength = (USHORT)
(UnicodeString->Length + sizeof(UNICODE_NULL));
return STATUS_SUCCESS;
}
}
}
return STATUS_INSUFFICIENT_RESOURCES;
} // pVideoPortCreateDeviceName()
VOID
VideoPortDebugPrint(
VIDEO_DEBUG_LEVEL DebugPrintLevel,
PCHAR DebugMessage,
...
)
/*++
Routine Description:
This routine allows the miniport drivers (as well as the port driver) to
display error messages to the debug port when running in the debug
environment.
When running a non-debugged system, all references to this call are
eliminated by the compiler.
Arguments:
DebugPrintLevel - Debug print level being:
0 = Error Level (always prints no matter what on a checked build)
1 = Warning Level (prints only when VIDEO filter is on for this level or higher)
2 = Trace Level (see above)
3 = Info Level (see above)
Return Value:
None.
--*/
{
va_list ap;
va_start(ap, DebugMessage);
if (VideoDebugLevel && (VideoDebugLevel >= (ULONG)DebugPrintLevel)) {
vDbgPrintEx(DPFLTR_VIDEO_ID, 0, DebugMessage, ap);
} else {
vDbgPrintEx(DPFLTR_VIDEO_ID, DebugPrintLevel, DebugMessage, ap);
}
va_end(ap);
} // VideoPortDebugPrint()
VOID
VpEnableDisplay(
PFDO_EXTENSION fdoExtension,
BOOLEAN bState
)
/*++
Routine Description:
This routine enables/disables the current display so that we can execut
the drivers FindAdapter code.
Arugments:
bState - Should the display be enabled or disabled
Returns:
none
Notes:
The device lock for the passed in fdoExtension must be held
before this routine is called!
--*/
{
if (!InbvCheckDisplayOwnership()) {
VIDEO_WIN32K_CALLBACKS_PARAMS calloutParams;
//
// The system is up and running. Notify GDI to enable/disable
// the current display.
//
calloutParams.CalloutType = VideoFindAdapterCallout;
calloutParams.Param = bState;
RELEASE_DEVICE_LOCK(fdoExtension);
VpWin32kCallout(&calloutParams);
ACQUIRE_DEVICE_LOCK(fdoExtension);
} else {
//
// The boot driver is still in control. Modify the state of the
// boot driver.
//
InbvEnableBootDriver(bState);
}
}
VOID
VpWin32kCallout(
PVIDEO_WIN32K_CALLBACKS_PARAMS calloutParams
)
/*++
Routine Description:
This routine makes a callout into win32k. It attaches to csrss
to guarantee that win32k is in the address space on hydra machines.
Arguments:
calloutParams - a pointer to the callout struture.
Returns:
none.
--*/
{
if (Win32kCallout && CsrProcess) {
KeAttachProcess(PEProcessToPKProcess(CsrProcess));
(*Win32kCallout)(calloutParams);
KeDetachProcess();
}
}
BOOLEAN
VpAllowFindAdapter(
PFDO_EXTENSION fdoExtension
)
/*++
Routine Description:
Determine if we allow this device to be used if part of a multi-function
board.
Arguments;
fdoExtension - The device extenstion for the object in question.
Returns:
TRUE if the device is allowed as part of a multi-function board.
FALSE otherwise.
--*/
{
BOOLEAN bRet = TRUE;
if ((fdoExtension->AdapterInterfaceType == PCIBus) &&
((fdoExtension->Flags & PNP_ENABLED) == PNP_ENABLED)) {
PCI_COMMON_CONFIG ConfigSpace;
if (PCI_COMMON_HDR_LENGTH ==
VideoPortGetBusData(fdoExtension->HwDeviceExtension,
PCIConfiguration,
0,
&ConfigSpace,
0,
PCI_COMMON_HDR_LENGTH)) {
if (PCI_MULTIFUNCTION_DEVICE(&ConfigSpace)) {
ULONG MultiFunc = 0;
//
// This is a multi-function device. So only allow
// HwInitialize if the INF indicated we'd be multi-function.
//
VideoPortGetRegistryParameters(fdoExtension->HwDeviceExtension,
L"MultiFunctionSupported",
FALSE,
VpRegistryCallback,
&MultiFunc);
if (MultiFunc == 0) {
pVideoDebugPrint((Warn, "VIDEOPRT: Multifunction board not allowed to start\n"));
bRet = FALSE;
}
}
}
}
return bRet;
}
NTSTATUS
pVideoPortDispatch(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
This routine is the main dispatch routine for the video port driver.
It accepts an I/O Request Packet, transforms it to a video Request
Packet, and forwards it to the appropriate miniport dispatch routine.
Upon returning, it completes the request and return the appropriate
status value.
Arguments:
DeviceObject - Pointer to the device object of the miniport driver to
which the request must be sent.
Irp - Pointer to the request packet representing the I/O request.
Return Value:
The function value os the status of the operation.
--*/
{
PFDO_EXTENSION combinedExtension;
PFDO_EXTENSION fdoExtension;
PVOID HwDeviceExtension;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
PCHILD_PDO_EXTENSION pdoExtension = NULL;
PIO_STACK_LOCATION irpStack;
PVOID ioBuffer;
ULONG inputBufferLength;
ULONG outputBufferLength;
PSTATUS_BLOCK statusBlock;
NTSTATUS finalStatus = -1;
ULONG ioControlCode;
VIDEO_REQUEST_PACKET vrp;
NTSTATUS status;
PBACKLIGHT_STATUS pVpBacklightStatus = &VpBacklightStatus;
ULONG ulACPIMethodParam1;
ULONG ulACPIMethodParam2;
HANDLE hkRegistry;
UNICODE_STRING UnicodeString;
OBJECT_ATTRIBUTES ObjectAttributes;
#if _X86_
PUCHAR BiosDataBuffer;
#define BIOS_DATA_SIZE 256
#endif
//
// Get pointer to the port driver's device extension.
//
combinedExtension = DeviceObject->DeviceExtension;
HwDeviceExtension = combinedExtension->HwDeviceExtension;
//
// Get pointer to the port driver's device extension.
//
if (IS_PDO(DeviceObject->DeviceExtension)) {
pdoExtension = DeviceObject->DeviceExtension;
fdoExtension = pdoExtension->pFdoExtension;
DoSpecificExtension = (PDEVICE_SPECIFIC_EXTENSION)(fdoExtension + 1);
} else if (IS_FDO(DeviceObject->DeviceExtension)) {
fdoExtension = DeviceObject->DeviceExtension;
DoSpecificExtension = (PDEVICE_SPECIFIC_EXTENSION)(fdoExtension + 1);
} else {
DoSpecificExtension = DeviceObject->DeviceExtension;
fdoExtension = DoSpecificExtension->pFdoExtension;
combinedExtension = fdoExtension;
}
//
// Get a pointer to the current location in the Irp. This is where
// the function codes and parameters are located.
//
irpStack = IoGetCurrentIrpStackLocation(Irp);
//
// Get the pointer to the status buffer.
// Assume SUCCESS for now.
//
statusBlock = (PSTATUS_BLOCK) &Irp->IoStatus;
//
// Synchronize execution of the dispatch routine by acquiring the device
// event object. This ensures all request are serialized.
// The synchronization must be done explicitly because the functions
// executed in the dispatch routine use system services that cannot
// be executed in the start I/O routine.
//
// The synchronization is done on the miniport's event so as not to
// block commands coming in for another device.
//
#if REMOVE_LOCK_ENABLED
status = IoAcquireRemoveLock(&combinedExtension->RemoveLock, Irp);
if (NT_SUCCESS(status) == FALSE) {
ASSERT(FALSE);
Irp->IoStatus.Information = 0;
Irp->IoStatus.Status = status;
IoCompleteRequest(Irp, IO_VIDEO_INCREMENT);
return status;
}
#endif
ACQUIRE_DEVICE_LOCK(combinedExtension);
//
// Get the requestor mode.
//
combinedExtension->CurrentIrpRequestorMode = Irp->RequestorMode;
ASSERT(irpStack->MajorFunction != IRP_MJ_PNP);
ASSERT(irpStack->MajorFunction != IRP_MJ_POWER);
//
// Case on the function being requested.
// If the function is operating specific, intercept the operation and
// perform directly. Otherwise, pass it on to the appropriate miniport.
//
switch (irpStack->MajorFunction) {
//
// Called by the display driver *or a user-mode application*
// to get exclusive access to the device.
// This access is given by the I/O system (based on a bit set during the
// IoCreateDevice() call).
//
case IRP_MJ_CREATE:
pVideoDebugPrint((Trace, "VIDEOPRT: IRP_MJ_CREATE\n"));
if (Irp->RequestorMode == UserMode)
{
statusBlock->Status = STATUS_SUCCESS;
break;
}
//
// Don't let an old driver start during the upgrade
//
if (fdoExtension->Flags & UPGRADE_FAIL_START)
{
statusBlock->Status = STATUS_ACCESS_DENIED;
break;
}
//
// Don't allow create's on children, unless they are a monitor
// which will receive calls from the display driver
//
if (IS_PDO(pdoExtension)) {
pVideoDebugPrint((Error, "VIDEOPRT: Create's on children devices not allowed.\n"));
statusBlock->Status = STATUS_ACCESS_DENIED;
break;
}
//
// Special hack to succeed on Attach, but do nothing ...
// If the device is already opened, do nothing.
//
if ((irpStack->Parameters.Create.SecurityContext->DesiredAccess ==
FILE_READ_ATTRIBUTES) ||
(DoSpecificExtension->DeviceOpened)) {
statusBlock->Information = FILE_OPEN;
statusBlock->Status = STATUS_SUCCESS;
break;
}
//
// If we haven't already done so, create our watchdog recovery event
//
if (VpThreadStuckEvent == NULL) {
UNICODE_STRING strName;
RtlInitUnicodeString(&strName, L"\\BaseNamedObjects\\StuckThreadEvent");
VpThreadStuckEvent = IoCreateNotificationEvent(&strName, &VpThreadStuckEventHandle);
if (VpThreadStuckEvent) {
KeClearEvent(VpThreadStuckEvent);
}
}
//
// Get out of the special setup mode that we may be in
//
VpSetupType = 0;
//
// If hwInitialize has been called then the system is done
// initializing and we are transitioning into gui mode.
//
VpSystemInitialized = TRUE;
//
// Now perform basic initialization to allow the Windows display
// driver to set up the device appropriately.
//
statusBlock->Information = FILE_OPEN;
//
// If the address space has not been set up in the server yet, do it now.
// NOTE: no need to map in IO ports since the server has IOPL
//
if (CsrProcess == NULL)
{
CsrProcess = PsGetCurrentProcess();
ObReferenceObject(CsrProcess);
}
pVideoPortInitializeInt10(fdoExtension);
if (!IsMirrorDriver(fdoExtension)) {
//
// Tell the kernel we are now taking ownership the display, but
// only do so if this is not a mirroring driver.
//
InbvNotifyDisplayOwnershipLost(pVideoPortResetDisplay);
}
if ((fdoExtension->Flags & FINDADAPTER_SUCCEEDED) == 0) {
statusBlock->Status = STATUS_DEVICE_CONFIGURATION_ERROR;
} else if ((fdoExtension->HwInitStatus == HwInitNotCalled) &&
(fdoExtension->HwInitialize(fdoExtension->HwDeviceExtension) == FALSE))
{
statusBlock->Status = STATUS_DEVICE_CONFIGURATION_ERROR;
fdoExtension->HwInitStatus = HwInitFailed;
} else if (fdoExtension->HwInitStatus == HwInitFailed) {
statusBlock->Status = STATUS_DEVICE_CONFIGURATION_ERROR;
} else {
fdoExtension->HwInitStatus = HwInitSucceeded;
statusBlock->Status = STATUS_SUCCESS;
}
//
// Mark the device as opened so we will fail future opens.
//
DoSpecificExtension->DeviceOpened = TRUE;
//
// We don't want GDI to use any drivers other than display
// or boot drivers during upgrade setup.
//
if (fdoExtension->Flags & UPGRADE_FAIL_HWINIT) {
statusBlock->Status = STATUS_ACCESS_DENIED;
}
break;
//
// Called when the display driver wishes to give up it's handle to the
// device.
//
case IRP_MJ_CLOSE:
pVideoDebugPrint((Trace, "VIDEOPRT: IRP_MJ_CLOSE\n"));
statusBlock->Status = STATUS_SUCCESS;
break;
//
// Device Controls are specific functions for the driver.
// First check for calls that must be intercepted and hidden from the
// miniport driver. These calls are hidden for simplicity.
// The other control functions are passed down to the miniport after
// the request structure has been filled out properly.
//
case IRP_MJ_DEVICE_CONTROL:
//
// Get the pointer to the input/output buffer and it's length
//
ioBuffer = Irp->AssociatedIrp.SystemBuffer;
inputBufferLength = irpStack->Parameters.DeviceIoControl.InputBufferLength;
outputBufferLength = irpStack->Parameters.DeviceIoControl.OutputBufferLength;
ioControlCode = irpStack->Parameters.DeviceIoControl.IoControlCode;
if (Irp->RequestorMode == UserMode)
{
statusBlock->Information = 0;
if ((ioControlCode != IOCTL_VIDEO_QUERY_SUPPORTED_BRIGHTNESS) &&
(ioControlCode != IOCTL_VIDEO_QUERY_DISPLAY_BRIGHTNESS) &&
(ioControlCode != IOCTL_VIDEO_SET_DISPLAY_BRIGHTNESS))
{
statusBlock->Status = STATUS_ACCESS_DENIED;
break;
}
}
#ifdef IOCTL_VIDEO_USE_DEVICE_IN_SESSION
//
// Validate session usage
//
// Note: Some IOCTLs are acceptable to call from non-console sessions
// These include all private IOCTLs which is completely under
// the drivers' control and may happen even when the device is
// disabled. IOCTL_VIDEO_REGISTER_VDM is also allowed since
// it doesn't access any hardware.
//
if ((ioControlCode & CTL_CODE(0x8000, 0x800, METHOD_BUFFERED, FILE_ANY_ACCESS)) == 0 &&
ioControlCode != IOCTL_VIDEO_REGISTER_VDM)
{
if (DoSpecificExtension->SessionId != VIDEO_DEVICE_INVALID_SESSION &&
PsGetCurrentProcessSessionId() != DoSpecificExtension->SessionId)
{
DbgPrint("VIDEOPRT: Trying to use display device in sessions %lu and %lu.\n",
DoSpecificExtension->SessionId,
PsGetCurrentProcessSessionId());
//
// We will also allow several other IOCTLs to be called from
// a non-console session.
// For simplicity we allow all IOCTLs for which the driver
// provides support with the exception of
// IOCTL_VIDEO_ENABLE_VDM.
//
if ((ioControlCode &
CTL_CODE(0x7fff, 0x7ff, METHOD_BUFFERED, FILE_ANY_ACCESS)) <=
IOCTL_VIDEO_USE_DEVICE_IN_SESSION)
{
ASSERT(FALSE);
}
else
{
DbgPrint("VIDEOPRT: Cross session use is acceptable in this case.\n");
}
}
}
#endif IOCTL_VIDEO_USE_DEVICE_IN_SESSION
//
// Enabling or disabling the VDM is done only by the port driver.
//
if (ioControlCode == IOCTL_VIDEO_REGISTER_VDM) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_REGISTER_VDM\n"));
ASSERT(IS_PDO(pdoExtension) == FALSE);
statusBlock->Status = pVideoPortRegisterVDM(fdoExtension,
(PVIDEO_VDM) ioBuffer,
inputBufferLength,
(PVIDEO_REGISTER_VDM) ioBuffer,
outputBufferLength,
&statusBlock->Information);
} else if (ioControlCode == IOCTL_VIDEO_DISABLE_VDM) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_DISABLE_VDM"));
ASSERT(IS_PDO(pdoExtension) == FALSE);
statusBlock->Status = pVideoPortEnableVDM(fdoExtension,
FALSE,
(PVIDEO_VDM) ioBuffer,
inputBufferLength);
} else if ((ioControlCode == IOCTL_VIDEO_SET_OUTPUT_DEVICE_POWER_STATE) ||
(ioControlCode == IOCTL_VIDEO_GET_OUTPUT_DEVICE_POWER_STATE)) {
//
// This handles the case where ntuser has signalled that it wants
// to change or detect the power state.
//
PCHILD_PDO_EXTENSION pChild;
PPOWER_BLOCK powerCtx = NULL;
ULONG count = 0;
UCHAR mFnc =
(ioControlCode == IOCTL_VIDEO_SET_OUTPUT_DEVICE_POWER_STATE) ?
IRP_MN_SET_POWER : IRP_MN_QUERY_POWER ;
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_%s_OUTPUT_DEVICE_POWER_STATE\n",
ioControlCode == IOCTL_VIDEO_SET_OUTPUT_DEVICE_POWER_STATE ? "SET" : "GET"));
//
// USER wants to set the power on the monitor, not the card.
// So let's find our child monitor and send it the power management
// function.
// If there is no power manageable monitor, then we can just fail
// the request right here.
//
ASSERT(IS_PDO(pdoExtension) == FALSE);
if (fdoExtension->ChildPdoList)
{
//
// Count the number of monitor devices so that the IRP will
// be completed properly after a power IRP has been requested
// for each.
//
for (pChild = fdoExtension->ChildPdoList;
pChild;
pChild = pChild->NextChild) {
if (pChild->VideoChildDescriptor->Type == Monitor)
count++;
}
for (pChild = fdoExtension->ChildPdoList;
pChild;
pChild = pChild->NextChild)
{
if (pChild->VideoChildDescriptor->Type == Monitor)
{
count--;
//
// Allocate memory for the IRP context.
//
powerCtx = ExAllocatePoolWithTag(NonPagedPool,
sizeof(POWER_BLOCK),
VP_TAG);
if (!powerCtx) {
pVideoDebugPrint ((Error, "VIDEOPRT: No memory for power context.\n"));
finalStatus = statusBlock->Status = STATUS_INSUFFICIENT_RESOURCES;
break;
}
powerCtx->Irp = Irp;
powerCtx->FinalFlag = FALSE;
//
// Since there is a least one monitor that a power IRP
// will be requested for, mark this IRP as pending as
// it will undoubtedly be returned with STATUS_PENDING.
//
IoMarkIrpPending(Irp);
if (count == 0) {
powerCtx->FinalFlag = TRUE;
}
//
// Since PoRequestPowerIrp always returns STATUS_PENDING
// if the IRP was sent, a double completion here is
// impossible as at the bottom of this function
// IoCompleteRequest will not be called if status is
// STATUS_PENDING
//
finalStatus =
PoRequestPowerIrp(pChild->ChildDeviceObject,
mFnc,
*(PPOWER_STATE)(ioBuffer),
pVideoPortPowerCompletionIoctl,
powerCtx,
NULL);
if (finalStatus != STATUS_PENDING) {
break;
}
}
}
}
} else if ((ioControlCode == IOCTL_VIDEO_SET_POWER_MANAGEMENT) ||
(ioControlCode == IOCTL_VIDEO_GET_POWER_MANAGEMENT)) {
statusBlock->Status = STATUS_SUCCESS;
} else if (ioControlCode == IOCTL_VIDEO_ENUM_MONITOR_PDO) {
ULONG szMonitorDevices;
PVIDEO_MONITOR_DEVICE pMonitorDevices = NULL, pMD;
PCHILD_PDO_EXTENSION pChildDeviceExtension;
PDEVICE_OBJECT pdo;
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_ENUM_MONITOR_PDO\n"));
szMonitorDevices = (fdoExtension->ChildPdoNumber+1)*sizeof(VIDEO_MONITOR_DEVICE);
pMonitorDevices = ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
szMonitorDevices,
VP_TAG);
if (pMonitorDevices == NULL) {
statusBlock->Status = STATUS_INSUFFICIENT_RESOURCES;
} else {
RtlZeroMemory(pMonitorDevices, szMonitorDevices);
//
// Walk our chain of children, and store them in the relations array.
//
pChildDeviceExtension = fdoExtension->ChildPdoList;
pMD = pMonitorDevices;
while (pChildDeviceExtension) {
if (pChildDeviceExtension->bIsEnumerated &&
pChildDeviceExtension->VideoChildDescriptor->Type == Monitor
)
{
ULONG UId, flag = VIDEO_CHILD_ACTIVE;
//
// Refcount the ChildDeviceObject.
//
ObReferenceObject(pChildDeviceExtension->ChildDeviceObject);
UId = pChildDeviceExtension->VideoChildDescriptor->UId;
if (!NT_SUCCESS
(pVideoMiniDeviceIoControl(DeviceObject,
IOCTL_VIDEO_GET_CHILD_STATE,
&UId,
sizeof(ULONG),
&flag,
sizeof(ULONG) ) )
)
{
//
// If driver driver doesn't handle IOCTL_VIDEO_GET_CHILD_STATE, set to default value
//
flag = pCheckActiveMonitor(pChildDeviceExtension) ? VIDEO_CHILD_ACTIVE : 0;
}
pMD->flag = flag;
pMD->pdo = pChildDeviceExtension->ChildDeviceObject;
pMD->HwID = pChildDeviceExtension->VideoChildDescriptor->UId;
pMD++;
}
pChildDeviceExtension = pChildDeviceExtension->NextChild;
}
//
// Return the information to GDI. The array terminated by a zero unit.
//
*((PVOID *)ioBuffer) = pMonitorDevices;
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = sizeof(PVOID);
}
} else if (ioControlCode == IOCTL_VIDEO_INIT_WIN32K_CALLBACKS) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_INIT_WIN32K_CALLBACKS\n"));
if (DoSpecificExtension->PhysDisp == NULL)
{
DoSpecificExtension->PhysDisp = ((PVIDEO_WIN32K_CALLBACKS)(ioBuffer))->PhysDisp;
}
if (Win32kCallout == NULL)
{
Win32kCallout = ((PVIDEO_WIN32K_CALLBACKS)(ioBuffer))->Callout;
}
((PVIDEO_WIN32K_CALLBACKS)ioBuffer)->bACPI = DoSpecificExtension->bACPI;
((PVIDEO_WIN32K_CALLBACKS)ioBuffer)->pPhysDeviceObject = fdoExtension->PhysicalDeviceObject;
((PVIDEO_WIN32K_CALLBACKS)ioBuffer)->DualviewFlags = DoSpecificExtension->DualviewFlags;
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = sizeof(VIDEO_WIN32K_CALLBACKS);
} else if (ioControlCode == IOCTL_VIDEO_IS_VGA_DEVICE) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_IS_VGA_DEVICE\n"));
*((PBOOLEAN)(ioBuffer)) = (BOOLEAN)(DeviceObject == DeviceOwningVga);
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = sizeof(BOOLEAN);
} else if (ioControlCode == IOCTL_VIDEO_PREPARE_FOR_EARECOVERY) {
KBUGCHECK_SECONDARY_DUMP_DATA DumpData;
ULONG ulDumpSize = 0;
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_PREPARE_FOR_EARECOVERY\n"));
//
// Save basic minidump on the disk (if we have any problem later
// during recovery we still will have something to work with)
//
if (VpDump) {
ulDumpSize = min((ULONG)((PMEMORY_DUMP)VpDump)->Header.RequiredDumpSpace.QuadPart,
TRIAGE_DUMP_SIZE);
pVpWriteFile(L"\\SystemRoot\\MEMORY.DMP",
VpDump,
ulDumpSize);
}
//
// As all the display devices to go into a mode where VGA works.
//
pVpGeneralBugcheckHandler(&DumpData);
if (VpDump) {
ULONG ulSize;
ULONG BugcheckDataSize = (VpBugcheckDeviceObject) ?
((PFDO_EXTENSION)VpBugcheckDeviceObject->DeviceExtension)->BugcheckDataSize :
0;
//
// Dump file created already so just add driver specific data
//
ulSize = pVpAppendSecondaryMinidumpData(VpBugcheckData,
BugcheckDataSize,
VpDump);
if (ulSize > ulDumpSize) {
//
// Write the data to disk
//
pVpWriteFile(L"\\SystemRoot\\MEMORY.DMP",
VpDump,
ulSize);
}
if (VpDump) {
ExFreePool(VpDump);
VpDump = NULL;
}
}
pVideoPortResetDisplay(80,25);
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = 0;
#ifdef IOCTL_VIDEO_USE_DEVICE_IN_SESSION
} else if (ioControlCode == IOCTL_VIDEO_USE_DEVICE_IN_SESSION) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_USE_DEVICE_IN_SESSION\n"));
if (((PVIDEO_DEVICE_SESSION_STATUS)ioBuffer)->bEnable)
{
if (DoSpecificExtension->SessionId == VIDEO_DEVICE_INVALID_SESSION)
{
DoSpecificExtension->SessionId = PsGetCurrentProcessSessionId();
((PVIDEO_DEVICE_SESSION_STATUS)ioBuffer)->bSuccess = TRUE;
}
else
{
((PVIDEO_DEVICE_SESSION_STATUS)ioBuffer)->bSuccess = FALSE;
}
}
else
{
if (DoSpecificExtension->SessionId == PsGetCurrentProcessSessionId())
{
DoSpecificExtension->SessionId = VIDEO_DEVICE_INVALID_SESSION;
((PVIDEO_DEVICE_SESSION_STATUS)ioBuffer)->bSuccess = TRUE;
}
else
{
((PVIDEO_DEVICE_SESSION_STATUS)ioBuffer)->bSuccess = FALSE;
}
}
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = sizeof(VIDEO_DEVICE_SESSION_STATUS);
#endif IOCTL_VIDEO_USE_DEVICE_IN_SESSION
//
// The following three IOCTLs support LCD backlight control.
//
} else if (ioControlCode == IOCTL_VIDEO_QUERY_SUPPORTED_BRIGHTNESS) {
//
// Note: this IOCTL must be called before:
// IOCTL_VIDEO_SET_DISPLAY_BRIGHTNESS
//
// This IOCTL will query the brightness capabilities
// of the BIOS.
//
//
// We expect a buffer size of 256 bytes.
//
ULONG ulNumReturnedLevels = 0;
PDEVICE_OBJECT AttachedDevice = NULL;
if (outputBufferLength < 256)
{
statusBlock->Status = ERROR_INSUFFICIENT_BUFFER;
break;
}
//
// Call VpQueryBacklightLevels to obtain the list of supported levels.
//
if (LCDPanelDevice) {
AttachedDevice = IoGetAttachedDeviceReference(LCDPanelDevice);
}
if (AttachedDevice) {
statusBlock->Status = VpQueryBacklightLevels(
AttachedDevice,
(PUCHAR) ioBuffer,
&ulNumReturnedLevels);
ObDereferenceObject(AttachedDevice);
}
else {
statusBlock->Status = ERROR_INVALID_PARAMETER;
}
if (!NT_SUCCESS(statusBlock->Status)) {
break;
}
//
// The new API is being used. If we haven't already read the applicable
// levels from the registry (per pVpInit) then write the keys to the
// registry and update pVpBacklightStatus as necessary.
//
if ((pVpBacklightStatus->bNewAPISupported == FALSE) ||
(pVpBacklightStatus->bACBrightnessInRegistry == FALSE) ||
(pVpBacklightStatus->bDCBrightnessInRegistry == FALSE)) {
// ZwCreateKey
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet\\"
L"Control\\Backlight");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
statusBlock->Status = ZwCreateKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes,
0L,
NULL,
REG_OPTION_NON_VOLATILE,
NULL);
if (NT_SUCCESS(statusBlock->Status)) {
ULONG ulRegData = 1;
PVOID pvData;
// NewAPISupported
RtlInitUnicodeString(
&UnicodeString,
L"NewAPISupported"
);
pvData = &ulRegData;
statusBlock->Status = ZwSetValueKey(
hkRegistry,
&UnicodeString,
0,
REG_DWORD,
pvData,
4
);
if (NT_SUCCESS(statusBlock->Status)) {
pVpBacklightStatus->bNewAPISupported = TRUE;
}
// ACBacklightLevel, use BIOS default value
RtlInitUnicodeString(
&UnicodeString,
L"ACBacklightLevel"
);
pVpBacklightStatus->ucACBrightness = pVpBacklightStatus->ucBIOSDefaultAC;
ulRegData = (UCHAR) pVpBacklightStatus->ucACBrightness;
pvData = &ulRegData;
statusBlock->Status = ZwSetValueKey(
hkRegistry,
&UnicodeString,
0,
REG_DWORD,
pvData,
4
);
if (NT_SUCCESS(statusBlock->Status)) {
pVpBacklightStatus->bACBrightnessInRegistry = TRUE;
}
// DCBacklightLevel, use BIOS default value
RtlInitUnicodeString(
&UnicodeString,
L"DCBacklightLevel"
);
pVpBacklightStatus->ucDCBrightness = pVpBacklightStatus->ucBIOSDefaultDC;
ulRegData = (UCHAR) pVpBacklightStatus->ucDCBrightness;
pvData = &ulRegData;
statusBlock->Status = ZwSetValueKey(
hkRegistry,
&UnicodeString,
0,
REG_DWORD,
pvData,
4
);
if (NT_SUCCESS(statusBlock->Status)) {
pVpBacklightStatus->bDCBrightnessInRegistry = TRUE;
}
ZwClose(hkRegistry);
pVpBacklightStatus->bACBrightnessKnown = TRUE;
pVpBacklightStatus->bDCBrightnessKnown = TRUE;
}
}
pVpBacklightStatus->bQuerySupportedBrightnessCalled = TRUE;
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = ulNumReturnedLevels;
} else if (ioControlCode == IOCTL_VIDEO_QUERY_DISPLAY_BRIGHTNESS) {
//
// If the backlight level is known for the display policy queried,
// it will be reported.
//
// No ACPI methods are used here. _BCL will be used in
// IOCTL_VIDEO_QUERY_SUPPORTED_BRIGHTNESS and _BCM will
// be used in IOCTL_VIDEO_SET_DISPLAY_BRIGHTNESS.
//
PDISPLAY_BRIGHTNESS pDisplayBrightness;
if (outputBufferLength < sizeof(DISPLAY_BRIGHTNESS))
{
statusBlock->Status = ERROR_INSUFFICIENT_BUFFER;
break;
}
pDisplayBrightness = (PDISPLAY_BRIGHTNESS) ioBuffer;
//
// Report AC value if available.
//
// Note: Value ~should~ be known, even if we need to use
// AC default.
//
if ((pVpBacklightStatus->bACBrightnessKnown == FALSE) &&
(pVpBacklightStatus->bBIOSDefaultACKnown == FALSE))
{
statusBlock->Status = ERROR_INVALID_PARAMETER;
break;
}
if (pVpBacklightStatus->bACBrightnessKnown == TRUE)
{
pDisplayBrightness->ucACBrightness = pVpBacklightStatus->ucACBrightness;
}
else
{
pDisplayBrightness->ucACBrightness = pVpBacklightStatus->ucBIOSDefaultAC;
}
//
// Report DC value if available.
//
// Note: Value ~should~ be known, even if we need to use
// DC default.
//
if ((pVpBacklightStatus->bDCBrightnessKnown == FALSE) &&
(pVpBacklightStatus->bBIOSDefaultDCKnown == FALSE))
{
statusBlock->Status = ERROR_INVALID_PARAMETER;
break;
}
if (pVpBacklightStatus->bDCBrightnessKnown == TRUE)
{
pDisplayBrightness->ucDCBrightness = pVpBacklightStatus->ucDCBrightness;
}
else
{
pDisplayBrightness->ucDCBrightness = pVpBacklightStatus->ucBIOSDefaultDC;
}
//
// Report the current power policy.
//
if (VpRunningOnAC == TRUE)
{
pDisplayBrightness->ucDisplayPolicy = DISPLAYPOLICY_AC;
}
else
{
pDisplayBrightness->ucDisplayPolicy = DISPLAYPOLICY_DC;
}
//
// No errors are expected here.
//
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = sizeof(DISPLAY_BRIGHTNESS);
} else if (ioControlCode == IOCTL_VIDEO_SET_DISPLAY_BRIGHTNESS) {
//
// This IOCTL will set the current brightness level of
// the backlight.
//
// The _BCM ACPI method will be used.
//
PDISPLAY_BRIGHTNESS pDisplayBrightness;
PDEVICE_OBJECT AttachedDevice = NULL;
BOOLEAN bSetPanelBrightness = FALSE;
//
// Check buffer size and that Brightness capabilities
// have been queried.
//
if (inputBufferLength < sizeof(DISPLAY_BRIGHTNESS))
{
statusBlock->Status = ERROR_INSUFFICIENT_BUFFER;
break;
}
//
// Videoprt insists that IOCTL_VIDEO_QUERY_SUPPORTED_BRIGHTNESS
// be called first.
//
if (pVpBacklightStatus->bQuerySupportedBrightnessCalled == FALSE)
{
statusBlock->Status = ERROR_INVALID_FUNCTION;
break;
}
//
// Videoprt will ensure no crashes, but caller should ensure
// they are setting a valid value.
//
pDisplayBrightness = (PDISPLAY_BRIGHTNESS) ioBuffer;
//
// Set the AC/DC brightness level, if appropriate.
// The caller could conceivably ask us to set a value
// for a power policy that is not currenlty active.
//
ulACPIMethodParam2 = 0;
if ((VpRunningOnAC == TRUE) &&
((pDisplayBrightness->ucDisplayPolicy) & DISPLAYPOLICY_AC))
{
bSetPanelBrightness = TRUE;
ulACPIMethodParam1 = (ULONG) pDisplayBrightness->ucACBrightness;
}
else if ((VpRunningOnAC == FALSE) &&
((pDisplayBrightness->ucDisplayPolicy) & DISPLAYPOLICY_DC))
{
bSetPanelBrightness = TRUE;
ulACPIMethodParam1 = (ULONG) pDisplayBrightness->ucDCBrightness;
}
statusBlock->Status = STATUS_SUCCESS;
if (bSetPanelBrightness == TRUE)
{
if (LCDPanelDevice) {
AttachedDevice = IoGetAttachedDeviceReference(LCDPanelDevice);
}
if (AttachedDevice) {
statusBlock->Status = pVideoPortACPIIoctl(
AttachedDevice,
(ULONG) ('MCB_'),
&ulACPIMethodParam1,
NULL,
0,
NULL);
ObDereferenceObject(AttachedDevice);
}
else {
statusBlock->Status = ERROR_INVALID_PARAMETER;
}
}
if (!NT_SUCCESS(statusBlock->Status))
{
statusBlock->Status = ERROR_INVALID_PARAMETER;
break;
}
//
// Save the new brightness levels.
//
if ((pDisplayBrightness->ucDisplayPolicy) & DISPLAYPOLICY_DC)
{
pVpBacklightStatus->ucDCBrightness = pDisplayBrightness->ucDCBrightness;
pVpBacklightStatus->bDCBrightnessKnown = TRUE;
}
if ((pDisplayBrightness->ucDisplayPolicy) & DISPLAYPOLICY_AC)
{
pVpBacklightStatus->ucACBrightness = pDisplayBrightness->ucACBrightness;
pVpBacklightStatus->bACBrightnessKnown = TRUE;
}
//
// Save the new AC / DC values in the Registry.
//
// ZwCreateKey
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet\\"
L"Control\\Backlight");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
statusBlock->Status = ZwCreateKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes,
0L,
NULL,
REG_OPTION_NON_VOLATILE,
NULL);
if (NT_SUCCESS(statusBlock->Status)) {
ULONG ulRegData = 0;
PVOID pvData;
// ACBacklightLevel
RtlInitUnicodeString(
&UnicodeString,
L"ACBacklightLevel"
);
ulRegData = (UCHAR) pVpBacklightStatus->ucACBrightness;
pvData = &ulRegData;
statusBlock->Status = ZwSetValueKey(
hkRegistry,
&UnicodeString,
0,
REG_DWORD,
pvData,
4
);
if (NT_SUCCESS(statusBlock->Status)) {
pVpBacklightStatus->bACBrightnessInRegistry = TRUE;
}
// DCBacklightLevel
RtlInitUnicodeString(
&UnicodeString,
L"DCBacklightLevel"
);
ulRegData = (UCHAR) pVpBacklightStatus->ucDCBrightness;
pvData = &ulRegData;
statusBlock->Status = ZwSetValueKey(
hkRegistry,
&UnicodeString,
0,
REG_DWORD,
pvData,
4
);
if (NT_SUCCESS(statusBlock->Status)) {
pVpBacklightStatus->bDCBrightnessInRegistry = TRUE;
}
ZwClose(hkRegistry);
}
//
// No errors are expected here unless the caller asked us to set a
// level that is not available.
//
statusBlock->Status = STATUS_SUCCESS;
statusBlock->Information = 0;
} else {
//
// All other request need to be passed to the miniport driver.
//
statusBlock->Status = STATUS_SUCCESS;
switch (ioControlCode) {
case IOCTL_VIDEO_ENABLE_VDM:
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_ENABLE_VDM\n"));
ASSERT(IS_PDO(pdoExtension) == FALSE);
statusBlock->Status = pVideoPortEnableVDM(fdoExtension,
TRUE,
(PVIDEO_VDM) ioBuffer,
inputBufferLength);
#if DBG
if (statusBlock->Status == STATUS_CONFLICTING_ADDRESSES) {
pVideoDebugPrint((Trace, "VIDEOPRT: pVideoPortEnableVDM failed\n"));
}
#endif
break;
#if _X86_
case IOCTL_VIDEO_SAVE_HARDWARE_STATE:
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_SAVE_HARDWARE_STATE\n"));
//
// allocate the memory required by the miniport driver so it can
// save its state to be returned to the caller.
//
ASSERT(IS_PDO(pdoExtension) == FALSE);
if (fdoExtension->HardwareStateSize == 0) {
statusBlock->Status = STATUS_NOT_IMPLEMENTED;
break;
}
//
// Must make sure the caller is a trusted subsystem with the
// appropriate privilege level before executing this call.
// If the calls returns FALSE we must return an error code.
//
if (!SeSinglePrivilegeCheck(RtlConvertLongToLuid(
SE_TCB_PRIVILEGE),
fdoExtension->CurrentIrpRequestorMode)) {
statusBlock->Status = STATUS_PRIVILEGE_NOT_HELD;
break;
}
((PVIDEO_HARDWARE_STATE)(ioBuffer))->StateLength =
fdoExtension->HardwareStateSize;
statusBlock->Status =
ZwAllocateVirtualMemory(NtCurrentProcess(),
(PVOID *) &(((PVIDEO_HARDWARE_STATE)(ioBuffer))->StateHeader),
0L,
&((PVIDEO_HARDWARE_STATE)(ioBuffer))->StateLength,
MEM_COMMIT,
PAGE_READWRITE);
if(!NT_SUCCESS(statusBlock->Status))
break;
BiosDataBuffer = ExAllocatePoolWithTag(PagedPool,
BIOS_DATA_SIZE,
VP_TAG);
if (BiosDataBuffer == NULL) {
statusBlock->Status = STATUS_INSUFFICIENT_RESOURCES;
} else {
statusBlock->Status =
pVideoPortGetVDMBiosData(fdoExtension,
BiosDataBuffer,
BIOS_DATA_SIZE);
if(!NT_SUCCESS(statusBlock->Status)) {
ExFreePool(BiosDataBuffer);
}
}
break;
#endif
case IOCTL_VIDEO_QUERY_PUBLIC_ACCESS_RANGES:
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_QUERY_PUBLIC_ACCESS_RANGES\n"));
//
// Must make sure the caller is a trusted subsystem with the
// appropriate privilege level before executing this call.
// If the calls returns FALSE we must return an error code.
//
if (!SeSinglePrivilegeCheck(RtlConvertLongToLuid(
SE_TCB_PRIVILEGE),
fdoExtension->CurrentIrpRequestorMode)) {
statusBlock->Status = STATUS_PRIVILEGE_NOT_HELD;
}
break;
case IOCTL_VIDEO_GET_CHILD_STATE:
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL_VIDEO_GET_CHILD_STATE\n"));
//
// If it's PDO, set the ID of the child device before letting it go
// to the miniports StartIo routine.
//
if (IS_PDO(pdoExtension)) {
if (outputBufferLength < sizeof (ULONG)) {
statusBlock->Status = STATUS_BUFFER_TOO_SMALL ;
break ;
}
*((PULONG)(ioBuffer)) = pdoExtension->ChildUId ;
}
break;
//
// The default case is when the port driver does not handle the
// request. We must then call the miniport driver.
//
default:
break;
} // switch (ioControlCode)
//
// All above cases call the miniport driver.
//
// only process it if no errors happened in the port driver
// processing.
//
if (NT_SUCCESS(statusBlock->Status)) {
pVideoDebugPrint((Trace, "VIDEOPRT: IOCTL fallthrough\n"));
vrp.IoControlCode = ioControlCode;
vrp.StatusBlock = statusBlock;
vrp.InputBuffer = ioBuffer;
vrp.InputBufferLength = inputBufferLength;
vrp.OutputBuffer = ioBuffer;
vrp.OutputBufferLength = outputBufferLength;
//
// Send the request to the miniport.
//
fdoExtension->HwStartIO(HwDeviceExtension, &vrp);
#if _X86_
if(ioControlCode == IOCTL_VIDEO_SAVE_HARDWARE_STATE) {
pVideoPortPutVDMBiosData(fdoExtension,
BiosDataBuffer,
BIOS_DATA_SIZE);
ExFreePool(BiosDataBuffer);
}
#endif
if (statusBlock->Status != NO_ERROR) {
//
// Make sure we don't tell the IO system to copy data
// on a real error.
//
if (statusBlock->Status != ERROR_MORE_DATA) {
statusBlock->Information = 0;
}
pVideoPortMapToNtStatus(statusBlock);
//
// !!! Compatibility:
// Do not require a miniport to support the REGISTER_VDM
// IOCTL, so if we get an error in that case, just
// return success.
//
// Do put up a message so people fix this.
//
if (ioControlCode == IOCTL_VIDEO_ENABLE_VDM) {
statusBlock->Status = STATUS_SUCCESS;
pVideoDebugPrint((Warn, "VIDEOPRT: The miniport driver does not support IOCTL_VIDEO_ENABLE_VDM. The video miniport driver *should* be fixed.\n"));
}
}
}
} // if (ioControlCode == ...
break;
case IRP_MJ_SHUTDOWN:
{
PEPROCESS csr;
//
// This little dance is just to make sure we never overdereference csr.
//
csr = InterlockedExchangePointer(&CsrProcess, NULL);
if (csr != NULL) {
ObDereferenceObject(csr);
}
//
// TODO: This is a temporary hack only till we get Nt/ZwQueryLastShutdownType()
// API implemented. On the next boot we need to know if the last shutdown was
// successful, and if not and if we have a watchdog event registered from that
// session we can assume the driver got stuck so we can notify user and
// optionally send a problem report to MS.
//
// N.B. This doesn't work for non-PnP devices (e.g. VGA) but it's still good
// enough, we're losing once piece of info but we'll still check for watchdog
// events from the last session.
//
{
#define VP_KEY_WATCHDOG L"\\Registry\\Machine\\System\\CurrentControlSet\\Control\\Watchdog"
#define VP_KEY_WATCHDOG_DISPLAY L"\\Registry\\Machine\\System\\CurrentControlSet\\Control\\Watchdog\\Display"
static BOOLEAN shutdownRegistered = FALSE;
if (FALSE == shutdownRegistered)
{
NTSTATUS ntStatus;
shutdownRegistered = TRUE;
//
// Is Watchdog\Display key already there?
//
ntStatus = RtlCheckRegistryKey(RTL_REGISTRY_ABSOLUTE, VP_KEY_WATCHDOG_DISPLAY);
if (!NT_SUCCESS(ntStatus)) {
//
// Is Watchdog key already there?
//
ntStatus = RtlCheckRegistryKey(RTL_REGISTRY_ABSOLUTE, VP_KEY_WATCHDOG);
if (!NT_SUCCESS(ntStatus)) {
//
// Create a new key.
//
ntStatus = RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE, VP_KEY_WATCHDOG);
}
if (NT_SUCCESS(ntStatus)) {
//
// Create a new key.
//
ntStatus = RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE, VP_KEY_WATCHDOG_DISPLAY);
}
}
if (NT_SUCCESS(ntStatus)) {
ULONG shutdownFlag = 1;
ULONG shutdownCount;
ULONG defaultShutdownCount = 0;
RTL_QUERY_REGISTRY_TABLE queryTable[] =
{
{NULL, RTL_QUERY_REGISTRY_DIRECT, L"ShutdownCount", &shutdownCount, REG_DWORD, &defaultShutdownCount, 4},
{NULL, 0, NULL}
};
//
// Get accumulated statistics from registry.
//
RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
VP_KEY_WATCHDOG_DISPLAY,
queryTable,
NULL,
NULL);
shutdownCount++;
//
// Update registry values.
//
RtlWriteRegistryValue(RTL_REGISTRY_ABSOLUTE,
VP_KEY_WATCHDOG_DISPLAY,
L"ShutdownCount",
REG_DWORD,
&shutdownCount,
sizeof (ULONG));
RtlWriteRegistryValue(RTL_REGISTRY_ABSOLUTE,
VP_KEY_WATCHDOG_DISPLAY,
L"Shutdown",
REG_DWORD,
&shutdownFlag,
sizeof (ULONG));
}
}
}
break;
}
//
// Other major entry points in the dispatch routine are not supported.
//
default:
statusBlock->Status = STATUS_SUCCESS;
break;
} // switch (irpStack->MajorFunction)
//
// save the final status so we can return it after the IRP is completed.
//
if (finalStatus == -1) {
finalStatus = statusBlock->Status;
}
RELEASE_DEVICE_LOCK(combinedExtension);
#if REMOVE_LOCK_ENABLED
IoReleaseRemoveLock(&combinedExtension->RemoveLock, Irp);
#endif
if (finalStatus == STATUS_PENDING) {
pVideoDebugPrint((Trace, "VIDEOPRT: Returned pending in pVideoPortDispatch.\n")) ;
return STATUS_PENDING ;
}
pVideoDebugPrint((Trace, "VIDEOPRT: IoCompleteRequest with Irp %x\n", Irp));
IoCompleteRequest(Irp,
IO_VIDEO_INCREMENT);
//
// We never have pending operation so always return the status code.
//
pVideoDebugPrint((Trace, "VIDEOPRT: final IOCTL status: %08lx\n",
finalStatus));
return finalStatus;
} // pVideoPortDispatch()
VOID
VideoPortFreeDeviceBase(
IN PVOID HwDeviceExtension,
IN PVOID MappedAddress
)
/*++
Routine Description:
VideoPortFreeDeviceBase frees a block of I/O addresses or memory space
previously mapped into the system address space by calling
VideoPortGetDeviceBase.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
MappedAddress - Specifies the base address of the block to be freed. This
value must be the same as the value returned by VideoPortGetDeviceBase.
Return Value:
None.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
pVideoPortFreeDeviceBase(HwDeviceExtension, MappedAddress);
return;
}
PVOID
pVideoPortFreeDeviceBase(
IN PVOID HwDeviceExtension,
IN PVOID MappedAddress
)
{
PMAPPED_ADDRESS nextMappedAddress;
PMAPPED_ADDRESS lastMappedAddress;
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
pVideoDebugPrint((Info, "VPFreeDeviceBase at mapped address is %08lx\n",
MappedAddress));
lastMappedAddress = NULL;
nextMappedAddress = fdoExtension->MappedAddressList;
while (nextMappedAddress) {
if (nextMappedAddress->MappedAddress == MappedAddress) {
//
// Count up how much memory a miniport driver is really taking
//
if (nextMappedAddress->bNeedsUnmapping) {
fdoExtension->MemoryPTEUsage -=
ADDRESS_AND_SIZE_TO_SPAN_PAGES(nextMappedAddress->MappedAddress,
nextMappedAddress->NumberOfUchars);
}
if (!(--nextMappedAddress->RefCount)) {
//
// Unmap address, if necessary.
//
if (nextMappedAddress->bNeedsUnmapping) {
if (nextMappedAddress->bLargePageRequest) {
MmUnmapVideoDisplay(nextMappedAddress->MappedAddress,
nextMappedAddress->NumberOfUchars);
} else {
MmUnmapIoSpace(nextMappedAddress->MappedAddress,
nextMappedAddress->NumberOfUchars);
}
}
//
// Remove mapped address from list.
//
if (lastMappedAddress == NULL) {
fdoExtension->MappedAddressList =
nextMappedAddress->NextMappedAddress;
} else {
lastMappedAddress->NextMappedAddress =
nextMappedAddress->NextMappedAddress;
}
ExFreePool(nextMappedAddress);
}
//
// We just return the value to show that the call succeeded.
//
return (nextMappedAddress);
} else {
lastMappedAddress = nextMappedAddress;
nextMappedAddress = nextMappedAddress->NextMappedAddress;
}
}
return NULL;
} // end VideoPortFreeDeviceBase()
PVOID
VideoPortGetDeviceBase(
IN PVOID HwDeviceExtension,
IN PHYSICAL_ADDRESS IoAddress,
IN ULONG NumberOfUchars,
IN UCHAR InIoSpace
)
/*++
Routine Description:
VideoPortGetDeviceBase maps a memory or I/O address range into the
system (kernel) address space. Access to this mapped address space
must follow these rules:
If the input value for InIoSpace is 1 (the address IS in I/O space),
the returned logical address should be used in conjunction with
VideoPort[Read/Write]Port[Uchar/Ushort/Ulong] functions.
^^^^
If the input value for InIoSpace is 0 (the address IS NOT in I/O
space), the returned logical address should be used in conjunction
with VideoPort[Read/Write]Register[Uchar/Ushort/Ulong] functions.
^^^^^^^^
Note that VideoPortFreeDeviceBase is used to unmap a previously mapped
range from the system address space.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
IoAddress - Specifies the base physical address of the range to be
mapped in the system address space.
NumberOfUchars - Specifies the number of bytes, starting at the base
address, to map in system space. The driver must not access
addresses outside this range.
InIoSpace - Specifies that the address is in the I/O space if 1.
Otherwise, the address is assumed to be in memory space.
Return Value:
This function returns a base address suitable for use by the hardware
access functions. VideoPortGetDeviceBase may be called several times
by the miniport driver.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
//
// We specify large page as FALSE for the default since the miniport could
// be using the address at raise IRQL in an ISR.
//
return pVideoPortGetDeviceBase(HwDeviceExtension,
IoAddress,
NumberOfUchars,
InIoSpace,
FALSE);
}
BOOLEAN
VpTranslateBusAddress(
IN PFDO_EXTENSION fdoExtension,
IN PPHYSICAL_ADDRESS IoAddress,
IN OUT PULONG addressSpace,
IN OUT PPHYSICAL_ADDRESS TranslatedAddress
)
/*++
Routine Description:
This routine finds the cpu relative address that matches the given
bus relative address.
Arguments:
fdoExtension - The device extension for the device in question.
IoAddress - The address which we mean to translate.
addressSpace - pointer to resource type (IO, memory, etc).
TranslatedAddress - a pointer to the location in which to store the
translated address.
Returns:
TRUE if successful,
FALSE otherwise.
--*/
{
BOOLEAN bStatus;
//
// We need to find a way to translate dense space before we can
// do this.
//
#if 0
if ((fdoExtension->Flags & LEGACY_DRIVER) != LEGACY_DRIVER) {
bStatus = VpTranslateResource(
fdoExtension,
addressSpace,
IoAddress,
TranslatedAddress);
} else {
#endif
bStatus = HalTranslateBusAddress(
fdoExtension->AdapterInterfaceType,
fdoExtension->SystemIoBusNumber,
*IoAddress,
addressSpace,
TranslatedAddress);
#if 0
}
#endif
return bStatus;
}
PVOID
pVideoPortGetDeviceBase(
IN PVOID HwDeviceExtension,
IN PHYSICAL_ADDRESS IoAddress,
IN ULONG NumberOfUchars,
IN UCHAR InIoSpace,
IN BOOLEAN bLargePage
)
{
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
PHYSICAL_ADDRESS cardAddress = IoAddress;
PVOID mappedAddress = NULL;
PMAPPED_ADDRESS newMappedAddress;
BOOLEAN bMapped;
ULONG addressSpace;
ULONG p6Caching = FALSE;
pVideoDebugPrint((Info, "VPGetDeviceBase reqested %08lx mem type. address is %08lx %08lx, length of %08lx\n",
InIoSpace, IoAddress.HighPart, IoAddress.LowPart, NumberOfUchars));
//
// Properly configure the flags for translation
//
addressSpace = InIoSpace & 0xFF;
p6Caching = addressSpace & VIDEO_MEMORY_SPACE_P6CACHE;
addressSpace &= ~VIDEO_MEMORY_SPACE_P6CACHE;
addressSpace &= ~VIDEO_MEMORY_SPACE_DENSE;
if (addressSpace & VIDEO_MEMORY_SPACE_USER_MODE) {
ASSERT(FALSE);
return NULL;
}
if ((((cardAddress.QuadPart >= 0x000C0000) && (cardAddress.QuadPart < 0x000C8000)) &&
(InIoSpace == 0) &&
(VpC0000Compatible == 2)) ||
VpTranslateBusAddress(fdoExtension,
&IoAddress,
&addressSpace,
&cardAddress)) {
//
// Use reference counting for addresses to support broken ATI !
// Return the previously mapped address if we find the same physical
// address.
//
PMAPPED_ADDRESS nextMappedAddress;
pVideoDebugPrint((Info, "VPGetDeviceBase requested %08lx mem type. physical address is %08lx %08lx, length of %08lx\n",
addressSpace, cardAddress.HighPart, cardAddress.LowPart, NumberOfUchars));
nextMappedAddress = fdoExtension->MappedAddressList;
while (nextMappedAddress) {
if ((nextMappedAddress->InIoSpace == InIoSpace) &&
(nextMappedAddress->NumberOfUchars == NumberOfUchars) &&
(nextMappedAddress->PhysicalAddress.QuadPart == cardAddress.QuadPart)) {
pVideoDebugPrint((Info, "VPGetDeviceBase : refCount hit on address %08lx \n",
nextMappedAddress->PhysicalAddress.LowPart));
nextMappedAddress->RefCount++;
//
// Count up how much memory a miniport driver is really taking
//
if (nextMappedAddress->bNeedsUnmapping) {
fdoExtension->MemoryPTEUsage +=
ADDRESS_AND_SIZE_TO_SPAN_PAGES(nextMappedAddress->MappedAddress,
nextMappedAddress->NumberOfUchars);
}
return (nextMappedAddress->MappedAddress);
} else {
nextMappedAddress = nextMappedAddress->NextMappedAddress;
}
}
//
// Allocate memory to store mapped address for unmap.
//
newMappedAddress = ExAllocatePoolWithTag(NonPagedPool,
sizeof(MAPPED_ADDRESS),
'trpV');
if (!newMappedAddress) {
pVideoDebugPrint((Error, "VIDEOPRT: Not enough memory to cache mapped address! \n"));
return NULL;
}
//
// If the address is in IO space, don't do anything.
// If the address is in memory space, map it and save the information.
//
if (addressSpace & VIDEO_MEMORY_SPACE_IO) {
mappedAddress = (PVOID) cardAddress.QuadPart;
bMapped = FALSE;
} else {
//
// Map the device base address into the virtual address space
//
// NOTE: This routine is order dependant, and changing flags like
// bLargePage will affect the caching of address we do earlier
// on in this routine.
//
if (p6Caching && EnableUSWC) {
mappedAddress = MmMapIoSpace(cardAddress,
NumberOfUchars,
MmFrameBufferCached);
if (mappedAddress == NULL) {
mappedAddress = MmMapIoSpace(cardAddress,
NumberOfUchars,
FALSE);
}
} else if (bLargePage) {
mappedAddress = MmMapVideoDisplay(cardAddress,
NumberOfUchars,
0);
} else {
mappedAddress = MmMapIoSpace(cardAddress,
NumberOfUchars,
FALSE);
}
if (mappedAddress == NULL) {
ExFreePool(newMappedAddress);
pVideoDebugPrint((Error, "VIDEOPRT: MmMapIoSpace FAILED\n"));
return NULL;
}
bMapped = TRUE;
fdoExtension->MemoryPTEUsage +=
ADDRESS_AND_SIZE_TO_SPAN_PAGES(mappedAddress,
NumberOfUchars);
}
//
// Save the reference
//
newMappedAddress->PhysicalAddress = cardAddress;
newMappedAddress->RefCount = 1;
newMappedAddress->MappedAddress = mappedAddress;
newMappedAddress->NumberOfUchars = NumberOfUchars;
newMappedAddress->InIoSpace = InIoSpace;
newMappedAddress->bNeedsUnmapping = bMapped;
newMappedAddress->bLargePageRequest = bLargePage;
//
// Link current list to new entry.
//
newMappedAddress->NextMappedAddress = fdoExtension->MappedAddressList;
//
// Point anchor at new list.
//
fdoExtension->MappedAddressList = newMappedAddress;
} else {
pVideoDebugPrint((Error, "VIDEOPRT: VpTranslateBusAddress failed\n"));
}
pVideoDebugPrint((Info, "VIDEOPRT: VideoPortGetDeviceBase mapped virtual address is %08lx\n",
mappedAddress));
return mappedAddress;
} // end VideoPortGetDeviceBase()
NTSTATUS
pVideoPortGetDeviceDataRegistry(
IN PVOID Context,
IN PUNICODE_STRING PathName,
IN INTERFACE_TYPE BusType,
IN ULONG BusNumber,
IN PKEY_VALUE_FULL_INFORMATION *BusInformation,
IN CONFIGURATION_TYPE ControllerType,
IN ULONG ControllerNumber,
IN PKEY_VALUE_FULL_INFORMATION *ControllerInformation,
IN CONFIGURATION_TYPE PeripheralType,
IN ULONG PeripheralNumber,
IN PKEY_VALUE_FULL_INFORMATION *PeripheralInformation
)
/*++
Routine Description:
Arguments:
Return Value:
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
//
// This macro should be in the io system header file.
//
#define GetIoQueryDeviceInfo(DeviceInfo, InfoType) \
((PVOID) ( ((PUCHAR) (*(DeviceInfo + InfoType))) + \
((ULONG_PTR) (*(DeviceInfo + InfoType))->DataOffset) ))
#define GetIoQueryDeviceInfoLength(DeviceInfo, InfoType) \
((*(DeviceInfo + InfoType))->DataLength)
PVP_QUERY_DEVICE queryDevice = Context;
PKEY_VALUE_FULL_INFORMATION *deviceInformation;
PCM_FULL_RESOURCE_DESCRIPTOR configurationData;
switch (queryDevice->DeviceDataType) {
case VpBusData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceDataCallback: BusData\n"));
configurationData = (PCM_FULL_RESOURCE_DESCRIPTOR)
GetIoQueryDeviceInfo(BusInformation,
IoQueryDeviceConfigurationData);
if (NO_ERROR == ((PMINIPORT_QUERY_DEVICE_ROUTINE)
queryDevice->CallbackRoutine)(
queryDevice->MiniportHwDeviceExtension,
queryDevice->MiniportContext,
queryDevice->DeviceDataType,
GetIoQueryDeviceInfo(BusInformation,
IoQueryDeviceIdentifier),
GetIoQueryDeviceInfoLength(BusInformation,
IoQueryDeviceIdentifier),
(PVOID) &(configurationData->PartialResourceList.PartialDescriptors[1]),
configurationData->PartialResourceList.PartialDescriptors[0].u.DeviceSpecificData.DataSize,
GetIoQueryDeviceInfo(BusInformation,
IoQueryDeviceComponentInformation),
GetIoQueryDeviceInfoLength(BusInformation,
IoQueryDeviceComponentInformation)
)) {
return STATUS_SUCCESS;
} else {
return STATUS_DEVICE_DOES_NOT_EXIST;
}
break;
case VpControllerData:
deviceInformation = ControllerInformation;
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceDataCallback: ControllerData\n"));
//
// This data we are getting is actually a CM_FULL_RESOURCE_DESCRIPTOR.
//
if (NO_ERROR == ((PMINIPORT_QUERY_DEVICE_ROUTINE)
queryDevice->CallbackRoutine)(
queryDevice->MiniportHwDeviceExtension,
queryDevice->MiniportContext,
queryDevice->DeviceDataType,
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceIdentifier),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceIdentifier),
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceConfigurationData),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceConfigurationData),
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceComponentInformation),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceComponentInformation)
)) {
return STATUS_SUCCESS;
} else {
return STATUS_DEVICE_DOES_NOT_EXIST;
}
break;
case VpMonitorData:
deviceInformation = PeripheralInformation;
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceDataCallback: MonitorData\n"));
//
// This data we are getting is actually a CM_FULL_RESOURCE_DESCRIPTOR.
//
if (NO_ERROR == ((PMINIPORT_QUERY_DEVICE_ROUTINE)
queryDevice->CallbackRoutine)(
queryDevice->MiniportHwDeviceExtension,
queryDevice->MiniportContext,
queryDevice->DeviceDataType,
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceIdentifier),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceIdentifier),
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceConfigurationData),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceConfigurationData),
GetIoQueryDeviceInfo(deviceInformation,
IoQueryDeviceComponentInformation),
GetIoQueryDeviceInfoLength(deviceInformation,
IoQueryDeviceComponentInformation)
)) {
return STATUS_SUCCESS;
} else {
return STATUS_DEVICE_DOES_NOT_EXIST;
}
break;
default:
ASSERT(FALSE);
return STATUS_UNSUCCESSFUL;
}
} // end pVideoPortGetDeviceDataRegistry()
VP_STATUS
VideoPortGetDeviceData(
PVOID HwDeviceExtension,
VIDEO_DEVICE_DATA_TYPE DeviceDataType,
PMINIPORT_QUERY_DEVICE_ROUTINE CallbackRoutine,
PVOID Context
)
/*++
Routine Description:
VideoPortGetDeviceData retrieves information from the hardware hive in
the registry. The information retrieved from the registry is
bus-specific or hardware-specific.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
DeviceDataType - Specifies the type of data being requested (as indicated
in VIDEO_DEVICE_DATA_TYPE).
CallbackRoutine - Points to a function that should be called back with
the requested information.
Context - Specifies a context parameter passed to the callback function.
Return Value:
This function returns the final status of the operation.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
#define CMOS_MAX_DATA_SIZE 66000
NTSTATUS ntStatus;
VP_STATUS vpStatus;
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
VP_QUERY_DEVICE queryDevice;
PUCHAR cmosData = NULL;
ULONG cmosDataSize;
ULONG exCmosDataSize;
UNICODE_STRING Identifier;
PULONG pConfiguration = NULL;
PULONG pComponent = NULL;
queryDevice.MiniportHwDeviceExtension = HwDeviceExtension;
queryDevice.DeviceDataType = DeviceDataType;
queryDevice.CallbackRoutine = CallbackRoutine;
queryDevice.MiniportStatus = NO_ERROR;
queryDevice.MiniportContext = Context;
switch (DeviceDataType) {
case VpMachineData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceData: MachineData\n"));
ntStatus = STATUS_UNSUCCESSFUL;
pConfiguration = ExAllocatePoolWithTag(PagedPool,
0x1000,
VP_TAG);
pComponent = ExAllocatePoolWithTag(PagedPool,
0x1000,
VP_TAG);
if (pConfiguration && pComponent)
{
RTL_QUERY_REGISTRY_TABLE QueryTable[] = {
{ NULL,
RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED,
L"Identifier",
&Identifier,
REG_NONE,
NULL,
0
},
{ NULL,
RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED,
L"Configuration Data",
pConfiguration,
REG_NONE,
NULL,
0
},
{ NULL,
RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED,
L"Component Information",
pComponent,
REG_NONE,
NULL,
0
},
// Null entry to mark the end
{ 0, 0, 0, 0, 0, 0, 0 }
};
//
// The first DWORD of the buffer contains the size of the buffer.
// Upon return, the first return contains the size of the data in the buffer.
//
// A NULL bufferint he UNICODE_STRING means the unicode string will be set up automatically
//
*pConfiguration = 0x1000 - 4;
*pComponent = 0x1000 - 4;
Identifier.Buffer = NULL;
if (NT_SUCCESS(RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
L"\\Registry\\Machine\\Hardware\\Description\\System",
QueryTable,
NULL,
NULL)))
{
vpStatus = ((PMINIPORT_QUERY_DEVICE_ROUTINE) CallbackRoutine)(
HwDeviceExtension,
Context,
DeviceDataType,
Identifier.Buffer,
Identifier.Length,
pConfiguration + 1,
*pConfiguration,
pComponent + 1,
*pComponent);
if (vpStatus == NO_ERROR)
{
ntStatus = STATUS_SUCCESS;
}
}
if (Identifier.Buffer)
{
ExFreePool(Identifier.Buffer);
}
}
//
// Free up the resources
//
if (pConfiguration)
{
ExFreePool(pConfiguration);
}
if (pComponent)
{
ExFreePool(pComponent);
}
break;
case VpCmosData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceData: CmosData - not implemented\n"));
#if !defined(NO_LEGACY_DRIVERS)
cmosData = ExAllocatePoolWithTag(PagedPool,
CMOS_MAX_DATA_SIZE,
VP_TAG);
//
// Allocate enough pool to store all the CMOS data.
//
if (!cmosData) {
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
break;
}
cmosDataSize = HalGetBusData(Cmos,
0, // bus 0 returns standard Cmos info
0, // no slot number
cmosData,
CMOS_MAX_DATA_SIZE);
exCmosDataSize = HalGetBusData(Cmos,
1, // bus 1 returns extended Cmos info
0, // no slot number
cmosData + cmosDataSize,
CMOS_MAX_DATA_SIZE - cmosDataSize);
//
// Call the miniport driver callback routine
//
if (NO_ERROR == CallbackRoutine(HwDeviceExtension,
Context,
DeviceDataType,
NULL,
0,
cmosData,
cmosDataSize + exCmosDataSize,
NULL,
0)) {
ntStatus = STATUS_SUCCESS;
} else {
ntStatus = STATUS_DEVICE_DOES_NOT_EXIST;
}
#endif // NO_LEGACY_DRIVERS
break;
break;
case VpBusData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceData: BusData\n"));
ntStatus = IoQueryDeviceDescription(&fdoExtension->AdapterInterfaceType,
&fdoExtension->SystemIoBusNumber,
NULL,
NULL,
NULL,
NULL,
&pVideoPortGetDeviceDataRegistry,
(PVOID)(&queryDevice));
break;
case VpControllerData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceData: ControllerData\n"));
//
// Increment the controller number since we want to get info on the
// new controller.
// We do a pre-increment since the number must remain the same for
// monitor queries.
//
VpQueryDeviceControllerNumber++;
ntStatus = IoQueryDeviceDescription(&fdoExtension->AdapterInterfaceType,
&fdoExtension->SystemIoBusNumber,
&VpQueryDeviceControllerType,
&VpQueryDeviceControllerNumber,
NULL,
NULL,
&pVideoPortGetDeviceDataRegistry,
(PVOID)(&queryDevice));
//
// Reset the Peripheral number to zero since we are working on a new
// Controller.
//
VpQueryDevicePeripheralNumber = 0;
break;
case VpMonitorData:
pVideoDebugPrint((Trace, "VIDEOPRT: VPGetDeviceData: MonitorData\n"));
ntStatus = IoQueryDeviceDescription(&fdoExtension->AdapterInterfaceType,
&fdoExtension->SystemIoBusNumber,
&VpQueryDeviceControllerType,
&VpQueryDeviceControllerNumber,
&VpQueryDevicePeripheralType,
&VpQueryDevicePeripheralNumber,
&pVideoPortGetDeviceDataRegistry,
(PVOID)(&queryDevice));
//
// Increment the peripheral number since we have the info on this
// monitor already.
//
VpQueryDevicePeripheralNumber++;
break;
default:
pVideoDebugPrint((Warn, "VIDEOPRT: VPGetDeviceData: invalid Data type\n"));
ASSERT(FALSE);
ntStatus = STATUS_UNSUCCESSFUL;
}
//
// Free the pool we may have allocated
//
if (cmosData) {
ExFreePool(cmosData);
}
if (NT_SUCCESS(ntStatus)) {
return NO_ERROR;
} else {
pVideoDebugPrint((Warn, "VPGetDeviceData failed: return status is %08lx\n", ntStatus));
return ERROR_INVALID_PARAMETER;
}
} // end VideoPortGetDeviceData()
NTSTATUS
pVideoPortGetRegistryCallback(
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
/*++
Routine Description:
This routine gets information from the system hive, user-specified
registry (as opposed to the information gathered by ntdetect.
Arguments:
ValueName - Pointer to a unicode String containing the name of the data
value being searched for.
ValueType - Type of the data value.
ValueData - Pointer to a buffer containing the information to be written
out to the registry.
ValueLength - Size of the data being written to the registry.
Context - Specifies a context parameter passed to the callback routine.
EntryContext - Specifies a second context parameter passed with the
request.
Return Value:
STATUS_SUCCESS
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
PVP_QUERY_DEVICE queryDevice = Context;
UNICODE_STRING unicodeString;
OBJECT_ATTRIBUTES objectAttributes;
NTSTATUS ntStatus = STATUS_SUCCESS;
HANDLE fileHandle = NULL;
IO_STATUS_BLOCK ioStatusBlock;
FILE_STANDARD_INFORMATION fileStandardInfo;
PVOID fileBuffer = NULL;
LARGE_INTEGER byteOffset;
//
// If the parameter was a file to be opened, perform the operation
// here. Otherwise just return the data.
//
if (queryDevice->DeviceDataType == VP_GET_REGISTRY_FILE) {
//
// For the name of the file to be valid, we must first append
// \DosDevices in front of it.
//
RtlInitUnicodeString(&unicodeString,
ValueData);
InitializeObjectAttributes(&objectAttributes,
&unicodeString,
OBJ_CASE_INSENSITIVE,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL);
ntStatus = ZwOpenFile(&fileHandle,
FILE_GENERIC_READ | SYNCHRONIZE,
&objectAttributes,
&ioStatusBlock,
0,
FILE_SYNCHRONOUS_IO_ALERT);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortGetRegistryParameters: Could not open file\n"));
goto EndRegistryCallback;
}
ntStatus = ZwQueryInformationFile(fileHandle,
&ioStatusBlock,
&fileStandardInfo,
sizeof(FILE_STANDARD_INFORMATION),
FileStandardInformation);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortGetRegistryParameters: Could not get size of file\n"));
goto EndRegistryCallback;
}
if (fileStandardInfo.EndOfFile.HighPart) {
//
// If file is too big, do not try to go further.
//
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto EndRegistryCallback;
}
ValueLength = fileStandardInfo.EndOfFile.LowPart;
fileBuffer = ExAllocatePoolWithTag(PagedPool,
ValueLength,
VP_TAG);
if (!fileBuffer) {
pVideoDebugPrint((Error, "VideoPortGetRegistryParameters: Could not allocate buffer to read file\n"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto EndRegistryCallback;
}
ValueData = fileBuffer;
//
// Read the entire file for the beginning.
//
byteOffset.QuadPart = 0;
ntStatus = ZwReadFile(fileHandle,
NULL,
NULL,
NULL,
&ioStatusBlock,
ValueData,
ValueLength,
&byteOffset,
NULL);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortGetRegistryParameters: Could not read file\n"));
goto EndRegistryCallback;
}
}
//
// Call the miniport with the appropriate information.
//
queryDevice->MiniportStatus = ((PMINIPORT_GET_REGISTRY_ROUTINE)
queryDevice->CallbackRoutine) (queryDevice->MiniportHwDeviceExtension,
queryDevice->MiniportContext,
ValueName,
ValueData,
ValueLength);
EndRegistryCallback:
if (fileHandle) {
ZwClose(fileHandle);
}
if (fileBuffer) {
ExFreePool(fileBuffer);
}
return ntStatus;
} // end pVideoPortGetRegistryCallback()
VP_STATUS
VideoPortGetRegistryParameters(
PVOID HwDeviceExtension,
PWSTR ParameterName,
UCHAR IsParameterFileName,
PMINIPORT_GET_REGISTRY_ROUTINE CallbackRoutine,
PVOID Context
)
/*++
Routine Description:
VideoPortGetRegistryParameters retrieves information from the
CurrentControlSet in the registry. The function automatically searches
for the specified parameter name under the \Devicexxx key for the
current driver.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
ParameterName - Points to a Unicode string that contains the name of the
data value being searched for in the registry.
IsParameterFileName - If 1, the data retrieved from the requested
parameter name is treated as a file name. The contents of the file are
returned, instead of the parameter itself.
CallbackRoutine - Points to a function that should be called back with
the requested information.
Context - Specifies a context parameter passed to the callback routine.
Return Value:
This function returns the final status of the operation.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
VP_STATUS vpStatus = ERROR_INVALID_PARAMETER;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension = GET_DSP_EXT(HwDeviceExtension);
if (DoSpecificExtension->DriverNewRegistryPath != NULL) {
vpStatus = VPGetRegistryParameters(HwDeviceExtension,
ParameterName,
IsParameterFileName,
CallbackRoutine,
Context,
DoSpecificExtension->DriverNewRegistryPath,
DoSpecificExtension->DriverNewRegistryPathLength);
} else {
vpStatus = VPGetRegistryParameters(HwDeviceExtension,
ParameterName,
IsParameterFileName,
CallbackRoutine,
Context,
DoSpecificExtension->DriverOldRegistryPath,
DoSpecificExtension->DriverOldRegistryPathLength);
}
return vpStatus;
}
VP_STATUS
VPGetRegistryParameters(
PVOID HwDeviceExtension,
PWSTR ParameterName,
UCHAR IsParameterFileName,
PMINIPORT_GET_REGISTRY_ROUTINE CallbackRoutine,
PVOID Context,
PWSTR RegistryPath,
ULONG RegistryPathLength
)
{
RTL_QUERY_REGISTRY_TABLE queryTable[2];
NTSTATUS ntStatus;
VP_QUERY_DEVICE queryDevice;
LPWSTR RegPath;
LPWSTR lpstrStart, lpstrEnd;
ASSERT (ParameterName != NULL);
//
// Check if there are subkeys to be entered
//
RegPath = (LPWSTR) ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
RegistryPathLength +
2 * (wcslen(ParameterName) + sizeof(WCHAR)),
VP_TAG);
if (RegPath == NULL) {
return ERROR_NOT_ENOUGH_MEMORY;
}
wcscpy(RegPath, RegistryPath);
if (!IsParameterFileName)
{
lpstrStart = RegPath + (RegistryPathLength / 2);
while (lpstrEnd = wcschr(ParameterName, L'\\'))
{
//
// Concat the string
//
*(lpstrStart++) = L'\\';
while (ParameterName != lpstrEnd) {
*(lpstrStart++) = *(ParameterName++);
}
*lpstrStart = UNICODE_NULL;
ParameterName++;
}
}
queryDevice.MiniportHwDeviceExtension = HwDeviceExtension;
queryDevice.DeviceDataType = IsParameterFileName ? VP_GET_REGISTRY_FILE : VP_GET_REGISTRY_DATA;
queryDevice.CallbackRoutine = CallbackRoutine;
queryDevice.MiniportStatus = NO_ERROR;
queryDevice.MiniportContext = Context;
//
// Can be simplified now since we don't have to go down a directory.
// It can be just one call.
//
queryTable[0].QueryRoutine = pVideoPortGetRegistryCallback;
queryTable[0].Flags = RTL_QUERY_REGISTRY_REQUIRED;
queryTable[0].Name = ParameterName;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = 0;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
queryTable[1].EntryContext = NULL;
queryTable[1].DefaultType = REG_NONE;
queryTable[1].DefaultData = 0;
queryTable[1].DefaultLength = 0;
ntStatus = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
RegPath,
queryTable,
&queryDevice,
NULL);
if (!NT_SUCCESS(ntStatus)) {
queryDevice.MiniportStatus = ERROR_INVALID_PARAMETER;
}
ExFreePool(RegPath);
return queryDevice.MiniportStatus;
} // end VideoPortGetRegistryParameters()
VOID
pVPInit(
VOID
)
/*++
Routine Description:
First time initialization of the video port.
Normally, this is the stuff we should put in the DriverEntry routine.
However, the video port is being loaded as a DLL, and the DriverEntry
is never called. It would just be too much work to add it back to the hive
and setup.
This little routine works just as well.
--*/
{
UNICODE_STRING UnicodeString;
OBJECT_ATTRIBUTES ObjectAttributes;
NTSTATUS ntStatus;
HANDLE hkRegistry;
UCHAR OptionsData[512];
HANDLE physicalMemoryHandle = NULL;
PBACKLIGHT_STATUS pVpBacklightStatus = &VpBacklightStatus;
PKEY_VALUE_PARTIAL_INFORMATION pKeyValueInfo = NULL;
HAL_DISPLAY_BIOS_INFORMATION HalBiosInfo;
ULONG HalBiosInfoLen = sizeof(ULONG);
SYSTEM_BASIC_INFORMATION basicInfo;
//
// Check for USWC disabling
//
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet"
L"\\Control\\GraphicsDrivers\\DisableUSWC");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwOpenKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
EnableUSWC = FALSE;
ZwClose(hkRegistry);
}
//
// Check for setup running
//
{
ULONG defaultValue = 0;
ULONG UpgradeInProgress = 0, SystemSetupInProgress = 0, MiniSetupInProgress = 0;
RTL_QUERY_REGISTRY_TABLE QueryTable[] = {
{NULL, RTL_QUERY_REGISTRY_DIRECT, L"SystemSetupInProgress",
&SystemSetupInProgress, REG_DWORD, &defaultValue, 4},
{NULL, RTL_QUERY_REGISTRY_DIRECT, L"UpgradeInProgress",
&UpgradeInProgress, REG_DWORD, &defaultValue, 4},
{NULL, RTL_QUERY_REGISTRY_DIRECT, L"MiniSetupInProgress",
&MiniSetupInProgress, REG_DWORD, &defaultValue, 4},
{NULL, 0, NULL}
};
RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
L"\\Registry\\Machine\\System\\Setup",
&QueryTable[0],
NULL,
NULL);
// System is doing an upgrade.
if (UpgradeInProgress)
{
ASSERT(SystemSetupInProgress);
VpSetupType = SETUPTYPE_UPGRADE;
}
// System is doing a clean install.
else if (SystemSetupInProgress && !MiniSetupInProgress)
{
VpSetupType = SETUPTYPE_FULL;
}
else
{
VpSetupType = SETUPTYPE_NONE;
}
VpSetupTypeAtBoot = VpSetupType;
}
//
// Check for basevideo from the start options
//
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet"
L"\\Control");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwOpenKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
PVOID pwszOptions;
ULONG returnSize;
RtlInitUnicodeString(&UnicodeString,
L"SystemStartOptions");
ntStatus = ZwQueryValueKey(hkRegistry,
&UnicodeString,
KeyValueFullInformation,
OptionsData,
sizeof(OptionsData),
&returnSize);
if ((NT_SUCCESS(ntStatus)) &&
(((PKEY_VALUE_FULL_INFORMATION)OptionsData)->DataLength) &&
(((PKEY_VALUE_FULL_INFORMATION)OptionsData)->DataOffset)) {
pwszOptions = ((PUCHAR)OptionsData) +
((PKEY_VALUE_FULL_INFORMATION)OptionsData)->DataOffset;
if (wcsstr(pwszOptions, L"BASEVIDEO")) {
VpBaseVideo = TRUE;
}
}
ZwClose(hkRegistry);
}
if (VpBaseVideo == TRUE)
{
//
// If we are in Basevideo mode, then create a key and value in the
// currentcontrolset part of the hardware profile that USER will
// read to determine if the vga driver should be used or not.
//
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet\\"
L"Control\\GraphicsDrivers\\BaseVideo");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwCreateKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes,
0L,
NULL,
REG_OPTION_VOLATILE,
NULL);
if (NT_SUCCESS(ntStatus)) {
ZwClose(hkRegistry);
} else {
ASSERT(FALSE);
}
}
//
// Determine if we have a VGA compatible machine
//
ntStatus = HalQuerySystemInformation(HalDisplayBiosInformation,
HalBiosInfoLen,
&HalBiosInfo,
&HalBiosInfoLen);
if (NT_SUCCESS(ntStatus)) {
if (HalBiosInfo == HalDisplayInt10Bios) {
VpC0000Compatible = 2;
} else {
// == HalDisplayEmulatedBios,
// == HalDisplayNoBios
VpC0000Compatible = 0;
}
} else {
//
// In case of an error in the API call, we just assume it's an old HAL
// and use the old behaviour of the video port which is to assume
// there is a BIOS at C000
//
VpC0000Compatible = 1;
}
//
// Lets open the physical memory section just once, for all drivers.
//
//
// Get a pointer to physical memory so we can map the
// video frame buffer (and possibly video registers) into
// the caller's address space whenever he needs it.
//
// - Create the name
// - Initialize the data to find the object
// - Open a handle to the oject and check the status
// - Get a pointer to the object
// - Free the handle
//
RtlInitUnicodeString(&UnicodeString,
L"\\Device\\PhysicalMemory");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
(HANDLE) NULL,
(PSECURITY_DESCRIPTOR) NULL);
ntStatus = ZwOpenSection(&physicalMemoryHandle,
SECTION_ALL_ACCESS,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
ntStatus = ObReferenceObjectByHandle(physicalMemoryHandle,
SECTION_ALL_ACCESS,
(POBJECT_TYPE) NULL,
KernelMode,
&PhysicalMemorySection,
(POBJECT_HANDLE_INFORMATION) NULL);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Warn, "VIDEOPRT: VPInit: Could not reference physical memory\n"));
ASSERT(PhysicalMemorySection == NULL);
}
ZwClose(physicalMemoryHandle);
}
VpSystemMemorySize = 0;
ntStatus = ZwQuerySystemInformation(SystemBasicInformation,
&basicInfo,
sizeof(basicInfo),
NULL);
if (NT_SUCCESS(ntStatus)) {
VpSystemMemorySize
= (ULONGLONG)basicInfo.NumberOfPhysicalPages * (ULONGLONG)basicInfo.PageSize;
}
//
// Initialize the fast mutex to protect the LCD Panel information
// Initialize the fast mutex to protect INT10
//
KeInitializeMutex (&LCDPanelMutex, 0);
KeInitializeMutex (&VpInt10Mutex, 0);
//
// Check if we should use the new way of generating the registry path
//
RtlInitUnicodeString(&UnicodeString,
SZ_USE_NEW_KEY);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwOpenKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
EnableNewRegistryKey = TRUE;
ZwClose(hkRegistry);
}
//
// Initialize our structure which tracks the state of
// a backlight for the LCD (when present)
//
pVpBacklightStatus->bQuerySupportedBrightnessCalled = FALSE;
pVpBacklightStatus->bACBrightnessKnown = FALSE;
pVpBacklightStatus->bDCBrightnessKnown = FALSE;
pVpBacklightStatus->bBIOSDefaultACKnown = FALSE;
pVpBacklightStatus->bBIOSDefaultDCKnown = FALSE;
pVpBacklightStatus->bNewAPISupported = FALSE;
pVpBacklightStatus->bACBrightnessInRegistry = FALSE;
pVpBacklightStatus->bACBrightnessInRegistry = FALSE;
//
// Read the registry to find out if the new API is supported.
// If so, retreive the AC/DC brightness values and set the
// brightness as applicable (depending upon VpRunningOnAC).
//
// Note: We will store the Backlight info in the following
// location in the registry:
//
// HKLM\System\CurrentControlSet\Control\Backlight
//
// The values there will be:
//
// NewAPISupported REG_DWORD 0 Not Supported
// 1 Supported
//
//
// ACBacklightLevel REG_DWORD 0-255
//
// DCBacklightLevel REG_DWORD 0-255
//
pKeyValueInfo = ExAllocatePoolWithTag(PagedPool,
sizeof(KEY_VALUE_PARTIAL_INFORMATION) + 3,
VP_TAG);
if (pKeyValueInfo == NULL) {
return;
}
RtlInitUnicodeString(&UnicodeString,
L"\\Registry\\Machine\\System\\CurrentControlSet"
L"\\Control\\Backlight");
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwOpenKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
//
// The key is there, read the 3 values listed above
//
ULONG ulReturnSize;
// NewAPISupported
RtlInitUnicodeString(&UnicodeString,
L"NewAPISupported");
ntStatus = ZwQueryValueKey(hkRegistry,
&UnicodeString,
KeyValuePartialInformation,
(PVOID) pKeyValueInfo,
sizeof(KEY_VALUE_PARTIAL_INFORMATION) + 3,
&ulReturnSize);
if (NT_SUCCESS(ntStatus)) {
if ((UCHAR) *pKeyValueInfo->Data) {
pVpBacklightStatus->bNewAPISupported = TRUE;
}
// ACBacklightLevel
RtlInitUnicodeString(&UnicodeString,
L"ACBacklightLevel");
ntStatus = ZwQueryValueKey(hkRegistry,
&UnicodeString,
KeyValuePartialInformation,
(PVOID) pKeyValueInfo,
sizeof(KEY_VALUE_PARTIAL_INFORMATION) + 3,
&ulReturnSize);
if (NT_SUCCESS(ntStatus)) {
pVpBacklightStatus->bACBrightnessInRegistry = TRUE;
pVpBacklightStatus->bACBrightnessKnown = TRUE;
pVpBacklightStatus->ucACBrightness = (UCHAR) *pKeyValueInfo->Data;
}
// DCBacklightLevel
RtlInitUnicodeString(&UnicodeString,
L"DCBacklightLevel");
ntStatus = ZwQueryValueKey(hkRegistry,
&UnicodeString,
KeyValuePartialInformation,
(PVOID) pKeyValueInfo,
sizeof(KEY_VALUE_PARTIAL_INFORMATION) + 3,
&ulReturnSize);
if (NT_SUCCESS(ntStatus)) {
pVpBacklightStatus->bDCBrightnessInRegistry = TRUE;
pVpBacklightStatus->bDCBrightnessKnown = TRUE;
pVpBacklightStatus->ucDCBrightness = (UCHAR) *pKeyValueInfo->Data;
}
}
ZwClose(hkRegistry);
}
if (pKeyValueInfo) {
ExFreePool(pKeyValueInfo);
}
//
// Most laptops respond incorrectly to HwSetPowerState calls
// on lid close. By default, we will support the XP behavior
// and not notify the miniport on lid close.
//
// The driver will be called if the following registry key
// is present:
//
// HKLM\System\CurrentControlSet\Control\GraphicsDrivers\LidCloseSetPower"
//
RtlInitUnicodeString(&UnicodeString,
SZ_LIDCLOSE);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
ntStatus = ZwOpenKey(&hkRegistry,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(ntStatus)) {
VpLidCloseSetPower = TRUE;
ZwClose(hkRegistry);
}
//
// Initialize the bugcheck callback record
//
KeInitializeCallbackRecord(&VpCallbackRecord);
//
// Regiter for bugcheck callbacks.
//
KeRegisterBugCheckReasonCallback(&VpCallbackRecord,
pVpBugcheckCallback,
KbCallbackSecondaryDumpData,
"Videoprt");
//
// Initialize the global video port mutex.
//
KeInitializeMutex(&VpGlobalLock, 0);
}
VOID
VpDriverUnload(
IN PDRIVER_OBJECT DriverObject
)
{
ULONG_PTR emptyList = 0;
BOOLEAN conflict;
//ULONG iReset;
//PDEVICE_OBJECT DeviceObject = DriverObject->DeviceObject;
//
// Release the resource we put in the resourcemap (if any).
//
IoReportResourceUsage(&VideoClassName,
DriverObject,
NULL,
0L,
NULL,
(PCM_RESOURCE_LIST) &emptyList,
sizeof(ULONG_PTR),
FALSE,
&conflict);
//
// Unregister LCD callbacks.
//
VpUnregisterLCDCallbacks();
//
// Unregister Dock/Undock callbacks.
//
if (DockCallbackHandle)
{
IoUnregisterPlugPlayNotification(DockCallbackHandle);
}
//
// Make absolutely certain there are no HwResetHw routines left
// for this devices controlled by this DriverObject.
//
//while (DeviceObject) {
//
// for (iReset=0; iReset<6; iReset++) {
//
// if (HwResetHw[iReset].HwDeviceExtension ==
// ((PDEVICE_EXTENSION)
// DeviceObject->DeviceExtension)->HwDeviceExtension) {
//
// HwResetHw[iReset].ResetFunction = NULL;
// break;
// }
// }
//
// DeviceObject = DeviceObject->NextDevice;
//}
// This is causing us to lose video on a number of systems
// during setup. This code can be used when the necessary
// additional checks are determined.
//
//if (CsrProcess) {
// ObDereferenceObject(CsrProcess);
// CsrProcess = NULL;
//}
//
// Unregister bugcheck callbacks
//
KeDeregisterBugCheckReasonCallback(&VpCallbackRecord);
return;
}
NTSTATUS
VpInitializeBusCallback(
IN PVOID Context,
IN PUNICODE_STRING PathName,
IN INTERFACE_TYPE BusType,
IN ULONG BusNumber,
IN PKEY_VALUE_FULL_INFORMATION *BusInformation,
IN CONFIGURATION_TYPE ControllerType,
IN ULONG ControllerNumber,
IN PKEY_VALUE_FULL_INFORMATION *ControllerInformation,
IN CONFIGURATION_TYPE PeripheralType,
IN ULONG PeripheralNumber,
IN PKEY_VALUE_FULL_INFORMATION *PeripheralInformation
)
{
return STATUS_SUCCESS;
} // end VpInitializeBusCallback()
VP_STATUS
VpRegistryCallback(
PVOID HwDeviceExtension,
PVOID Context,
PWSTR ValueName,
PVOID ValueData,
ULONG ValueLength
)
{
if (ValueLength && ValueData) {
*((PULONG)Context) = *((PULONG)ValueData);
return NO_ERROR;
} else {
return ERROR_INVALID_PARAMETER;
}
}
NTSTATUS
VpAddDevice(
IN PDRIVER_OBJECT DriverObject,
IN PDEVICE_OBJECT PhysicalDeviceObject
)
{
NTSTATUS ntStatus;
PDEVICE_OBJECT functionalDeviceObject;
PDEVICE_OBJECT attachedTo;
PFDO_EXTENSION fdoExtension;
ULONG extensionAllocationSize;
PVIDEO_PORT_DRIVER_EXTENSION DriverObjectExtension;
PVIDEO_HW_INITIALIZATION_DATA HwInitializationData;
pVideoDebugPrint((Trace, "VIDEOPRT: VpAddDevice\n"));
DriverObjectExtension = (PVIDEO_PORT_DRIVER_EXTENSION)
IoGetDriverObjectExtension(DriverObject,
DriverObject);
HwInitializationData = &DriverObjectExtension->HwInitData;
extensionAllocationSize = HwInitializationData->HwDeviceExtensionSize +
sizeof(FDO_EXTENSION) +
sizeof(DEVICE_SPECIFIC_EXTENSION);
ntStatus = VpCreateDevice(DriverObject,
extensionAllocationSize,
&functionalDeviceObject);
if (NT_SUCCESS(ntStatus)) {
PCHILD_PDO_EXTENSION PdoExtension = PhysicalDeviceObject->DeviceExtension;
VideoDeviceNumber++;
fdoExtension = (PFDO_EXTENSION)functionalDeviceObject->DeviceExtension;
//
// Set any deviceExtension fields here that are PnP specific
//
fdoExtension->ChildPdoNumber = 0;
fdoExtension->ChildPdoList = NULL;
fdoExtension->PhysicalDeviceObject = PhysicalDeviceObject;
//
// Since the pnp system is notifying us of our device, this is
// not a legacy device.
//
fdoExtension->Flags = PNP_ENABLED;
//
// Now attach to the PDO we were given.
//
attachedTo = IoAttachDeviceToDeviceStack(functionalDeviceObject,
PhysicalDeviceObject);
if (attachedTo == NULL) {
pVideoDebugPrint((Error, "VIDEOPRT: Could not attach in AddDevice.\n"));
ASSERT(attachedTo != NULL);
//
// Couldn't attach. Delete the FDO, and tear down anything that has
// been allocated so far.
//
VideoDeviceNumber--;
IoDeleteDevice (functionalDeviceObject);
return STATUS_NO_SUCH_DEVICE;
}
//
// Initialize the remove lock.
//
IoInitializeRemoveLock(&fdoExtension->RemoveLock, 0, 0, 256);
fdoExtension->AttachedDeviceObject = attachedTo;
fdoExtension->VpDmaAdapterHead = NULL ;
//
// Set the power management flag indicating that device mapping
// has not been done yet.
//
fdoExtension->IsMappingReady = FALSE ;
//
// Clear DO_DEVICE_INITIALIZING flag.
//
functionalDeviceObject->Flags |= DO_BUFFERED_IO | DO_POWER_PAGABLE;
functionalDeviceObject->Flags &= ~(DO_DEVICE_INITIALIZING | DO_POWER_INRUSH);
//
// Save the function pointers to the new 5.0 miniport driver callbacks.
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwQueryInterface)) {
fdoExtension->HwSetPowerState = HwInitializationData->HwSetPowerState;
fdoExtension->HwGetPowerState = HwInitializationData->HwGetPowerState;
fdoExtension->HwQueryInterface = HwInitializationData->HwQueryInterface;
fdoExtension->HwGetVideoChildDescriptor = HwInitializationData->HwGetVideoChildDescriptor;
}
}
pVideoDebugPrint((Trace, "VIDEOPRT: VpAddDevice returned: 0x%x\n", ntStatus));
return ntStatus;
}
NTSTATUS
VpCreateDevice(
IN PDRIVER_OBJECT DriverObject,
IN ULONG DeviceExtensionSize,
OUT PDEVICE_OBJECT *FunctionalDeviceObject
)
{
WCHAR deviceNameBuffer[STRING_LENGTH];
UNICODE_STRING deviceNameUnicodeString;
NTSTATUS ntStatus;
PFDO_EXTENSION fdoExtension;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
ntStatus = pVideoPortCreateDeviceName(L"\\Device\\Video",
VideoDeviceNumber,
&deviceNameUnicodeString,
deviceNameBuffer);
//
// Create a device object to represent the Video Adapter.
//
if (NT_SUCCESS(ntStatus)) {
ntStatus = IoCreateDevice(DriverObject,
DeviceExtensionSize,
&deviceNameUnicodeString,
FILE_DEVICE_VIDEO,
0,
TRUE,
FunctionalDeviceObject);
if (NT_SUCCESS(ntStatus)) {
ntStatus = IoRegisterShutdownNotification(*FunctionalDeviceObject);
if (!NT_SUCCESS(ntStatus)) {
IoDeleteDevice(*FunctionalDeviceObject);
*FunctionalDeviceObject = NULL;
} else {
(*FunctionalDeviceObject)->DeviceType = FILE_DEVICE_VIDEO;
fdoExtension = (*FunctionalDeviceObject)->DeviceExtension;
//
// Set any deviceExtension fields here
//
DoSpecificExtension = (PVOID)(fdoExtension + 1);
DoSpecificExtension->DeviceNumber = VideoDeviceNumber;
DoSpecificExtension->pFdoExtension = fdoExtension;
DoSpecificExtension->Signature = VP_TAG;
DoSpecificExtension->ExtensionType = TypeDeviceSpecificExtension;
DoSpecificExtension->HwDeviceExtension = (PVOID)(DoSpecificExtension + 1);
DoSpecificExtension->DualviewFlags = 0;
#ifdef IOCTL_VIDEO_USE_DEVICE_IN_SESSION
DoSpecificExtension->SessionId = VIDEO_DEVICE_INVALID_SESSION;
#endif IOCTL_VIDEO_USE_DEVICE_IN_SESSION
fdoExtension->pFdoExtension = fdoExtension;
fdoExtension->Signature = VP_TAG;
fdoExtension->ExtensionType = TypeFdoExtension;
fdoExtension->FunctionalDeviceObject = *FunctionalDeviceObject;
fdoExtension->DriverObject = DriverObject;
KeInitializeMutex(&fdoExtension->SyncMutex,
0);
}
}
}
return ntStatus;
}
ULONG
VideoPortInitialize(
IN PVOID Argument1, // DriverObject
IN PVOID Argument2, // RegistryPath
IN PVIDEO_HW_INITIALIZATION_DATA HwInitializationData,
IN PVOID HwContext
)
{
PDRIVER_OBJECT driverObject = Argument1;
NTSTATUS ntStatus;
PUNICODE_STRING registryPath = (PUNICODE_STRING) Argument2;
ULONG PnpFlags;
if (VPFirstTime)
{
VPFirstTime = FALSE;
pVPInit();
}
//
// Check if the size of the pointer, or the size of the data passed in
// are OK.
//
ASSERT(HwInitializationData != NULL);
if (HwInitializationData->HwInitDataSize >
sizeof(VIDEO_HW_INITIALIZATION_DATA) ) {
pVideoDebugPrint((Error, "VIDEOPRT: Invalid initialization data size\n"));
return ((ULONG) STATUS_REVISION_MISMATCH);
}
//
// Check that each required entry is not NULL.
//
if ((!HwInitializationData->HwFindAdapter) ||
(!HwInitializationData->HwInitialize) ||
(!HwInitializationData->HwStartIO)) {
pVideoDebugPrint((Error, "VIDEOPRT: Miniport missing required entry\n"));
return ((ULONG)STATUS_REVISION_MISMATCH);
}
//
// Check the registry for PnP Flags. Currently we recongnize the
// following values:
//
// PnPEnabled - If this value is set with a non-zero value, we
// will treat behave like a PnP driver.
//
// LegacyDetect - If this value is non-zero, we will report
// a non-pci device to the system via
// IoReportDetectedDevice.
//
// If we don't get the flags, we don't know how to run this driver.
// return failure
//
if (!(NT_SUCCESS(VpGetFlags(registryPath,
HwInitializationData,
&PnpFlags))))
{
return STATUS_UNSUCCESSFUL;
}
//
// During an upgrade don't allow a driver to start unless it was written
// for this version of Windows.
//
if ((VpSetupTypeAtBoot == SETUPTYPE_UPGRADE) &&
(HwInitializationData->HwInitDataSize < sizeof(VIDEO_HW_INITIALIZATION_DATA)))
{
pVideoDebugPrint((0, "We don't allow pre WinXP drivers to start during upgrade.\n"));
return STATUS_UNSUCCESSFUL;
}
//
// Set up the device driver entry points.
//
driverObject->DriverUnload = VpDriverUnload;
driverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = pVideoPortDispatch;
driverObject->MajorFunction[IRP_MJ_CREATE] = pVideoPortDispatch;
driverObject->MajorFunction[IRP_MJ_CLOSE] = pVideoPortDispatch;
driverObject->MajorFunction[IRP_MJ_SHUTDOWN] = pVideoPortDispatch;
//
// Check that the device extension size is reasonable.
//
#if DBG
if (HwInitializationData->HwDeviceExtensionSize > 0x4000) {
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortInitialize:\n"
"Warning: Device Extension is stored in non-paged pool\n"
" Do you need a 0x%x byte device extension?\n",
HwInitializationData->HwDeviceExtensionSize));
}
#endif
//
// PnP drivers have new rules.
//
if (PnpFlags & PNP_ENABLED)
{
pVideoDebugPrint((Trace, "VIDEOPRT: VideoPortInitialize with PNP_ENABLED\n"));
//
// We also can't be plug and play compatible if the driver passes
// info in HwContext. This is because we can't store this.
//
if ((PnpFlags & VGA_DRIVER) ||
(HwContext != NULL))
{
pVideoDebugPrint((Error, "VIDEOPRT: This video driver can not be "
"PNP due to passing info in HwContext.\n"));
ASSERT(FALSE);
return STATUS_INVALID_PARAMETER;
}
} else {
//
// Don't allow a non PnP driver to start after the system is up and
// running. Instead require a reboot first.
//
if (VpSystemInitialized) {
#if defined STATUS_REBOOT_REQUIRED
return STATUS_REBOOT_REQUIRED;
#else
return STATUS_INVALID_PARAMETER;
#endif
}
}
//
// Never do legacy detection of PnP drivers on the PCI Bus.
//
if (HwInitializationData->AdapterInterfaceType == PCIBus) {
pVideoDebugPrint((Trace, "VIDEOPRT: VideoPortInitialize on PCI Bus\n"));
if ( (PnpFlags & PNP_ENABLED) &&
((PnpFlags & LEGACY_DETECT) ||
(PnpFlags & REPORT_DEVICE)) ) {
pVideoDebugPrint((Error, "VIDEOPRT: Trying to detect PnP driver on PCI - fail\n"));
return STATUS_INVALID_PARAMETER;
}
}
//
// Set this information for all PnP Drivers
//
// Special !!! - we cannot do this in the LEGACY_DETECT because the system
// will think we failed to load and return a failure code.
//
if ( (PnpFlags & PNP_ENABLED) &&
(!(PnpFlags & LEGACY_DETECT)) )
{
PVIDEO_PORT_DRIVER_EXTENSION DriverObjectExtension;
pVideoDebugPrint((Info, "VIDEOPRT: We have a PnP Device.\n"));
//
// Fill in the new PnP entry points.
//
driverObject->DriverExtension->AddDevice = VpAddDevice;
driverObject->MajorFunction[IRP_MJ_PNP] = pVideoPortPnpDispatch;
driverObject->MajorFunction[IRP_MJ_POWER] = pVideoPortPowerDispatch;
driverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = pVideoPortSystemControl;
//
// we'll do findadapter during the START_DEVICE irp
//
// Store away arguments, so we can retrieve them when we need them.
//
// Try to create a DriverObjectExtension
//
if (DriverObjectExtension = (PVIDEO_PORT_DRIVER_EXTENSION)
IoGetDriverObjectExtension(driverObject,
driverObject))
{
DriverObjectExtension->HwInitData = *HwInitializationData;
ntStatus = STATUS_SUCCESS;
}
else if (NT_SUCCESS(IoAllocateDriverObjectExtension(
driverObject,
driverObject,
sizeof(VIDEO_PORT_DRIVER_EXTENSION),
&DriverObjectExtension)))
{
DriverObjectExtension->RegistryPath = *registryPath;
DriverObjectExtension->RegistryPath.MaximumLength += sizeof(WCHAR);
DriverObjectExtension->RegistryPath.Buffer =
ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
DriverObjectExtension->RegistryPath.MaximumLength,
'trpV');
ASSERT(DriverObjectExtension->RegistryPath.Buffer);
RtlCopyUnicodeString(&(DriverObjectExtension->RegistryPath),
registryPath);
DriverObjectExtension->HwInitData = *HwInitializationData;
ntStatus = STATUS_SUCCESS;
}
else
{
//
// Something went wrong. We should have a
// DriverObjectExtension by now.
//
pVideoDebugPrint((Error, "VIDEOPRT: IoAllocateDriverExtensionObject failed!\n"));
ASSERT(FALSE);
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
}
}
//
// If we are doing legacy detection or reporting, create the FDO
// right now ...
//
if ((!(PnpFlags & PNP_ENABLED)) ||
(PnpFlags & LEGACY_DETECT) ||
(PnpFlags & VGA_DRIVER) ||
(PnpFlags & REPORT_DEVICE) ||
(HwContext != NULL)) {
pVideoDebugPrint((Trace, "VIDEOPRT: VideoPortInitialize on PCI Bus\n"));
pVideoDebugPrint((Info, "Legacy FindAdapter Interface %s\n",
BusType[HwInitializationData->AdapterInterfaceType]));
ntStatus = VideoPortLegacyFindAdapter(Argument1,
Argument2,
HwInitializationData,
HwContext,
PnpFlags);
}
return ntStatus;
}
VOID
UpdateRegValue(
IN PUNICODE_STRING RegistryPath,
IN PWCHAR RegValue,
IN ULONG Value
)
{
PWSTR Path;
Path = ExAllocatePoolWithTag(PagedPool,
RegistryPath->Length + sizeof(UNICODE_NULL),
VP_TAG);
if (Path) {
RtlCopyMemory(Path,
RegistryPath->Buffer,
RegistryPath->Length);
*(Path + (RegistryPath->Length / sizeof(UNICODE_NULL))) = UNICODE_NULL;
RtlWriteRegistryValue(
RTL_REGISTRY_ABSOLUTE,
Path,
RegValue,
REG_DWORD,
&Value,
sizeof(ULONG));
ExFreePool(Path);
}
}
ULONG
VideoPortLegacyFindAdapter(
IN PDRIVER_OBJECT DriverObject,
IN PVOID Argument2,
IN PVIDEO_HW_INITIALIZATION_DATA HwInitializationData,
IN PVOID HwContext,
IN ULONG PnpFlags
)
{
ULONG busNumber = 0;
ULONG extensionAllocationSize;
NTSTATUS ntStatus;
UCHAR nextMiniport;
ULONG registryIndex = 0;
//
// Reset the controller number used in IoQueryDeviceDescription to zero
// since we are restarting on a new type of bus.
// Note: PeripheralNumber is reset only when we find a new controller.
//
VpQueryDeviceControllerNumber = 0xFFFFFFFF;
//
// Determine size of the device extension to allocate.
//
extensionAllocationSize = sizeof(FDO_EXTENSION) +
sizeof(DEVICE_SPECIFIC_EXTENSION) +
HwInitializationData->HwDeviceExtensionSize;
//
// Check if we are on the right Bus Adapter type.
// If we are not, then return immediately.
//
ASSERT (HwInitializationData->AdapterInterfaceType < MaximumInterfaceType);
//
// Assume we are going to fail this the IoQueryDeviceDescription call
// and that no device is found.
//
ntStatus = STATUS_NO_SUCH_DEVICE;
pVideoDebugPrint((Trace, "Legacy FindAdapter Interface %s, Bus %d\n",
BusType[HwInitializationData->AdapterInterfaceType],
busNumber));
while (NT_SUCCESS(IoQueryDeviceDescription(
&HwInitializationData->AdapterInterfaceType,
&busNumber,
NULL,
NULL,
NULL,
NULL,
&VpInitializeBusCallback,
NULL))) {
//
// This is to support the multiple initialization as described by the
// again paramter in HwFindAdapter.
// We must repeat almost everything in this function until FALSE is
// returned by the miniport. This is why we test for the condition at
// the end. Freeing of data structure has to be done also since we want
// to delete a device object for a device that did not load properly.
//
do {
PDEVICE_OBJECT deviceObject = NULL;
PDEVICE_OBJECT PnPDeviceObject = NULL;
PFDO_EXTENSION fdoExtension;
UNICODE_STRING tmpString;
nextMiniport = FALSE;
//
// Allocate the buffer in which the miniport driver will store all the
// configuration information.
//
ntStatus = VpCreateDevice(DriverObject,
extensionAllocationSize,
&deviceObject);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortLegacyFindAdapter: Could not create device object\n"));
return (ULONG)ntStatus;
}
fdoExtension = deviceObject->DeviceExtension;
fdoExtension->SystemIoBusNumber = busNumber;
fdoExtension->AdapterInterfaceType =
HwInitializationData->AdapterInterfaceType;
fdoExtension->RegistryIndex = registryIndex;
//
// Initialize the remove lock.
//
IoInitializeRemoveLock(&fdoExtension->RemoveLock, 0, 0, 256);
//
// If we came through this code path, we are a legacy device
//
fdoExtension->Flags = PnpFlags | LEGACY_DRIVER;
fdoExtension->VpDmaAdapterHead = NULL ;
//
// Make the VGA driver report resources "for detection" during
// FindAdapter. Later we'll remove the "LEGACY_DETECT" flag and
// try to claim the resources for real.
//
if (fdoExtension->Flags & VGA_DRIVER) {
fdoExtension->Flags |= VGA_DETECT;
}
ntStatus = VideoPortFindAdapter(DriverObject,
Argument2,
HwInitializationData,
HwContext,
deviceObject,
&nextMiniport);
if (fdoExtension->Flags & VGA_DRIVER) {
fdoExtension->Flags &= ~VGA_DETECT;
}
pVideoDebugPrint((Info, "VIDEOPRT: Legacy VideoPortFindAdapter status = %08lx\n", ntStatus));
pVideoDebugPrint((Info, "VIDEOPRT: Legacy VideoPortFindAdapter nextMiniport = %d\n", nextMiniport));
if ((NT_SUCCESS(ntStatus) == FALSE) || (PnpFlags & LEGACY_DETECT))
{
pVideoDebugPrint((1, "Deleting Device Object.\n"));
IoDeleteDevice(deviceObject);
}
if (NT_SUCCESS(ntStatus))
{
//
// We use this variable to know if at least one of the tries at
// loading the device succeded.
//
registryIndex++;
}
else
{
continue;
}
//
// If this is the VGA driver, store this device extension for
// us to play around with the resources later on (so we can release
// the resources if we install a driver on the fly).
//
// Otherwise, determine if we want to report it to the IO system
// so it can be used as a PNP device later on.
// ... Never report a device found on the PCI bus.
//
if (PnpFlags & VGA_DRIVER)
{
VgaHwDeviceExtension = fdoExtension->HwDeviceExtension;
if (NT_SUCCESS(ntStatus)) {
//
// Claim the VGA resources again without the
// VGA_DETECT flag.
//
VideoPortVerifyAccessRanges(VgaHwDeviceExtension,
NumVgaAccessRanges,
VgaAccessRanges);
}
}
else if (PnpFlags & REPORT_DEVICE)
{
ULONG_PTR emptyList = 0;
BOOLEAN conflict;
PDEVICE_OBJECT attachedTo;
ASSERT (HwInitializationData->AdapterInterfaceType != PCIBus);
//
// Release the resource we put in the resourcemap (if any).
//
IoReportResourceUsage(&VideoClassName,
DriverObject,
NULL,
0L,
deviceObject,
(PCM_RESOURCE_LIST) &emptyList,
sizeof(ULONG_PTR),
FALSE,
&conflict);
pVideoDebugPrint((Info, "VIDEOPRT: Reporting new device to the system.\n"));
pVideoDebugPrint((Info, "VIDEOPRT: ResourceList...\n"));
if (fdoExtension->ResourceList) {
#if DBG
DumpResourceList(fdoExtension->ResourceList);
#endif
} else {
pVideoDebugPrint((Info, "\tnone.\n"));
}
ntStatus = IoReportDetectedDevice(
DriverObject,
InterfaceTypeUndefined,
-1,
-1,
fdoExtension->ResourceList,
NULL,
FALSE,
&PnPDeviceObject);
pVideoDebugPrint((Info, "VIDEOPRT: New device reported ntStatus %08lx\n", ntStatus));
ASSERT(NT_SUCCESS(ntStatus));
//
// Now we can release the memory used to hold
// the resources pointed to by ResourceList.
//
if (fdoExtension->ResourceList) {
ExFreePool(fdoExtension->ResourceList);
fdoExtension->ResourceList = NULL;
}
attachedTo = IoAttachDeviceToDeviceStack(deviceObject,
PnPDeviceObject);
ASSERT(attachedTo != NULL);
fdoExtension->AttachedDeviceObject = attachedTo;
fdoExtension->PhysicalDeviceObject = PnPDeviceObject;
//
// Clear the ReportDevice value in the registry so
// that we don't try to report the device again in
// the future.
//
UpdateRegValue(Argument2, L"ReportDevice", FALSE);
}
} while (nextMiniport);
//
// We finished finding all the device on the current bus
// Go on to the next bus.
//
busNumber++;
}
//
// If at least one device loaded, then return success, otherwise return
// the last available error message.
//
if (registryIndex > 0) {
return STATUS_SUCCESS;
} else {
return ((ULONG)ntStatus);
}
}
NTSTATUS
VideoPortFindAdapter(
IN PDRIVER_OBJECT DriverObject,
IN PVOID Argument2,
IN PVIDEO_HW_INITIALIZATION_DATA HwInitializationData,
IN PVOID HwContext,
PDEVICE_OBJECT DeviceObject,
PUCHAR nextMiniport
)
{
NTSTATUS status;
PVOID vgaDE = VgaHwDeviceExtension;
PFDO_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
POWER_STATE state;
//
// During boot of upgrade install, only let VGA,
// boot video drivers start. Other types of drivers don't get
// a chance to start until after the vga or a boot driver tries'
// to start.
//
// The logic relies on the fact that today PNP drivers try to
// start before legacy drivers (including our system vga driver).a3844
//
// All other drivers are disabled so we have a chance of
// 1) booting the machine
// 2) installing the PnP drivers
//
if ((VpSetupType == SETUPTYPE_UPGRADE) &&
((fdoExtension->Flags & (BOOT_DRIVER | VGA_DRIVER)) == 0) &&
(VpSetupAllowDriversToStart == FALSE))
{
status = STATUS_NO_SUCH_DEVICE;
}
else
{
//
// If we get here during setup we may be trying to start the
// vga driver. As soon as it is started we will allow othere
// devices to start.
//
VpSetupAllowDriversToStart = TRUE;
//
// Allow PNP adapters to start so that we can enumerate their
// children. But don't let IRP_MJ_CREATE succeed, so GDI
// won't try to use the device during gui mode setup.
//
if ((VpSetupType == SETUPTYPE_UPGRADE) &&
(fdoExtension->Flags & PNP_ENABLED)) {
fdoExtension->Flags |= UPGRADE_FAIL_HWINIT;
}
//
// If the VGA driver has the VGA resources, unclaim them temporarily
//
if (vgaDE) {
pVideoDebugPrint((Info, "VIDEOPRT: Freeing VGA resources\n"));
VpReleaseResources(GET_FDO_EXT(vgaDE));
}
status = VideoPortFindAdapter2(DriverObject,
Argument2,
HwInitializationData,
HwContext,
DeviceObject,
nextMiniport);
//
// Try to reclaim the vga ports. FYI, may not get them
// back if the new driver claimed them.
//
if (vgaDE) {
pVideoDebugPrint((Info, "VIDEOPRT: Try to allocate back vga resources.\n"));
if (((DeviceObject == DeviceOwningVga) && NT_SUCCESS(status)) ||
VideoPortVerifyAccessRanges(vgaDE,
NumVgaAccessRanges,
VgaAccessRanges) != NO_ERROR) {
//
// We couldn't reclaim the vga resources, so another driver
// must have claimed them. Lets release our resources.
//
if (VgaAccessRanges) {
ExFreePool(VgaAccessRanges);
}
VgaHwDeviceExtension = NULL;
VgaAccessRanges = NULL;
NumVgaAccessRanges = 0;
pVideoDebugPrint((Warn, "VIDEOPRT: Resource re-allocation failed.\n"));
}
}
}
if (NT_SUCCESS(status))
{
//
// Initialize Power stuff.
// Set the devices current power state.
// NOTE - we assume the device is on at this point in time ...
//
fdoExtension->DevicePowerState = PowerDeviceD0;
state.DeviceState = fdoExtension->DevicePowerState;
state = PoSetPowerState(DeviceObject,
DevicePowerState,
state);
//
// Register and enable the interface
//
VpEnableAdapterInterface((PDEVICE_SPECIFIC_EXTENSION)
(fdoExtension + 1));
//
// Mark this object as supporting buffered I/O so that the I/O system
// will only supply simple buffers in IRPs.
//
// Set and clear the two power fields to ensure we only get called
// as passive level to do power management operations.
//
// Finally, tell the system we are done with Device Initialization
//
DeviceObject->Flags |= DO_BUFFERED_IO | DO_POWER_PAGABLE;
DeviceObject->Flags &= ~(DO_DEVICE_INITIALIZING | DO_POWER_INRUSH);
fdoExtension->Flags |= FINDADAPTER_SUCCEEDED;
//
// Track the number of started devices (don't count mirroring drivers)
//
if (!IsMirrorDriver(fdoExtension)) {
NumDevicesStarted++;
}
}
return status;
}
NTSTATUS
VideoPortFindAdapter2(
IN PDRIVER_OBJECT DriverObject,
IN PVOID Argument2,
IN PVIDEO_HW_INITIALIZATION_DATA HwInitializationData,
IN PVOID HwContext,
PDEVICE_OBJECT DeviceObject,
PUCHAR nextMiniport
)
{
WCHAR deviceNameBuffer[STRING_LENGTH];
POBJECT_NAME_INFORMATION deviceName;
ULONG strLength;
NTSTATUS ntStatus;
WCHAR deviceSubpathBuffer[STRING_LENGTH];
UNICODE_STRING deviceSubpathUnicodeString;
WCHAR deviceLinkBuffer[STRING_LENGTH];
UNICODE_STRING deviceLinkUnicodeString;
KAFFINITY affinity;
PVIDEO_PORT_CONFIG_INFO miniportConfigInfo = NULL;
PDEVICE_OBJECT deviceObject;
PFDO_EXTENSION fdoExtension;
VP_STATUS findAdapterStatus = ERROR_DEV_NOT_EXIST;
ULONG driverKeySize;
PWSTR driverKeyName = NULL;
BOOLEAN symbolicLinkCreated = FALSE;
ULONG MaxObjectNumber;
PDEVICE_OBJECT pdo;
BOOLEAN ChildObject=FALSE;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
ntStatus = STATUS_NO_SUCH_DEVICE;
deviceObject = DeviceObject;
fdoExtension = deviceObject->DeviceExtension;
DoSpecificExtension = (PVOID)(fdoExtension + 1);
pdo = fdoExtension->PhysicalDeviceObject;
deviceName = (POBJECT_NAME_INFORMATION) deviceNameBuffer;
ObQueryNameString(deviceObject,
deviceName,
STRING_LENGTH * sizeof(WCHAR),
&strLength);
//
// Allocate the buffer in which the miniport driver will store all the
// configuration information.
//
miniportConfigInfo = (PVIDEO_PORT_CONFIG_INFO)
ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
sizeof(VIDEO_PORT_CONFIG_INFO),
VP_TAG);
if (miniportConfigInfo == NULL) {
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto EndOfInitialization;
}
RtlZeroMemory((PVOID) miniportConfigInfo,
sizeof(VIDEO_PORT_CONFIG_INFO));
miniportConfigInfo->Length = sizeof(VIDEO_PORT_CONFIG_INFO);
//
// Put in the BusType specified within the HW_INITIALIZATION_DATA
// structure by the miniport and the bus number inthe miniport config info.
//
miniportConfigInfo->SystemIoBusNumber = fdoExtension->SystemIoBusNumber;
miniportConfigInfo->AdapterInterfaceType = fdoExtension->AdapterInterfaceType;
//
// Initialize the pointer to VpGetProcAddress.
//
miniportConfigInfo->VideoPortGetProcAddress = VpGetProcAddress;
//
// Initialize the type of interrupt based on the bus type.
//
switch (miniportConfigInfo->AdapterInterfaceType) {
case Internal:
case MicroChannel:
case PCIBus:
miniportConfigInfo->InterruptMode = LevelSensitive;
break;
default:
miniportConfigInfo->InterruptMode = Latched;
break;
}
//
// Set up device extension pointers and sizes
//
fdoExtension->HwDeviceExtension = (PVOID)((ULONG_PTR)(fdoExtension) +
sizeof(FDO_EXTENSION) + sizeof(DEVICE_SPECIFIC_EXTENSION));
fdoExtension->HwDeviceExtensionSize =
HwInitializationData->HwDeviceExtensionSize;
fdoExtension->MiniportConfigInfo = miniportConfigInfo;
//
// Save the dependent driver routines in the device extension.
//
fdoExtension->HwFindAdapter = HwInitializationData->HwFindAdapter;
fdoExtension->HwInitialize = HwInitializationData->HwInitialize;
fdoExtension->HwInterrupt = HwInitializationData->HwInterrupt;
fdoExtension->HwStartIO = HwInitializationData->HwStartIO;
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwLegacyResourceCount)) {
fdoExtension->HwLegacyResourceList = HwInitializationData->HwLegacyResourceList;
fdoExtension->HwLegacyResourceCount = HwInitializationData->HwLegacyResourceCount;
}
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, AllowEarlyEnumeration)) {
fdoExtension->AllowEarlyEnumeration = HwInitializationData->AllowEarlyEnumeration;
}
//
// Create the name we will be storing in the \DeviceMap.
// This name is a PWSTR, not a unicode string
// This is the name of the driver with an appended device number
//
if (!NT_SUCCESS(pVideoPortCreateDeviceName(L"\\Device",
HwInitializationData->StartingDeviceNumber,
&deviceSubpathUnicodeString,
deviceSubpathBuffer))) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: Could not create device subpath number\n"));
goto EndOfInitialization;
}
DoSpecificExtension->DriverOldRegistryPathLength =
((PUNICODE_STRING)Argument2)->Length +
deviceSubpathUnicodeString.Length;
driverKeySize =
DoSpecificExtension->DriverOldRegistryPathLength +
2 * sizeof(UNICODE_NULL);
if ( (driverKeyName = (PWSTR) ExAllocatePoolWithTag(PagedPool,
driverKeySize,
VP_TAG) ) == NULL) {
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
goto EndOfInitialization;
}
RtlMoveMemory(driverKeyName,
((PUNICODE_STRING)Argument2)->Buffer,
((PUNICODE_STRING)Argument2)->Length);
RtlMoveMemory((PWSTR)( (ULONG_PTR)driverKeyName +
((PUNICODE_STRING)Argument2)->Length ),
deviceSubpathBuffer,
deviceSubpathUnicodeString.Length);
//
// Put two NULLs at the end so we can play around with the string later.
//
*((PWSTR) ((ULONG_PTR)driverKeyName +
DoSpecificExtension->DriverOldRegistryPathLength))
= UNICODE_NULL;
*((PWSTR) ((ULONG_PTR)driverKeyName +
DoSpecificExtension->DriverOldRegistryPathLength
+ sizeof(UNICODE_NULL))) = UNICODE_NULL;
//
// There is a bug in Lotus Screen Cam where it will only work if our
// reg path is \REGISTRY\Machine\System not \REGISTRY\MACHINE\SYSTEM.
// so replace the appropriate strings.
//
if (wcsstr(driverKeyName, L"MACHINE")) {
wcsncpy(wcsstr(driverKeyName, L"MACHINE"), L"Machine", sizeof("Machine")-1);
}
if (wcsstr(driverKeyName, L"SYSTEM")) {
wcsncpy(wcsstr(driverKeyName, L"SYSTEM"), L"System", sizeof("System")-1);
}
//
// Store the old key
//
DoSpecificExtension->DriverOldRegistryPath = driverKeyName;
//
// Store the new key
// If this is not a Whistler driver, then use the old key.
//
if (EnableNewRegistryKey) {
#if _X86_
if (HwInitializationData->HwInitDataSize > SIZE_OF_W2K_VIDEO_HW_INITIALIZATION_DATA)
#endif // _X86_
VpEnableNewRegistryKey(fdoExtension,
DoSpecificExtension,
(PUNICODE_STRING)Argument2,
fdoExtension->RegistryIndex);
}
//
// Store the path name of the location of the driver in the registry.
//
if (DoSpecificExtension->DriverNewRegistryPath != NULL) {
DoSpecificExtension->DriverRegistryPath =
DoSpecificExtension->DriverNewRegistryPath;
DoSpecificExtension->DriverRegistryPathLength =
DoSpecificExtension->DriverNewRegistryPathLength;
} else {
DoSpecificExtension->DriverRegistryPath =
DoSpecificExtension->DriverOldRegistryPath;
DoSpecificExtension->DriverRegistryPathLength =
DoSpecificExtension->DriverOldRegistryPathLength;
}
miniportConfigInfo->DriverRegistryPath = DoSpecificExtension->DriverRegistryPath;
//
// Let the driver know how much system memory is present.
//
miniportConfigInfo->SystemMemorySize = VpSystemMemorySize;
//
// Initialize the DPC object used for error logging.
//
KeInitializeDpc(&fdoExtension->ErrorLogDpc,
pVideoPortLogErrorEntryDPC,
deviceObject);
//
// If the machine is using a Intel 450NX PCIset with a
// 82451NX Memory & I/O Controller (MIOC) then we must
// disable write combining to work around a bug in the
// chipset.
//
// We also want to disable USWC on the Compaq fiat
// chipset.
//
if (EnableUSWC) {
if ((VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x8086, 0x84CA, 0, 0, 0, 0)) ||
(VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x0E11, 0x6010, 0, 0, 0, 0)) ||
(VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x1166, 0x0009, 0, 0, 0, 0))) {
pVideoDebugPrint((Warn, "VIDEOPRT: USWC disabled due to to MIOC bug.\n"));
EnableUSWC = FALSE;
}
//
// Disable USWC on HPs 6 way box.
//
if (VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x1166, 0x0008, 0, 0, 0, 0) &&
(VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x103C, 0x1219, 0, 0, 0, 0) ||
VideoPortCheckForDeviceExistence(fdoExtension->HwDeviceExtension, 0x103C, 0x121A, 0, 0, 0, 0))) {
pVideoDebugPrint((Warn, "USWC disabled due to to RCC USWC bug on HP 6-way system.\n"));
EnableUSWC = FALSE;
}
}
//
// Turn on the debug level based on the miniport driver entry
//
VideoPortGetRegistryParameters(fdoExtension->HwDeviceExtension,
L"VideoDebugLevel",
FALSE,
VpRegistryCallback,
&VideoDebugLevel);
if (VpAllowFindAdapter(fdoExtension)) {
ACQUIRE_DEVICE_LOCK(fdoExtension);
//
// Notify the boot driver that we will be accessing the display
// hardware.
//
VpEnableDisplay(fdoExtension, FALSE);
findAdapterStatus =
fdoExtension->HwFindAdapter(fdoExtension->HwDeviceExtension,
HwContext,
NULL,
miniportConfigInfo,
nextMiniport);
VpEnableDisplay(fdoExtension, TRUE);
RELEASE_DEVICE_LOCK(fdoExtension);
}
//
// If the adapter is not found, display an error.
//
if (findAdapterStatus != NO_ERROR) {
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortFindAdapter: Find adapter failed\n"));
ntStatus = STATUS_UNSUCCESSFUL;
goto EndOfInitialization;
}
//
// Store the emulator data in the device extension so we can use it
// later.
//
fdoExtension->NumEmulatorAccessEntries =
miniportConfigInfo->NumEmulatorAccessEntries;
fdoExtension->EmulatorAccessEntries =
miniportConfigInfo->EmulatorAccessEntries;
fdoExtension->EmulatorAccessEntriesContext =
miniportConfigInfo->EmulatorAccessEntriesContext;
fdoExtension->VdmPhysicalVideoMemoryAddress =
miniportConfigInfo->VdmPhysicalVideoMemoryAddress;
fdoExtension->VdmPhysicalVideoMemoryLength =
miniportConfigInfo->VdmPhysicalVideoMemoryLength;
//
// Store the required information in the device extension for later use.
//
fdoExtension->HardwareStateSize =
miniportConfigInfo->HardwareStateSize;
//
// If the device supplies an interrupt service routine, we must
// set up all the structures to support interrupts. Otherwise,
// they can be ignored.
//
if (fdoExtension->HwInterrupt) {
if ((miniportConfigInfo->BusInterruptLevel != 0) ||
(miniportConfigInfo->BusInterruptVector != 0)) {
#if defined(NO_LEGACY_DRIVERS)
affinity = fdoExtension->InterruptAffinity;
#else
if (fdoExtension->Flags & LEGACY_DRIVER) {
//
// Note: the spinlock for the interrupt object is created
// internally by the IoConnectInterrupt() call. It is also
// used internally by KeSynchronizeExecution.
//
fdoExtension->InterruptVector =
HalGetInterruptVector(fdoExtension->AdapterInterfaceType,
fdoExtension->SystemIoBusNumber,
miniportConfigInfo->BusInterruptLevel,
miniportConfigInfo->BusInterruptVector,
&fdoExtension->InterruptIrql,
&affinity);
} else {
affinity = fdoExtension->InterruptAffinity;
}
#endif
fdoExtension->InterruptMode = miniportConfigInfo->InterruptMode;
fdoExtension->InterruptsEnabled = TRUE;
ntStatus = IoConnectInterrupt(&fdoExtension->InterruptObject,
(PKSERVICE_ROUTINE) pVideoPortInterrupt,
deviceObject,
NULL,
fdoExtension->InterruptVector,
fdoExtension->InterruptIrql,
fdoExtension->InterruptIrql,
fdoExtension->InterruptMode,
(BOOLEAN) ((miniportConfigInfo->InterruptMode ==
LevelSensitive) ? TRUE : FALSE),
affinity,
FALSE);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: Can't connect interrupt\n"));
goto EndOfInitialization;
}
} else {
pVideoDebugPrint((Warn, "VIDEOPRT: Warning: No interrupt resources assigned to this device.\n"));
}
} else {
fdoExtension->HwInterrupt = NULL;
}
//
// Initialize DPC Support
//
KeInitializeDpc(&fdoExtension->Dpc,
pVideoPortDpcDispatcher,
fdoExtension->HwDeviceExtension);
//
// DMA support
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwStartDma)) {
fdoExtension->HwStartDma = HwInitializationData->HwStartDma;
//
// Determine if a Dma Adapter must be allocated.
//
if (fdoExtension->DmaAdapterObject == NULL &&
(miniportConfigInfo->Master ||
miniportConfigInfo->DmaChannel != 0)) {
DEVICE_DESCRIPTION deviceDescription;
ULONG numberOfMapRegisters;
//
// Get the adapter object for this card.
//
RtlZeroMemory(&deviceDescription, sizeof(deviceDescription));
deviceDescription.Version = DEVICE_DESCRIPTION_VERSION;
deviceDescription.DmaChannel = miniportConfigInfo->DmaChannel;
deviceDescription.BusNumber = miniportConfigInfo->SystemIoBusNumber;
deviceDescription.DmaWidth = miniportConfigInfo->DmaWidth;
deviceDescription.DmaSpeed = miniportConfigInfo->DmaSpeed;
deviceDescription.ScatterGather = miniportConfigInfo->ScatterGather;
deviceDescription.Master = miniportConfigInfo->Master;
deviceDescription.DmaPort = miniportConfigInfo->DmaPort;
deviceDescription.AutoInitialize = FALSE;
deviceDescription.DemandMode = miniportConfigInfo->DemandMode;
deviceDescription.MaximumLength = miniportConfigInfo->MaximumTransferLength;
deviceDescription.InterfaceType = fdoExtension->AdapterInterfaceType;
fdoExtension->DmaAdapterObject =
IoGetDmaAdapter(fdoExtension->PhysicalDeviceObject,
&deviceDescription,
&numberOfMapRegisters);
ASSERT(fdoExtension->DmaAdapterObject);
}
} // end if HW_DATA_SIZE > ... HWStartDma
//
// New, Optional.
// Setup the timer if it is specified by a driver.
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwTimer)){
fdoExtension->HwTimer = HwInitializationData->HwTimer;
if (fdoExtension->HwTimer) {
ntStatus = IoInitializeTimer(deviceObject,
pVideoPortHwTimer,
NULL);
//
// If we fail forget about the timer !
//
if (!NT_SUCCESS(ntStatus)) {
ASSERT(FALSE);
fdoExtension->HwTimer = NULL;
}
}
}
//
// New, Optional.
// Reset Hw function.
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwResetHw)) {
ULONG iReset;
for (iReset=0; iReset<6; iReset++) {
if (HwResetHw[iReset].ResetFunction == NULL) {
HwResetHw[iReset].ResetFunction = HwInitializationData->HwResetHw;
HwResetHw[iReset].HwDeviceExtension = fdoExtension->HwDeviceExtension;
break;
}
}
}
//
// The FdoList is for debugging purpose
//
{
ULONG i;
for(i = 0; i < 8; i++) {
if(FdoList[i] == NULL) {
FdoList[i] = fdoExtension;
break;
}
}
}
fdoExtension->NextFdoExtension = FdoHead;
FdoHead = fdoExtension;
//
// NOTE:
//
// We only want to reinitialize the device once the Boot sequence has
// been completed and the HAL does not need to access the device again.
// So the initialization entry point will be called when the device is
// opened.
//
if (!NT_SUCCESS(pVideoPortCreateDeviceName(L"\\DosDevices\\DISPLAY",
DoSpecificExtension->DeviceNumber + 1,
&deviceLinkUnicodeString,
deviceLinkBuffer))) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: Could not create device subpath number\n"));
goto EndOfInitialization;
}
ntStatus = IoCreateSymbolicLink(&deviceLinkUnicodeString,
&deviceName->Name);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: SymbolicLink Creation failed\n"));
goto EndOfInitialization;
}
symbolicLinkCreated = TRUE;
//
// Once initialization is finished, load the required information in the
// registry so that the appropriate display drivers can be loaded.
//
ntStatus = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
VideoClassString,
deviceName->Name.Buffer,
REG_SZ,
DoSpecificExtension->DriverRegistryPath,
DoSpecificExtension->DriverRegistryPathLength +
sizeof(UNICODE_NULL));
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: Could not store name in DeviceMap\n"));
}
if (fdoExtension->Flags & LEGACY_DRIVER) {
//
// If we successfully found a legacy driver, increment the
// global device number.
//
VideoDeviceNumber++;
}
//
// Tell win32k how many objects to try to open
//
MaxObjectNumber = VideoDeviceNumber - 1;
ntStatus = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
VideoClassString,
L"MaxObjectNumber",
REG_DWORD,
&MaxObjectNumber,
sizeof(ULONG));
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortFindAdapter: Could not store name in DeviceMap\n"));
} else {
pVideoDebugPrint((Info, "VIDEOPRT: VideoPortFindAdapter: %d is stored in MaxObjectNumber of DeviceMap\n",
MaxObjectNumber));
}
//
// Save the function pointers to the new 5.0 miniport driver callbacks.
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwQueryInterface)) {
fdoExtension->HwSetPowerState = HwInitializationData->HwSetPowerState;
fdoExtension->HwGetPowerState = HwInitializationData->HwGetPowerState;
fdoExtension->HwQueryInterface = HwInitializationData->HwQueryInterface;
if (!ChildObject)
{
fdoExtension->HwGetVideoChildDescriptor = HwInitializationData->HwGetVideoChildDescriptor;
}
}
//
// Check if the minitor should always be D0 or D3
//
{
ULONG OverrideMonitorPower = 0;
VideoPortGetRegistryParameters(fdoExtension->HwDeviceExtension,
L"OverrideMonitorPower",
FALSE,
VpRegistryCallback,
&OverrideMonitorPower);
fdoExtension->OverrideMonitorPower = (OverrideMonitorPower != 0);
}
EndOfInitialization:
//
// If we are doing detection, then don't save all of these objects.
// We just want to see if the driver would load or not
//
if ( (fdoExtension->Flags & LEGACY_DETECT) ||
(!NT_SUCCESS(ntStatus)) )
{
//
// Free the miniport config info buffer.
//
if (miniportConfigInfo) {
ExFreePool(miniportConfigInfo);
}
//
// Free the rom image if we grabbed one.
//
if (fdoExtension->RomImage) {
ExFreePool(fdoExtension->RomImage);
fdoExtension->RomImage = NULL;
}
//
// Release the resource we put in the resourcemap (if any).
//
if ((fdoExtension->Flags & LEGACY_DETECT) ||
(findAdapterStatus != NO_ERROR)) {
ULONG_PTR emptyList = 0;
BOOLEAN conflict;
IoReportResourceUsage(&VideoClassName,
DriverObject,
NULL,
0L,
deviceObject,
(PCM_RESOURCE_LIST) &emptyList, // an empty resource list
sizeof(ULONG_PTR),
FALSE,
&conflict);
}
//
// These are the things we want to delete if they were created and
// the initialization *FAILED* at a later time.
//
if (fdoExtension->InterruptObject) {
IoDisconnectInterrupt(fdoExtension->InterruptObject);
}
if (driverKeyName) {
ExFreePool(driverKeyName);
DoSpecificExtension->DriverOldRegistryPath = NULL;
}
if (DoSpecificExtension->DriverNewRegistryPath) {
ExFreePool(DoSpecificExtension->DriverNewRegistryPath);
DoSpecificExtension->DriverNewRegistryPath = NULL;
}
DoSpecificExtension->DriverRegistryPath = NULL;
if (symbolicLinkCreated) {
IoDeleteSymbolicLink(&deviceLinkUnicodeString);
}
//
// Free up any memory mapped in by the miniport using
// VideoPort GetDeviceBase.
//
while (fdoExtension->MappedAddressList != NULL)
{
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortFindAdapter: unfreed address %08lx, physical %08lx, size %08lx\n",
fdoExtension->MappedAddressList->MappedAddress,
fdoExtension->MappedAddressList->PhysicalAddress.LowPart,
fdoExtension->MappedAddressList->NumberOfUchars));
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortFindAdapter: unfreed refcount %d, unmapping %d\n\n",
fdoExtension->MappedAddressList->RefCount,
fdoExtension->MappedAddressList->bNeedsUnmapping));
VideoPortFreeDeviceBase(fdoExtension->HwDeviceExtension,
fdoExtension->MappedAddressList->MappedAddress);
}
//
// Remove any HwResetHw function we may have added for this device.
//
if (HwInitializationData->HwInitDataSize >
FIELD_OFFSET(VIDEO_HW_INITIALIZATION_DATA, HwResetHw)) {
ULONG iReset;
for (iReset=0; iReset<6; iReset++) {
if (HwResetHw[iReset].HwDeviceExtension ==
fdoExtension->HwDeviceExtension) {
HwResetHw[iReset].ResetFunction = NULL;
break;
}
}
}
} else {
HwInitializationData->StartingDeviceNumber++;
}
return ntStatus;
}
BOOLEAN
pVideoPortInterrupt(
IN PKINTERRUPT Interrupt,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This function is the main interrupt service routine. If finds which
miniport driver the interrupt was for and forwards it.
Arguments:
Interrupt -
DeviceObject -
Return Value:
Returns TRUE if the interrupt was expected.
--*/
{
PFDO_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
BOOLEAN bRet;
UNREFERENCED_PARAMETER(Interrupt);
//
// If there is no interrupt routine, fail the assertion
//
ASSERT (fdoExtension->HwInterrupt);
if (fdoExtension->InterruptsEnabled) {
bRet = fdoExtension->HwInterrupt(fdoExtension->HwDeviceExtension);
} else {
bRet = FALSE; // this device did not handle the interrupt
}
return bRet;
} // pVideoPortInterrupt()
VOID
VideoPortLogError(
IN PVOID HwDeviceExtension,
IN PVIDEO_REQUEST_PACKET Vrp OPTIONAL,
IN VP_STATUS ErrorCode,
IN ULONG UniqueId
)
/*++
Routine Description:
This routine saves the error log information so it can be processed at
any IRQL.
Arguments:
HwDeviceExtension - Supplies the HBA miniport driver's adapter data storage.
Vrp - Supplies an optional pointer to a video request packet if there is
one.
ErrorCode - Supplies an error code indicating the type of error.
UniqueId - Supplies a unique identifier for the error.
Return Value:
None.
--*/
{
VP_ERROR_LOG_ENTRY errorLogEntry;
//
// Save the information in a local errorLogEntry structure.
//
errorLogEntry.DeviceExtension = GET_FDO_EXT(HwDeviceExtension);
if (Vrp != NULL) {
errorLogEntry.IoControlCode = Vrp->IoControlCode;
} else {
errorLogEntry.IoControlCode = 0;
}
errorLogEntry.ErrorCode = ErrorCode;
errorLogEntry.UniqueId = UniqueId;
//
// Call the sync routine so we are synchronized when writting in
// the device extension.
//
pVideoPortSynchronizeExecution(HwDeviceExtension,
VpMediumPriority,
pVideoPortLogErrorEntry,
&errorLogEntry);
return;
} // end VideoPortLogError()
BOOLEAN
pVideoPortLogErrorEntry(
IN PVOID Context
)
/*++
Routine Description:
This function is the synchronized LogError functions.
Arguments:
Context - Context value used here as the VP_ERROR_LOG_ENTRY for this
particular error
Return Value:
None.
--*/
{
PVP_ERROR_LOG_ENTRY logEntry = Context;
PFDO_EXTENSION fdoExtension = logEntry->DeviceExtension;
//
// If the error log entry is already full, then dump the error.
//
if (fdoExtension->InterruptFlags & VP_ERROR_LOGGED) {
pVideoDebugPrint((Trace, "VIDEOPRT: VideoPortLogError: Dumping video error log packet.\n"));
pVideoDebugPrint((Info, "\tControlCode = %x, ErrorCode = %x, UniqueId = %x.\n",
logEntry->IoControlCode, logEntry->ErrorCode,
logEntry->UniqueId));
return TRUE;
}
//
// Indicate that the error log entry is in use.
//
fdoExtension->InterruptFlags |= VP_ERROR_LOGGED;
fdoExtension->ErrorLogEntry = *logEntry;
//
// Now queue a DPC so we can process the error.
//
KeInsertQueueDpc(&fdoExtension->ErrorLogDpc,
NULL,
NULL);
return TRUE;
} // end pVideoPortLogErrorEntry();
VOID
pVideoPortLogErrorEntryDPC(
IN PKDPC Dpc,
IN PVOID DeferredContext,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
)
/*++
Routine Description:
This function allocates an I/O error log record, fills it in and writes it
to the I/O error log.
Arguments:
Dpc - Pointer to the DPC object.
DeferredContext - Context parameter that was passed to the DPC
initialization routine. It contains a pointer to the deviceObject.
SystemArgument1 - Unused.
SystemArgument2 - Unused.
Return Value:
None.
--*/
{
PDEVICE_OBJECT DeviceObject = DeferredContext;
PFDO_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
PIO_ERROR_LOG_PACKET errorLogPacket;
errorLogPacket = (PIO_ERROR_LOG_PACKET)
IoAllocateErrorLogEntry(DeviceObject,
sizeof(IO_ERROR_LOG_PACKET) + sizeof(ULONG));
if (errorLogPacket != NULL) {
errorLogPacket->MajorFunctionCode = IRP_MJ_DEVICE_CONTROL;
errorLogPacket->RetryCount = 0;
errorLogPacket->NumberOfStrings = 0;
errorLogPacket->StringOffset = 0;
errorLogPacket->EventCategory = 0;
//
// Translate the miniport error code into the NT I\O driver.
//
switch (fdoExtension->ErrorLogEntry.ErrorCode) {
case ERROR_INVALID_FUNCTION:
case ERROR_INVALID_PARAMETER:
errorLogPacket->ErrorCode = IO_ERR_INVALID_REQUEST;
break;
case ERROR_NOT_ENOUGH_MEMORY:
case ERROR_INSUFFICIENT_BUFFER:
errorLogPacket->ErrorCode = IO_ERR_INSUFFICIENT_RESOURCES;
break;
case ERROR_DEV_NOT_EXIST:
errorLogPacket->ErrorCode = IO_ERR_CONFIGURATION_ERROR;
break;
case ERROR_IO_PENDING:
ASSERT(FALSE);
case ERROR_MORE_DATA:
case NO_ERROR:
errorLogPacket->ErrorCode = 0;
break;
//
// If it is another error code, than assume it is private to the
// driver and just pass as-is.
//
default:
errorLogPacket->ErrorCode =
fdoExtension->ErrorLogEntry.ErrorCode;
break;
}
errorLogPacket->UniqueErrorValue =
fdoExtension->ErrorLogEntry.UniqueId;
errorLogPacket->FinalStatus = STATUS_SUCCESS;
errorLogPacket->SequenceNumber = 0;
errorLogPacket->IoControlCode =
fdoExtension->ErrorLogEntry.IoControlCode;
errorLogPacket->DumpDataSize = sizeof(ULONG);
errorLogPacket->DumpData[0] =
fdoExtension->ErrorLogEntry.ErrorCode;
IoWriteErrorLogEntry(errorLogPacket);
}
fdoExtension->InterruptFlags &= ~VP_ERROR_LOGGED;
} // end pVideoPortLogErrorEntry();
VOID
pVideoPortMapToNtStatus(
IN PSTATUS_BLOCK StatusBlock
)
/*++
Routine Description:
This function maps a Win32 error code to an NT error code, making sure
the inverse translation will map back to the original status code.
Arguments:
StatusBlock - Pointer to the status block
Return Value:
None.
--*/
{
PNTSTATUS status = &StatusBlock->Status;
switch (*status) {
case ERROR_INVALID_FUNCTION:
*status = STATUS_NOT_IMPLEMENTED;
break;
case ERROR_NOT_ENOUGH_MEMORY:
*status = STATUS_INSUFFICIENT_RESOURCES;
break;
case ERROR_INVALID_PARAMETER:
*status = STATUS_INVALID_PARAMETER;
break;
case ERROR_INSUFFICIENT_BUFFER:
*status = STATUS_BUFFER_TOO_SMALL;
//
// Make sure we zero out the information block if we get an
// insufficient buffer.
//
StatusBlock->Information = 0;
break;
case ERROR_MORE_DATA:
*status = STATUS_BUFFER_OVERFLOW;
break;
case ERROR_DEV_NOT_EXIST:
*status = STATUS_DEVICE_DOES_NOT_EXIST;
break;
case ERROR_IO_PENDING:
ASSERT(FALSE);
// Fall through.
case NO_ERROR:
*status = STATUS_SUCCESS;
break;
default:
pVideoDebugPrint((Error, "VIDEOPRT: Invalid return value from HwStartIo!\n"));
ASSERT(FALSE);
//
// Since the driver did not see fit to follow the
// rules about returning correct error codes. Videoprt will do it for
// them.
//
*status = STATUS_UNSUCCESSFUL;
break;
}
return;
} // end pVideoPortMapToNtStatus()
NTSTATUS
pVideoPortMapUserPhysicalMem(
IN PFDO_EXTENSION FdoExtension,
IN HANDLE ProcessHandle OPTIONAL,
IN PHYSICAL_ADDRESS PhysicalAddress,
IN OUT PULONG Length,
IN OUT PULONG InIoSpace,
IN OUT PVOID *VirtualAddress
)
/*++
Routine Description:
This function maps a view of a block of physical memory into a process'
virtual address space.
Arguments:
HwDeviceExtension - Pointer to the miniport driver's device extension.
ProcessHandle - Optional handle to the process into which the memory must
be mapped.
PhysicalAddress - Offset from the beginning of physical memory, in bytes.
Length - Pointer to a variable that will receive that actual size in
bytes of the view. The length is rounded to a page boundary. THe
length may not be zero.
InIoSpace - Specifies if the address is in the IO space if TRUE; otherwise,
the address is assumed to be in memory space.
VirtualAddress - Pointer to a variable that will receive the base
address of the view. If the initial value is not NULL, then the view
will be allocated starting at teh specified virtual address rounded
down to the next 64kb addess boundary.
Return Value:
STATUS_UNSUCCESSFUL if the length was zero.
STATUS_SUCCESS otherwise.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
NTSTATUS ntStatus;
HANDLE physicalMemoryHandle;
PHYSICAL_ADDRESS physicalAddressBase;
PHYSICAL_ADDRESS physicalAddressEnd;
PHYSICAL_ADDRESS viewBase;
PHYSICAL_ADDRESS mappedLength;
HANDLE processHandle;
BOOLEAN translateBaseAddress;
BOOLEAN translateEndAddress;
ULONG inIoSpace2;
ULONG inIoSpace1;
ULONG MapViewFlags;
//
// Check for a length of zero. If it is, the entire physical memory
// would be mapped into the process' address space. An error is returned
// in this case.
//
if (!*Length) {
return STATUS_INVALID_PARAMETER_4;
}
if (!(*InIoSpace & VIDEO_MEMORY_SPACE_USER_MODE)) {
return STATUS_INVALID_PARAMETER_5;
}
//
// Get a handle to the physical memory section using our pointer.
// If this fails, return.
//
ntStatus = ObOpenObjectByPointer(PhysicalMemorySection,
0L,
(PACCESS_STATE) NULL,
SECTION_ALL_ACCESS,
(POBJECT_TYPE) NULL,
KernelMode,
&physicalMemoryHandle);
if (!NT_SUCCESS(ntStatus)) {
return ntStatus;
}
//
// No flags are used in translation
//
inIoSpace1 = *InIoSpace & VIDEO_MEMORY_SPACE_IO;
inIoSpace2 = *InIoSpace & VIDEO_MEMORY_SPACE_IO;
//
// Initialize the physical addresses that will be translated
//
physicalAddressEnd.QuadPart = PhysicalAddress.QuadPart + (*Length - 1);
//
// Translate the physical addresses.
//
translateBaseAddress =
VpTranslateBusAddress(FdoExtension,
&PhysicalAddress,
&inIoSpace1,
&physicalAddressBase);
translateEndAddress =
VpTranslateBusAddress(FdoExtension,
&physicalAddressEnd,
&inIoSpace2,
&physicalAddressEnd);
if ( !(translateBaseAddress && translateEndAddress) ) {
ZwClose(physicalMemoryHandle);
return STATUS_DEVICE_CONFIGURATION_ERROR;
}
ASSERT(inIoSpace1 == inIoSpace2);
//
// Calcualte the length of the memory to be mapped
//
mappedLength.QuadPart = physicalAddressEnd.QuadPart -
physicalAddressBase.QuadPart + 1;
//
// If the mappedlength is zero, somthing very weird happened in the HAL
// since the Length was checked against zero.
//
ASSERT (mappedLength.QuadPart != 0);
//
// If the address is in io space, just return the address, otherwise
// go through the mapping mechanism
//
if ( (*InIoSpace) & (ULONG)0x01 ) {
(ULONG_PTR) *VirtualAddress = (ULONG_PTR) physicalAddressBase.QuadPart;
} else {
//
// If no process handle was passed, get the handle to the current
// process.
//
if (ProcessHandle) {
processHandle = ProcessHandle;
} else {
processHandle = NtCurrentProcess();
}
//
// initialize view base that will receive the physical mapped
// address after the MapViewOfSection call.
//
viewBase = physicalAddressBase;
//
// Map the section
//
if ((*InIoSpace) & VIDEO_MEMORY_SPACE_P6CACHE) {
MapViewFlags = PAGE_READWRITE | PAGE_WRITECOMBINE;
} else {
MapViewFlags = PAGE_READWRITE | PAGE_NOCACHE;
}
ntStatus = ZwMapViewOfSection(physicalMemoryHandle,
processHandle,
VirtualAddress,
0L,
(ULONG_PTR) mappedLength.QuadPart,
&viewBase,
(PULONG_PTR) (&(mappedLength.QuadPart)),
ViewUnmap,
0,
MapViewFlags);
//
// Close the handle since we only keep the pointer reference to the
// section.
//
ZwClose(physicalMemoryHandle);
//
// Mapping the section above rounded the physical address down to the
// nearest 64 K boundary. Now return a virtual address that sits where
// we wnat by adding in the offset from the beginning of the section.
//
(ULONG_PTR) *VirtualAddress += (ULONG_PTR) (physicalAddressBase.QuadPart -
viewBase.QuadPart);
}
//
// Restore all the other FLAGS
//
*InIoSpace = inIoSpace1 | *InIoSpace & ~VIDEO_MEMORY_SPACE_IO;
*Length = mappedLength.LowPart;
return ntStatus;
} // end pVideoPortMapUserPhysicalMem()
PVOID
VideoPortAllocatePool(
IN PVOID HwDeviceExtension,
IN VP_POOL_TYPE PoolType,
IN SIZE_T NumberOfBytes,
IN ULONG Tag)
/*++
Routine Description:
Allocates Memory
Arguments:
HwDeviceExtension - pointer to the miniports device extension
VpPoolType - The type of pool to allocate:
VpNonPagedPool
VpPagedPool
VpNonPagedPoolCacheAligned
VpPagedPoolCacheAligned
NumberOfBytes - Supplies the number of bytes to allocate.
Tag - Supplies the caller's identifying tag.
Return Value:
NULL - The memory allocation failed.
NON-NULL - Returns a pointer to the allocated pool.
--*/
{
ASSERT(HwDeviceExtension != NULL);
return ExAllocatePoolWithTag(PoolType, NumberOfBytes, Tag);
}
VOID
VideoPortFreePool(
IN PVOID HwDeviceExtension,
IN PVOID Ptr
)
/*++
Routine Description:
Free's allocated memory
Arguments:
HwDeviceExtension - pointer to the miniports device extension
Ptr - pointer to the memory to free
Return Value:
none
--*/
{
ASSERT(HwDeviceExtension != NULL);
ASSERT(Ptr != NULL);
ExFreePool(Ptr);
}
VP_STATUS
VideoPortAllocateBuffer(
IN PVOID HwDeviceExtension,
IN ULONG Size,
OUT PVOID *Buffer
)
{
pVideoDebugPrint((1, "Obsolete function: Please use VideoPortAllocatePool instead\n"));
*Buffer = VideoPortAllocatePool(HwDeviceExtension, VpPagedPool, Size, ' pmV');
if (*Buffer) {
return NO_ERROR;
} else {
return ERROR_NOT_ENOUGH_MEMORY;
}
}
VOID
VideoPortReleaseBuffer(
IN PVOID HwDeviceExtension,
IN PVOID Buffer
)
{
pVideoDebugPrint((1, "Obsolete function: Please use VideoPortFreePool instead\n"));
VideoPortFreePool(HwDeviceExtension, Buffer);
}
VP_STATUS
VideoPortMapBankedMemory(
PVOID HwDeviceExtension,
PHYSICAL_ADDRESS PhysicalAddress,
PULONG Length,
PULONG InIoSpace,
PVOID *VirtualAddress,
ULONG BankLength,
UCHAR ReadWriteBank,
PBANKED_SECTION_ROUTINE BankRoutine,
PVOID Context
)
/*++
Routine Description:
VideoPortMapMemory allows the miniport driver to map a section of
physical memory (either memory or registers) into the calling process'
address space (eventhough we are in kernel mode, this function is
executed within the same context as the user-mode process that initiated
the call).
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
PhysicalAddress - Specifies the physical address to be mapped.
Length - Points to the number of bytes of physical memory to be mapped.
This argument returns the actual amount of memory mapped.
InIoSpace - Points to a variable that is 1 if the address is in I/O
space. Otherwise, the address is assumed to be in memory space.
VirtualAddress - A pointer to a location containing:
on input: An optional handle to the process in which the memory must
be mapped. 0 must be used to map the memory for the display
driver (in the context of the windows server process).
on output: The return value is the virtual address at which the
physical address has been mapped.
BankLength - Size of the bank on the device.
ReadWriteBank - TRUE is the bank is READ\WRITE, FALSE if there are
two independent READ and WRITE banks.
BankRoutine - Pointer to the banking routine.
Context - Context parameter passed in by the miniport supplied on
each callback to the miniport.
Return Value:
VideoPortMapBankedMemory returns the status of the operation.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
VP_STATUS status;
HANDLE processHandle;
//
// Save the process ID, but don't change it since MapMemory relies
// on it also
//
if (*VirtualAddress == NULL) {
processHandle = NtCurrentProcess();
} else {
processHandle = (HANDLE) *VirtualAddress;
}
status = VideoPortMapMemory(HwDeviceExtension,
PhysicalAddress,
Length,
InIoSpace,
VirtualAddress);
if (status == NO_ERROR) {
NTSTATUS ntstatus;
ntstatus = MmSetBankedSection(processHandle,
*VirtualAddress,
BankLength,
ReadWriteBank,
BankRoutine,
Context);
if (!NT_SUCCESS(ntstatus)) {
ASSERT (FALSE);
status = ERROR_INVALID_PARAMETER;
}
}
return status;
} // end VideoPortMapBankedMemory()
VP_STATUS
VideoPortMapMemory(
PVOID HwDeviceExtension,
PHYSICAL_ADDRESS PhysicalAddress,
PULONG Length,
PULONG InIoSpace,
PVOID *VirtualAddress
)
/*++
Routine Description:
VideoPortMapMemory allows the miniport driver to map a section of
physical memory (either memory or registers) into the calling process'
address space (eventhough we are in kernel mode, this function is
executed within the same context as the user-mode process that initiated
the call).
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
PhysicalAddress - Specifies the physical address to be mapped.
Length - Points to the number of bytes of physical memory to be mapped.
This argument returns the actual amount of memory mapped.
InIoSpace - Points to a variable that is 1 if the address is in I/O
space. Otherwise, the address is assumed to be in memory space.
VirtualAddress - A pointer to a location containing:
on input: An optional handle to the process in which the memory must
be mapped. 0 must be used to map the memory for the display
driver (in the context of the windows server process).
on output: The return value is the virtual address at which the
physical address has been mapped.
Return Value:
VideoPortMapMemory returns the status of the operation.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
NTSTATUS ntStatus;
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
HANDLE processHandle;
//
// Check for valid pointers.
//
if (!(ARGUMENT_PRESENT(Length)) ||
!(ARGUMENT_PRESENT(InIoSpace)) ||
!(ARGUMENT_PRESENT(VirtualAddress)) ) {
ASSERT(FALSE);
return ERROR_INVALID_PARAMETER;
}
//
// Let's handle the special memory types here.
//
// NOTE
// Large pages is automatic - the caller need not specify this attribute
// since it does not affect the device.
//
// Save the process handle and zero out the Virtual address field
//
if (*VirtualAddress == NULL) {
if (*InIoSpace & VIDEO_MEMORY_SPACE_USER_MODE)
{
ASSERT(FALSE);
return ERROR_INVALID_PARAMETER;
}
ntStatus = STATUS_SUCCESS;
//
// We specify TRUE for large pages since we know the addrses will only
// be used in the context of the display driver, at normal IRQL.
//
*VirtualAddress = pVideoPortGetDeviceBase(HwDeviceExtension,
PhysicalAddress,
*Length,
(UCHAR) (*InIoSpace),
TRUE);
//
// Zero can only be success if the driver is calling to MAP
// address 0. Otherwise, it is an error.
//
if (*VirtualAddress == NULL) {
//
// Only on X86 can the logical address also be 0.
//
#if defined (_X86_) || defined(_IA64_)
if (PhysicalAddress.QuadPart != 0)
#endif
ntStatus = STATUS_INVALID_PARAMETER;
}
} else {
if (!(*InIoSpace & VIDEO_MEMORY_SPACE_USER_MODE))
{
//
// We can not assert since this is an existing path and old
// drivers will not have this flag set.
//
// ASSERT(FALSE);
// return ERROR_INVALID_PARAMETER;
//
*InIoSpace |= VIDEO_MEMORY_SPACE_USER_MODE;
}
processHandle = (HANDLE) *VirtualAddress;
*VirtualAddress = NULL;
ntStatus = pVideoPortMapUserPhysicalMem(fdoExtension,
processHandle,
PhysicalAddress,
Length,
InIoSpace,
VirtualAddress);
}
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VideoPortMapMemory failed with NtStatus = %08lx\n",
ntStatus));
pVideoDebugPrint((Error, "*VirtualAddress = 0x%x\n", *VirtualAddress));
pVideoDebugPrint((Error, "Length = 0x%x\n", *Length));
pVideoDebugPrint((Error,
"PhysicalAddress.LowPart = 0x%08lx, PhysicalAddress.HighPart = 0x%08lx\n",
PhysicalAddress.LowPart, PhysicalAddress.HighPart));
*VirtualAddress = NULL;
return ERROR_INVALID_PARAMETER;
} else {
return NO_ERROR;
}
} // end VideoPortMapMemory()
VOID
pVideoPortPowerCompletionIoctl(
IN PDEVICE_OBJECT DeviceObject,
IN UCHAR MinorFunction,
IN POWER_STATE PowerState,
IN PVOID Context,
IN PIO_STATUS_BLOCK IoStatus
)
/*++
Routine Description:
Completion routine that is called when one of our power irps has been
comeplted. This allows us to fill out the status code for the request.
Arguments:
DeviceObject - Pointer to the device object
MinorFunction - Minor function of the IRP
PowerState - Power state that was set
Context - Context paramter
IoStatus - Status block for that IRP
Return Value:
VOID
Environment:
--*/
{
PPOWER_BLOCK powerContext = (PPOWER_BLOCK) Context;
if (powerContext->FinalFlag == TRUE) {
powerContext->Irp->IoStatus.Status = IoStatus->Status;
IoCompleteRequest (powerContext->Irp, IO_VIDEO_INCREMENT);
}
ExFreePool(Context);
return;
}
BOOLEAN
pVideoPortResetDisplay(
IN ULONG Columns,
IN ULONG Rows
)
/*++
Routine Description:
Callback for the HAL that calls the miniport driver.
Arguments:
Columns - The number of columns of the video mode.
Rows - The number of rows for the video mode.
Return Value:
We always return FALSE so the HAL will always reste the mode afterwards.
Environment:
Non-paged only.
Used in BugCheck and soft-reset calls.
--*/
{
ULONG iReset;
BOOLEAN bRetVal = FALSE;
for (iReset=0;
(iReset < 6) && (HwResetHw[iReset].HwDeviceExtension);
iReset++) {
PFDO_EXTENSION fdoExtension =
GET_FDO_EXT(HwResetHw[iReset].HwDeviceExtension);
//
// We can only reset devices which are on the hibernation path, otherwise
// we are running into the risk that IO / MMIO decode has been disabled
// by PCI.SYS for that device during power management cycle.
//
if (HwResetHw[iReset].ResetFunction &&
(fdoExtension->HwInitStatus == HwInitSucceeded) &&
(fdoExtension->OnHibernationPath == TRUE)) {
bRetVal &= HwResetHw[iReset].ResetFunction(HwResetHw[iReset].HwDeviceExtension,
Columns,
Rows);
}
}
return bRetVal;
} // end pVideoPortResetDisplay()
BOOLEAN
VideoPortScanRom(
PVOID HwDeviceExtension,
PUCHAR RomBase,
ULONG RomLength,
PUCHAR String
)
/*++
Routine Description:
Does a case *SENSITIVE* search for a string in the ROM.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
RomBase - Base address at which the search should start.
RomLength - Size, in bytes, of the ROM area in which to perform the
search.
String - String to search for
Return Value:
Returns TRUE if the string was found.
Returns FALSE if it was not found.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
ULONG stringLength, length;
ULONG_PTR startOffset;
PUCHAR string1, string2;
BOOLEAN match;
UNREFERENCED_PARAMETER(HwDeviceExtension);
stringLength = strlen(String);
for (startOffset = 0;
startOffset < RomLength - stringLength + 1;
startOffset++) {
length = stringLength;
string1 = RomBase + startOffset;
string2 = String;
match = TRUE;
while (length--) {
if (READ_REGISTER_UCHAR(string1++) - (*string2++)) {
match = FALSE;
break;
}
}
if (match) {
return TRUE;
}
}
return FALSE;
} // end VideoPortScanRom()
VP_STATUS
VideoPortSetRegistryParameters(
PVOID HwDeviceExtension,
PWSTR ValueName,
PVOID ValueData,
ULONG ValueLength
)
/*++
Routine Description:
VideoPortSetRegistryParameters writes information to the CurrentControlSet
in the registry. The function automatically searches for or creates the
specified parameter name under the parameter key of the current driver.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
ValueName - Points to a Unicode string that contains the name of the
data value being written in the registry.
ValueData - Points to a buffer containing the information to be written
to the registry.
ValueLength - Specifies the size of the data being written to the registry.
Return Value:
This function returns the final status of the operation.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
VP_STATUS vpStatus;
ASSERT (ValueName != NULL);
DoSpecificExtension = GET_DSP_EXT(HwDeviceExtension);
if (DoSpecificExtension->DriverNewRegistryPath != NULL) {
vpStatus = VPSetRegistryParameters(HwDeviceExtension,
ValueName,
ValueData,
ValueLength,
DoSpecificExtension->DriverNewRegistryPath,
DoSpecificExtension->DriverNewRegistryPathLength);
} else {
vpStatus = VPSetRegistryParameters(HwDeviceExtension,
ValueName,
ValueData,
ValueLength,
DoSpecificExtension->DriverOldRegistryPath,
DoSpecificExtension->DriverOldRegistryPathLength);
}
return vpStatus;
}
VP_STATUS
VPSetRegistryParameters(
PVOID HwDeviceExtension,
PWSTR ValueName,
PVOID ValueData,
ULONG ValueLength,
PWSTR RegistryPath,
ULONG RegistryPathLength
)
{
NTSTATUS ntStatus;
LPWSTR RegPath;
LPWSTR lpstrStart, lpstrEnd;
//
// Check if there are subkeys need to be created
//
RegPath = (LPWSTR) ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
RegistryPathLength +
2 * (wcslen(ValueName) + sizeof(WCHAR)),
VP_TAG);
if (RegPath == NULL) {
return ERROR_NOT_ENOUGH_MEMORY;
}
wcscpy(RegPath, RegistryPath);
lpstrStart = RegPath + (RegistryPathLength / 2);
while (lpstrEnd = wcschr(ValueName, L'\\'))
{
//
// Concat the string
//
*(lpstrStart++) = L'\\';
while (ValueName != lpstrEnd) {
*(lpstrStart++) = *(ValueName++);
}
*lpstrStart = UNICODE_NULL;
//
// Create the Key.
//
ntStatus = RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE, RegPath);
if (!NT_SUCCESS(ntStatus)) {
ExFreePool(RegPath);
return ERROR_INVALID_PARAMETER;
}
ValueName++;
}
//
// Don't let people store as DefaultSettings anymore ...
// Must still work for older drivers through.
//
if (wcsncmp(ValueName,
L"DefaultSettings.",
sizeof(L"DefaultSettings.")) == 0) {
ASSERT(FALSE);
//
// check for NT 5.0
//
if (GET_FDO_EXT(HwDeviceExtension)->Flags & PNP_ENABLED) {
ExFreePool(RegPath);
return ERROR_INVALID_PARAMETER;
}
}
ntStatus = RtlWriteRegistryValue(RTL_REGISTRY_ABSOLUTE,
RegPath,
ValueName,
REG_BINARY,
ValueData,
ValueLength);
ExFreePool(RegPath);
if (!NT_SUCCESS(ntStatus)) {
return ERROR_INVALID_PARAMETER;
}
return NO_ERROR;
} // end VideoPortSetRegistryParamaters()
VP_STATUS
VideoPortFlushRegistry(
PVOID HwDeviceExtension
)
/*++
Routine Description:
This routine will flush the registry keys and values associated with
the given miniport driver.
Arguments:
HwDeviceExtension - Pointer to the miniport device extension.
Return Value:
Status Code.
--*/
{
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING KeyName;
HANDLE RegKey;
NTSTATUS Status;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
DoSpecificExtension = GET_DSP_EXT(HwDeviceExtension);
if (DoSpecificExtension->DriverNewRegistryPath != NULL) {
KeyName.Length = (USHORT)DoSpecificExtension->DriverNewRegistryPathLength;
KeyName.MaximumLength = (USHORT)DoSpecificExtension->DriverNewRegistryPathLength;
KeyName.Buffer = DoSpecificExtension->DriverNewRegistryPath;
} else {
KeyName.Length = (USHORT)DoSpecificExtension->DriverOldRegistryPathLength;
KeyName.MaximumLength = (USHORT)DoSpecificExtension->DriverOldRegistryPathLength;
KeyName.Buffer = DoSpecificExtension->DriverOldRegistryPath;
}
//
// Flush the registry key
//
InitializeObjectAttributes(&ObjectAttributes,
&KeyName,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
Status = ZwOpenKey(&RegKey,
KEY_READ | KEY_WRITE,
&ObjectAttributes);
if (NT_SUCCESS(Status)) {
ZwFlushKey(RegKey);
ZwClose(RegKey);
}
return NO_ERROR;
}
VOID
pVideoPortHwTimer(
IN PDEVICE_OBJECT DeviceObject,
PVOID Context
)
/*++
Routine Description:
This function is the main entry point for the timer routine that we then
forward to the miniport driver.
Arguments:
DeviceObject -
Context - Not needed
Return Value:
None.
--*/
{
PFDO_EXTENSION fdoExtension = DeviceObject->DeviceExtension;
UNREFERENCED_PARAMETER(Context);
fdoExtension->HwTimer(fdoExtension->HwDeviceExtension);
return;
} // pVideoPortInterrupt()
VOID
VideoPortStartTimer(
PVOID HwDeviceExtension
)
/*++
Routine Description:
Enables the timer specified in the HW_INITIALIZATION_DATA structure
passed to the video port driver at init time.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
Return Value:
None
--*/
{
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
if (fdoExtension->HwTimer == NULL) {
ASSERT(fdoExtension->HwTimer != NULL);
} else {
IoStartTimer(fdoExtension->FunctionalDeviceObject);
}
return;
}
VOID
VideoPortStopTimer(
PVOID HwDeviceExtension
)
/*++
Routine Description:
Disables the timer specified in the HW_INITIALIZATION_DATA structure
passed to the video port driver at init time.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
Return Value:
None
--*/
{
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
if (fdoExtension->HwTimer == NULL) {
ASSERT(fdoExtension->HwTimer != NULL);
} else {
IoStopTimer(fdoExtension->FunctionalDeviceObject);
}
return;
}
BOOLEAN
VideoPortSynchronizeExecution(
PVOID HwDeviceExtension,
VIDEO_SYNCHRONIZE_PRIORITY Priority,
PMINIPORT_SYNCHRONIZE_ROUTINE SynchronizeRoutine,
PVOID Context
)
/*++
Stub so we can allow the miniports to link directly
--*/
{
return pVideoPortSynchronizeExecution(HwDeviceExtension,
Priority,
SynchronizeRoutine,
Context);
} // end VideoPortSynchronizeExecution()
BOOLEAN
pVideoPortSynchronizeExecution(
PVOID HwDeviceExtension,
VIDEO_SYNCHRONIZE_PRIORITY Priority,
PMINIPORT_SYNCHRONIZE_ROUTINE SynchronizeRoutine,
PVOID Context
)
/*++
Routine Description:
VideoPortSynchronizeExecution synchronizes the execution of a miniport
driver function in the following manner:
- If Priority is equal to VpLowPriority, the current thread is
raised to the highest non-interrupt-masking priority. In
other words, the current thread can only be pre-empted by an ISR.
- If Priority is equal to VpMediumPriority and there is an
ISR associated with the video device, then the function specified
by SynchronizeRoutine is synchronized with the ISR.
If no ISR is connected, synchronization is made at VpHighPriority
level.
- If Priority is equal to VpHighPriority, the current IRQL is
raised to HIGH_LEVEL, which effectively masks out ALL interrupts
in the system. This should be done sparingly and for very short
periods -- it will completely freeze up the entire system.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
Priority - Specifies the type of priority at which the SynchronizeRoutine
must be executed (found in VIDEO_SYNCHRONIZE_PRIORITY).
SynchronizeRoutine - Points to the miniport driver function to be
synchronized.
Context - Specifies a context parameter to be passed to the miniport's
SynchronizeRoutine.
Return Value:
This function returns TRUE if the operation is successful. Otherwise, it
returns FALSE.
--*/
{
BOOLEAN status;
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
KIRQL oldIrql;
//
// Switch on which type of priority.
//
switch (Priority) {
case VpMediumPriority:
case VpHighPriority:
//
// This is synchronized with the interrupt object
//
if (fdoExtension->InterruptObject) {
status = KeSynchronizeExecution(fdoExtension->InterruptObject,
(PKSYNCHRONIZE_ROUTINE)
SynchronizeRoutine,
Context);
return status;
}
//
// Fall through for Medium Priority
//
case VpLowPriority:
//
// Just normal level
//
status = SynchronizeRoutine(Context);
return status;
default:
return FALSE;
}
}
VP_STATUS
VideoPortUnmapMemory(
PVOID HwDeviceExtension,
PVOID VirtualAddress,
HANDLE ProcessHandle
)
/*++
Routine Description:
VideoPortUnmapMemory allows the miniport driver to unmap a physical
address range previously mapped into the calling process' address space
using the VideoPortMapMemory function.
Arguments:
HwDeviceExtension - Points to the miniport driver's device extension.
VirtualAddress - Points to the virtual address to unmap from the
address space of the caller.
// InIoSpace - Specifies whether the address is in I/O space (1) or memory
// space (0).
ProcessHandle - Handle to the process from which memory must be unmapped.
Return Value:
This function returns a status code of NO_ERROR if the operation succeeds.
It returns ERROR_INVALID_PARAMETER if an error occurs.
Environment:
This routine cannot be called from a miniport routine synchronized with
VideoPortSynchronizeRoutine or from an ISR.
--*/
{
NTSTATUS ntstatus;
VP_STATUS vpStatus = NO_ERROR;
PFDO_EXTENSION fdoExtension = GET_FDO_EXT(HwDeviceExtension);
//
// Backwards compatibility to when the ProcessHandle was actually
// ULONG InIoSpace.
//
if (((ULONG_PTR)(ProcessHandle)) == 1) {
pVideoDebugPrint((Warn,"VIDEOPRT: VideoPortUnmapMemory - interface change, must pass in process handle\n\n"));
ASSERT(FALSE);
return NO_ERROR;
}
if (((ULONG_PTR)(ProcessHandle)) == 0) {
//
// If the process handle is zero, it means it was mapped by the display
// driver and is therefore in kernel mode address space.
//
if (!pVideoPortFreeDeviceBase(HwDeviceExtension, VirtualAddress)) {
ASSERT(FALSE);
vpStatus = ERROR_INVALID_PARAMETER;
}
} else {
//
// A process handle is passed in.
// This ms it was mapped for use by an application (DCI \ DirectDraw).
//
ntstatus = ZwUnmapViewOfSection ( ProcessHandle,
(PVOID) ( ((ULONG_PTR)VirtualAddress) & (~(PAGE_SIZE - 1)) ) );
if ( (!NT_SUCCESS(ntstatus)) &&
(ntstatus != STATUS_PROCESS_IS_TERMINATING) ) {
ASSERT(FALSE);
vpStatus = ERROR_INVALID_PARAMETER;
}
}
return NO_ERROR;
} // end VideoPortUnmapMemory()
BOOLEAN
VideoPortSignalDmaComplete(
IN PVOID HwDeviceExtension,
IN PDMA pDmaHandle
)
/*++
Routine Description:
This function is obsolete.
--*/
{
return FALSE;
}
VP_STATUS
VideoPortCreateSecondaryDisplay(
IN PVOID HwDeviceExtension,
IN OUT PVOID *SecondaryDeviceExtension,
IN ULONG ulFlag
)
/*++
Routine Description:
This routine creates a secondary device object for the given device. This
will allow for dual-view support.
Arguments:
HwDeviceExtension - The HwDeviceExtension for the device which wants to
create additional output devices.
SecondaryDeviceExtension - The location in which to store the
HwDeviceExtension for the secondary display.
Returns:
VP_STATUS
--*/
{
WCHAR deviceNameBuffer[STRING_LENGTH];
POBJECT_NAME_INFORMATION deviceName;
ULONG strLength;
UNICODE_STRING deviceNameUnicodeString;
PDEVICE_OBJECT DeviceObject;
NTSTATUS ntStatus;
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension;
PFDO_EXTENSION FdoExtension = GET_FDO_EXT(HwDeviceExtension);
PVIDEO_PORT_DRIVER_EXTENSION DriverObjectExtension;
PUNICODE_STRING RegistryPath;
WCHAR deviceSubpathBuffer[STRING_LENGTH];
UNICODE_STRING deviceSubpathUnicodeString;
WCHAR deviceLinkBuffer[STRING_LENGTH];
UNICODE_STRING deviceLinkUnicodeString;
ULONG driverKeySize;
PWSTR driverKeyName = NULL;
ULONG MaxObjectNumber;
//
// Retrieve the data we cached away during VideoPortInitialize.
//
DriverObjectExtension = (PVIDEO_PORT_DRIVER_EXTENSION)
IoGetDriverObjectExtension(
FdoExtension->DriverObject,
FdoExtension->DriverObject);
ASSERT(DriverObjectExtension);
ASSERT(SecondaryDeviceExtension != NULL);
RegistryPath = &DriverObjectExtension->RegistryPath;
ntStatus = pVideoPortCreateDeviceName(L"\\Device\\Video",
VideoDeviceNumber,
&deviceNameUnicodeString,
deviceNameBuffer);
//
// Create a device object to represent the Video Adapter.
//
if (NT_SUCCESS(ntStatus)) {
ntStatus = IoCreateDevice(FdoExtension->DriverObject,
sizeof(DEVICE_SPECIFIC_EXTENSION) +
FdoExtension->HwDeviceExtensionSize,
&deviceNameUnicodeString,
FILE_DEVICE_VIDEO,
0,
TRUE,
&DeviceObject);
if (NT_SUCCESS(ntStatus)) {
DeviceObject->DeviceType = FILE_DEVICE_VIDEO;
DoSpecificExtension = DeviceObject->DeviceExtension;
//
// Initialize DeviceSpecificExtension
//
DoSpecificExtension->DeviceNumber = VideoDeviceNumber;
DoSpecificExtension->pFdoExtension = FdoExtension;
DoSpecificExtension->Signature = VP_TAG;
DoSpecificExtension->ExtensionType = TypeDeviceSpecificExtension;
DoSpecificExtension->HwDeviceExtension = (PVOID)(DoSpecificExtension + 1);
DoSpecificExtension->DualviewFlags = ulFlag | VIDEO_DUALVIEW_SECONDARY;
#ifdef IOCTL_VIDEO_USE_DEVICE_IN_SESSION
DoSpecificExtension->SessionId = VIDEO_DEVICE_INVALID_SESSION;
#endif IOCTL_VIDEO_USE_DEVICE_IN_SESSION
deviceName = (POBJECT_NAME_INFORMATION) deviceNameBuffer;
ObQueryNameString(DeviceObject,
deviceName,
STRING_LENGTH * sizeof(WCHAR),
&strLength);
//
// Create the name we will be storing in the \DeviceMap.
// This name is a PWSTR, not a unicode string
// This is the name of the driver with an appended device number
//
if (!NT_SUCCESS(pVideoPortCreateDeviceName(
L"\\Device",
DriverObjectExtension->HwInitData.StartingDeviceNumber + 1,
&deviceSubpathUnicodeString,
deviceSubpathBuffer)))
{
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: Could not create device subpath number\n"));
IoDeleteDevice(DeviceObject);
return STATUS_INSUFFICIENT_RESOURCES;
}
DoSpecificExtension->DriverOldRegistryPathLength =
RegistryPath->Length +
deviceSubpathUnicodeString.Length;
driverKeySize =
DoSpecificExtension->DriverOldRegistryPathLength +
2 * sizeof(UNICODE_NULL);
if ( (driverKeyName = (PWSTR) ExAllocatePoolWithTag(PagedPool,
driverKeySize,
VP_TAG) ) == NULL)
{
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: Fail to allocate driverKeyName\n"));
IoDeleteDevice(DeviceObject);
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlMoveMemory(driverKeyName,
RegistryPath->Buffer,
RegistryPath->Length);
RtlMoveMemory((PWSTR)((ULONG_PTR)driverKeyName +
RegistryPath->Length),
deviceSubpathBuffer,
deviceSubpathUnicodeString.Length);
//
// Put two NULLs at the end so we can play around with the string later.
//
*((PWSTR) ((ULONG_PTR)driverKeyName +
DoSpecificExtension->DriverOldRegistryPathLength))
= UNICODE_NULL;
*((PWSTR) ((ULONG_PTR)driverKeyName +
(DoSpecificExtension->DriverOldRegistryPathLength
+ sizeof(UNICODE_NULL)))) = UNICODE_NULL;
//
// There is a bug in Lotus Screen Cam where it will only work if our
// reg path is \REGISTRY\Machine\System not \REGISTRY\MACHINE\SYSTEM.
// so replace the appropriate strings.
//
if (wcsstr(driverKeyName, L"MACHINE")) {
wcsncpy(wcsstr(driverKeyName, L"MACHINE"), L"Machine", sizeof("Machine")-1);
}
if (wcsstr(driverKeyName, L"SYSTEM")) {
wcsncpy(wcsstr(driverKeyName, L"SYSTEM"), L"System", sizeof("System")-1);
}
//
// Store the old key
//
DoSpecificExtension->DriverOldRegistryPath = driverKeyName;
//
// Store the new key
// If this is not a Whistler driver, then use the old key.
//
if (EnableNewRegistryKey) {
#if _X86_
if (DriverObjectExtension->HwInitData.HwInitDataSize > SIZE_OF_W2K_VIDEO_HW_INITIALIZATION_DATA)
#endif // _X86_
VpEnableNewRegistryKey(FdoExtension,
DoSpecificExtension,
RegistryPath,
FdoExtension->RegistryIndex + 1);
}
//
// Store the path name of the location of the driver in the registry.
//
if (DoSpecificExtension->DriverNewRegistryPath != NULL) {
DoSpecificExtension->DriverRegistryPath =
DoSpecificExtension->DriverNewRegistryPath;
DoSpecificExtension->DriverRegistryPathLength =
DoSpecificExtension->DriverNewRegistryPathLength;
} else {
DoSpecificExtension->DriverRegistryPath =
DoSpecificExtension->DriverOldRegistryPath;
DoSpecificExtension->DriverRegistryPathLength =
DoSpecificExtension->DriverOldRegistryPathLength;
}
//
// NOTE:
//
// We only want to reinitialize the device once the Boot sequence has
// been completed and the HAL does not need to access the device again.
// So the initialization entry point will be called when the device is
// opened.
//
if (!NT_SUCCESS(pVideoPortCreateDeviceName(L"\\DosDevices\\DISPLAY",
DoSpecificExtension->DeviceNumber + 1,
&deviceLinkUnicodeString,
deviceLinkBuffer)))
{
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: Could not create device subpath number\n"));
ExFreePool(driverKeyName);
IoDeleteDevice(DeviceObject);
return STATUS_INSUFFICIENT_RESOURCES;
}
ntStatus = IoCreateSymbolicLink(&deviceLinkUnicodeString,
&deviceName->Name);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: SymbolicLink Creation failed\n"));
ExFreePool(driverKeyName);
IoDeleteDevice(DeviceObject);
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Once initialization is finished, load the required information in the
// registry so that the appropriate display drivers can be loaded.
//
ntStatus = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
VideoClassString,
deviceName->Name.Buffer,
REG_SZ,
DoSpecificExtension->DriverRegistryPath,
DoSpecificExtension->DriverRegistryPathLength +
sizeof(UNICODE_NULL));
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: Could not store name in DeviceMap\n"));
}
//
// Tell win32k how many objects to try to open
//
MaxObjectNumber = VideoDeviceNumber - 1;
ntStatus = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
VideoClassString,
L"MaxObjectNumber",
REG_DWORD,
&MaxObjectNumber,
sizeof(ULONG));
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error, "VIDEOPRT: VideoPortCreateSecondaryDisplay: Could not store name in DeviceMap\n"));
}
//
// Register and enable the interface
//
VpEnableAdapterInterface(DoSpecificExtension);
//
// Finally, tell the system we are done with Device Initialization
//
DeviceObject->Flags |= DO_BUFFERED_IO | DO_POWER_PAGABLE;
DeviceObject->Flags &= ~(DO_DEVICE_INITIALIZING | DO_POWER_INRUSH);
VideoDeviceNumber++;
FdoExtension->RegistryIndex++;
}
}
if (NT_SUCCESS(ntStatus)) {
*SecondaryDeviceExtension = (PVOID)(DoSpecificExtension + 1);
DriverObjectExtension->HwInitData.StartingDeviceNumber++;
//
// Mark the primary view
//
((PDEVICE_SPECIFIC_EXTENSION)(FdoExtension + 1))->DualviewFlags = VIDEO_DUALVIEW_PRIMARY;
}
return ntStatus;
}
#if DBG
PIO_RESOURCE_REQUIREMENTS_LIST
BuildRequirements(
PCM_RESOURCE_LIST pcmResourceList
)
{
ULONG i;
PCM_PARTIAL_RESOURCE_DESCRIPTOR pcmDescript;
PIO_RESOURCE_REQUIREMENTS_LIST Requirements;
PIO_RESOURCE_DESCRIPTOR pioDescript;
ULONG RequirementsListSize;
ULONG RequirementCount;
pVideoDebugPrint((Trace, "VIDEOPRT: BuildRequirements()\n"));
RequirementCount = pcmResourceList->List[0].PartialResourceList.Count;
RequirementsListSize = sizeof(IO_RESOURCE_REQUIREMENTS_LIST) +
((RequirementCount - 1) *
sizeof(IO_RESOURCE_DESCRIPTOR));
Requirements = (PIO_RESOURCE_REQUIREMENTS_LIST) ExAllocatePool(
PagedPool,
RequirementsListSize);
Requirements->ListSize = RequirementsListSize;
Requirements->InterfaceType = pcmResourceList->List[0].InterfaceType;
Requirements->BusNumber = pcmResourceList->List[0].BusNumber;
Requirements->SlotNumber = -1; // ???
Requirements->AlternativeLists = 0; // ???
Requirements->List[0].Version = pcmResourceList->List[0].PartialResourceList.Version;
Requirements->List[0].Revision = pcmResourceList->List[0].PartialResourceList.Revision;
Requirements->List[0].Count = RequirementCount;
pcmDescript = &(pcmResourceList->List[0].PartialResourceList.PartialDescriptors[0]);
pioDescript = &(Requirements->List[0].Descriptors[0]);
for (i=0; i<RequirementCount; i++) {
pioDescript->Option = IO_RESOURCE_PREFERRED;
pioDescript->Type = pcmDescript->Type;
pioDescript->ShareDisposition = pcmDescript->ShareDisposition;
pioDescript->Flags = pcmDescript->Flags;
switch (pcmDescript->Type) {
case CmResourceTypePort:
pioDescript->u.Port.Length = pcmDescript->u.Port.Length;
pioDescript->u.Port.Alignment = 1;
pioDescript->u.Port.MinimumAddress =
pioDescript->u.Port.MaximumAddress = pcmDescript->u.Port.Start;
break;
case CmResourceTypeMemory:
pioDescript->u.Memory.Length = pcmDescript->u.Memory.Length;
pioDescript->u.Memory.Alignment = 1;
pioDescript->u.Memory.MinimumAddress =
pioDescript->u.Memory.MaximumAddress = pcmDescript->u.Memory.Start;
break;
default:
//
// We don't have to handle the other stuff, because we only
// want to report Ports and Memory to the system.
//
break;
}
pioDescript++;
pcmDescript++;
}
return Requirements;
}
VOID
DumpRequirements(
PIO_RESOURCE_REQUIREMENTS_LIST Requirements
)
{
ULONG i;
PIO_RESOURCE_DESCRIPTOR pioDescript;
ULONG RequirementsListSize;
ULONG RequirementCount = Requirements->List[0].Count;
char *Table[] = { "Internal",
"Isa",
"Eisa",
"MicroChannel",
"TurboChannel",
"PCIBus",
"VMEBus",
"NuBus",
"PCMCIABus",
"CBus",
"MPIBus",
"MPSABus",
"ProcessorInternal",
"InternalPowerBus",
"PNPISABus",
"MaximumInterfaceType"
};
pVideoDebugPrint((Info, "VIDEOPRT: Beginning dump of requirements list:\n"));
pVideoDebugPrint((Info, "ListSize: 0x%x\n"
"InterfaceType: %s\n"
"BusNumber: 0x%x\n"
"SlotNumber: 0x%x\n"
"AlternativeLists: 0x%x\n",
Requirements->ListSize,
Table[Requirements->InterfaceType],
Requirements->BusNumber,
Requirements->SlotNumber,
Requirements->AlternativeLists));
pVideoDebugPrint((Info, "List[0].Version: 0x%x\n"
"List[0].Revision: 0x%x\n"
"List[0].Count: 0x%x\n",
Requirements->List[0].Version,
Requirements->List[0].Revision,
Requirements->List[0].Count));
pioDescript = &(Requirements->List[0].Descriptors[0]);
for (i=0; i<RequirementCount; i++) {
pVideoDebugPrint((Info, "\n"
"Option: 0x%x\n"
"Type: 0x%x\n"
"ShareDisposition: 0x%x\n"
"Flags: 0x%x\n",
pioDescript->Option,
pioDescript->Type,
pioDescript->ShareDisposition,
pioDescript->Flags));
switch (pioDescript->Type) {
case CmResourceTypePort:
pVideoDebugPrint((Info, "\nPort...\n"
"\tLength: 0x%x\n"
"\tAlignment: 0x%x\n"
"\tMinimumAddress: 0x%x\n"
"\tMaximumAddress: 0x%x\n",
pioDescript->u.Port.Length,
pioDescript->u.Port.Alignment,
pioDescript->u.Port.MinimumAddress,
pioDescript->u.Port.MaximumAddress));
break;
case CmResourceTypeMemory:
pVideoDebugPrint((Info, "\nMemory...\n"
"\tLength: 0x%x\n"
"\tAlignment: 0x%x\n"
"\tMinimumAddress: 0x%x\n"
"\tMaximumAddress: 0x%x\n",
pioDescript->u.Memory.Length,
pioDescript->u.Memory.Alignment,
pioDescript->u.Memory.MinimumAddress,
pioDescript->u.Memory.MaximumAddress));
break;
case CmResourceTypeInterrupt:
pVideoDebugPrint((Info, "\nInterrupt...\n"
"\tMinimum Vector: 0x%x\n"
"\tMaximum Vector: 0x%x\n",
pioDescript->u.Interrupt.MinimumVector,
pioDescript->u.Interrupt.MaximumVector));
break;
default:
//
// We don't have to handle the other stuff, because we only
// want to report Ports and Memory to the system.
//
break;
}
pioDescript++;
}
return;
}
VOID
DumpResourceList(
PCM_RESOURCE_LIST pcmResourceList)
{
ULONG i, j;
PCM_FULL_RESOURCE_DESCRIPTOR pcmFull;
PCM_PARTIAL_RESOURCE_LIST pcmPartial;
PCM_PARTIAL_RESOURCE_DESCRIPTOR pcmDescript;
pVideoDebugPrint((Trace, "VIDEOPRT: Beginning dump of resource list:\n"));
pcmFull = &(pcmResourceList->List[0]);
for (i=0; i<pcmResourceList->Count; i++) {
pVideoDebugPrint((Info, "List[%d]\n", i));
pVideoDebugPrint((Info, "InterfaceType = 0x%x\n", pcmFull->InterfaceType));
pVideoDebugPrint((Info, "BusNumber = 0x%x\n", pcmFull->BusNumber));
pcmPartial = &(pcmFull->PartialResourceList);
pVideoDebugPrint((Info, "Version = 0x%x\n", pcmPartial->Version));
pVideoDebugPrint((Info, "Revision = 0x%x\n", pcmPartial->Revision));
pcmDescript = &(pcmPartial->PartialDescriptors[0]);
for (j=0; j<pcmPartial->Count; j++) {
switch (pcmDescript->Type) {
case CmResourceTypePort:
pVideoDebugPrint((Info, "Port: 0x%x Length: 0x%x\n",
pcmDescript->u.Port.Start.LowPart,
pcmDescript->u.Port.Length));
break;
case CmResourceTypeInterrupt:
pVideoDebugPrint((Info, "Interrupt: 0x%x Level: 0x%x\n",
pcmDescript->u.Interrupt.Vector,
pcmDescript->u.Interrupt.Level));
break;
case CmResourceTypeMemory:
pVideoDebugPrint((Info, "Start: 0x%x Length: 0x%x\n",
pcmDescript->u.Memory.Start.LowPart,
pcmDescript->u.Memory.Length));
break;
case CmResourceTypeDma:
pVideoDebugPrint((Info, "Dma Channel: 0x%x Port: 0x%x\n",
pcmDescript->u.Dma.Channel,
pcmDescript->u.Dma.Port));
break;
}
pcmDescript++;
}
pcmFull = (PCM_FULL_RESOURCE_DESCRIPTOR) pcmDescript;
}
pVideoDebugPrint((Info, "VIDEOPRT: EndResourceList\n"));
}
VOID
DumpUnicodeString(
IN PUNICODE_STRING p
)
{
PUSHORT pus = p->Buffer;
UCHAR buffer[256]; // the string better not be longer than 255 chars!
PUCHAR puc = buffer;
ULONG i;
for (i = 0; i < p->Length; i++) {
*puc++ = (UCHAR) *pus++;
}
*puc = 0; // null terminate the string
pVideoDebugPrint((Info, "VIDEOPRT: UNICODE STRING: %s\n", buffer));
}
#endif
VP_STATUS
VideoPortEnumerateChildren(
IN PVOID HwDeviceExtension,
IN PVOID Reserved
)
/*++
Routine Description:
Allows a miniport to force a re-enumeration of it's children.
Arguments:
HwDeviceExtension - The miniports device extension
Reserved - Not currently used, should be NULL.
Returns:
Status
--*/
{
PFDO_EXTENSION fdoExtension = GET_FDO_EXT (HwDeviceExtension);
PDEVICE_OBJECT pFDO = fdoExtension->FunctionalDeviceObject;
PDEVICE_OBJECT pPDO = fdoExtension->PhysicalDeviceObject;
ASSERT(Reserved == NULL);
IoInvalidateDeviceRelations(pPDO, BusRelations);
return NO_ERROR;
}
VIDEOPORT_API
VP_STATUS
VideoPortQueryServices(
IN PVOID pHwDeviceExtension,
IN VIDEO_PORT_SERVICES servicesType,
IN OUT PINTERFACE pInterface
)
/*++
Routine Description:
This routine exposes interfaces to services supported by the videoprt.
Arguments:
pHwDeviceExtension - Points to per-adapter device extension.
servicesType - Requested services type.
pInterface - Points to services interface structure.
Returns:
NO_ERROR - Valid interface in the pInterface.
Error code - Unsupported / unavailable services.
--*/
{
VP_STATUS vpStatus;
PAGED_CODE();
ASSERT(NULL != pHwDeviceExtension);
ASSERT(NULL != pInterface);
ASSERT(IS_HW_DEVICE_EXTENSION(pHwDeviceExtension) == TRUE);
if (VideoPortServicesAGP == servicesType)
{
if ((pInterface->Version == VIDEO_PORT_AGP_INTERFACE_VERSION_2) &&
(pInterface->Size == sizeof (VIDEO_PORT_AGP_INTERFACE_2)))
{
PVIDEO_PORT_AGP_INTERFACE_2 pAgpInterface = (PVIDEO_PORT_AGP_INTERFACE_2)pInterface;
vpStatus = VpGetAgpServices2(pHwDeviceExtension, pAgpInterface);
}
else if ((pInterface->Version == VIDEO_PORT_AGP_INTERFACE_VERSION_1) &&
(pInterface->Size == sizeof (VIDEO_PORT_AGP_INTERFACE)))
{
PVIDEO_PORT_AGP_INTERFACE pAgpInterface = (PVIDEO_PORT_AGP_INTERFACE)pInterface;
pAgpInterface->Context = pHwDeviceExtension;
pAgpInterface->InterfaceReference = VpInterfaceDefaultReference;
pAgpInterface->InterfaceDereference = VpInterfaceDefaultDereference;
if (VideoPortGetAgpServices(pHwDeviceExtension,
(PVIDEO_PORT_AGP_SERVICES)&(pAgpInterface->AgpReservePhysical)) == TRUE)
{
//
// Reference the interface before handing it out.
//
pAgpInterface->InterfaceReference(pAgpInterface->Context);
vpStatus = NO_ERROR;
}
else
{
vpStatus = ERROR_DEV_NOT_EXIST;
}
}
else
{
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortQueryServices: Unsupported interface version\n"));
vpStatus = ERROR_INVALID_PARAMETER;
}
}
else if (VideoPortServicesI2C == servicesType)
{
if ((pInterface->Version == VIDEO_PORT_I2C_INTERFACE_VERSION_2) &&
(pInterface->Size == sizeof (VIDEO_PORT_I2C_INTERFACE_2)))
{
PVIDEO_PORT_I2C_INTERFACE_2 pI2CInterface = (PVIDEO_PORT_I2C_INTERFACE_2)pInterface;
pI2CInterface->Context = pHwDeviceExtension;
pI2CInterface->InterfaceReference = VpInterfaceDefaultReference;
pI2CInterface->InterfaceDereference = VpInterfaceDefaultDereference;
pI2CInterface->I2CStart = I2CStart2;
pI2CInterface->I2CStop = I2CStop2;
pI2CInterface->I2CWrite = I2CWrite2;
pI2CInterface->I2CRead = I2CRead2;
//
// Reference the interface before handing it out.
//
pI2CInterface->InterfaceReference(pI2CInterface->Context);
vpStatus = NO_ERROR;
}
else if ((pInterface->Version == VIDEO_PORT_I2C_INTERFACE_VERSION_1) &&
(pInterface->Size == sizeof (VIDEO_PORT_I2C_INTERFACE)))
{
PVIDEO_PORT_I2C_INTERFACE pI2CInterface = (PVIDEO_PORT_I2C_INTERFACE)pInterface;
pI2CInterface->Context = pHwDeviceExtension;
pI2CInterface->InterfaceReference = VpInterfaceDefaultReference;
pI2CInterface->InterfaceDereference = VpInterfaceDefaultDereference;
pI2CInterface->I2CStart = I2CStart;
pI2CInterface->I2CStop = I2CStop;
pI2CInterface->I2CWrite = I2CWrite;
pI2CInterface->I2CRead = I2CRead;
//
// Reference the interface before handing it out.
//
pI2CInterface->InterfaceReference(pI2CInterface->Context);
vpStatus = NO_ERROR;
}
else
{
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortQueryServices: Unsupported interface version\n"));
vpStatus = ERROR_INVALID_PARAMETER;
}
}
else if (VideoPortServicesInt10 == servicesType)
{
if ((pInterface->Version == VIDEO_PORT_INT10_INTERFACE_VERSION_1) &&
(pInterface->Size == sizeof(VIDEO_PORT_INT10_INTERFACE)))
{
PVIDEO_PORT_INT10_INTERFACE pInt10 = (PVIDEO_PORT_INT10_INTERFACE)pInterface;
pInt10->Context = pHwDeviceExtension;
pInt10->InterfaceReference = VpInterfaceDefaultReference;
pInt10->InterfaceDereference = VpInterfaceDefaultDereference;
pInt10->Int10AllocateBuffer = VpInt10AllocateBuffer;
pInt10->Int10FreeBuffer = VpInt10FreeBuffer;
pInt10->Int10ReadMemory = VpInt10ReadMemory;
pInt10->Int10WriteMemory = VpInt10WriteMemory;
pInt10->Int10CallBios = VpInt10CallBios;
//
// Reference the interface before handing it out.
//
pInt10->InterfaceReference(pInt10->Context);
vpStatus = NO_ERROR;
}
else
{
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortQueryServices: Unsupported interface version\n"));
vpStatus = ERROR_INVALID_PARAMETER;
}
}
else
{
pVideoDebugPrint((Warn, "VIDEOPRT: VideoPortQueryServices: Unsupported service type\n"));
vpStatus = ERROR_INVALID_PARAMETER;
}
return vpStatus;
} // VideoPortQueryServices()
VIDEOPORT_API
LONGLONG
VideoPortQueryPerformanceCounter(
IN PVOID pHwDeviceExtension,
OUT PLONGLONG pllPerformanceFrequency OPTIONAL
)
/*++
Routine Description:
This routine provides the finest grained running count available in the system.
Use this routine as infrequently as possible. Depending on the platform,
VideoPortQueryPerformanceCounter can disable system-wide interrupts for a minimal interval.
Consequently, calling this routine frequently or repeatedly, as in an iteration, defeats its
purpose of returning very fine-grained, running time-stamp information. Calling this routine
too frequently can degrade I/O performance for the calling driver and for the system as a whole.
Arguments:
pHwDeviceExtension - Points to per-adapter device extension.
pllPerformanceFrequency - Specifies an optional pointer to a variable that is to receive the
performance counter frequency.
Returns:
The performance counter value in units of ticks.
--*/
{
LARGE_INTEGER li;
//
// No ASSERT() allowed - nonpagable code.
//
li = KeQueryPerformanceCounter((PLARGE_INTEGER)pllPerformanceFrequency);
return *((PLONGLONG) &li);
} // VideoPortQueryPerformanceCounter()
VOID
VpInterfaceDefaultReference(
IN PVOID pContext
)
/*++
Routine Description:
This routine is default callback for interfaces exposed from the videoprt.
Should be called by the client before it starts using an interface.
Arguments:
pContext - Context returned by the VideoPortQueryServices() in the
pInterface->Context field.
--*/
{
UNREFERENCED_PARAMETER(pContext);
PAGED_CODE();
} // VpInterfaceDefaultReference()
VOID
VpInterfaceDefaultDereference(
IN PVOID pContext
)
/*++
Routine Description:
This routine is default callback for interfaces exposed from the videoprt.
Should be called by the client when it stops using an interface.
Arguments:
pContext - Context returned by the VideoPortQueryServices() in the
pInterface->Context field.
--*/
{
UNREFERENCED_PARAMETER(pContext);
PAGED_CODE();
} // VpInterfaceDefaultDereference()
BOOLEAN
VpEnableAdapterInterface(
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension
)
/*++
Routine Description:
This routine registers and enables a display adapter interface.
It also writes the interface name to the registry.
Arguments:
DoSpecificExtension - Pointer to the functional device object
specific extension.
--*/
{
PFDO_EXTENSION fdoExtension = NULL;
UNICODE_STRING SymbolicLinkName;
BOOLEAN Success = FALSE;
UNICODE_STRING VolatileSettingsString;
OBJECT_ATTRIBUTES VolatileSettingsAttributes;
HANDLE VolatileSettingsKey;
PAGED_CODE();
ASSERT ((DoSpecificExtension != NULL) &&
(DoSpecificExtension->ExtensionType == TypeDeviceSpecificExtension));
VolatileSettingsString.Buffer = NULL;
fdoExtension = DoSpecificExtension->pFdoExtension;
ASSERT (IS_FDO(fdoExtension));
if (fdoExtension->PhysicalDeviceObject == NULL) {
//
// This fdo doesn't have a physical device object (e.g. vga).
// In this case, we can't create an interface.
//
goto Fallout;
}
//
// Register the interface
//
if (IoRegisterDeviceInterface(fdoExtension->PhysicalDeviceObject,
&GUID_DISPLAY_ADAPTER_INTERFACE,
NULL,
&SymbolicLinkName) != STATUS_SUCCESS) {
goto Fallout;
}
//
// Enable the interface
//
if (IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE) != STATUS_SUCCESS) {
goto Cleanup;
}
//
// Write the interface name to registry
//
ASSERT (DoSpecificExtension->DriverRegistryPath != NULL);
VolatileSettingsString.Length = 0;
VolatileSettingsString.MaximumLength =
(USHORT)DoSpecificExtension->DriverRegistryPathLength + 40;
VolatileSettingsString.Buffer = ExAllocatePoolWithTag(
PagedPool | POOL_COLD_ALLOCATION,
VolatileSettingsString.MaximumLength,
VP_TAG);
if (VolatileSettingsString.Buffer == NULL) {
goto Cleanup;
}
if (RtlAppendUnicodeToString(&VolatileSettingsString,
DoSpecificExtension->DriverRegistryPath) != STATUS_SUCCESS) {
goto Cleanup;
}
if (RtlAppendUnicodeToString(&VolatileSettingsString,
L"\\VolatileSettings") != STATUS_SUCCESS) {
goto Cleanup;
}
InitializeObjectAttributes(&VolatileSettingsAttributes,
&VolatileSettingsString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
if (ZwCreateKey(&VolatileSettingsKey,
GENERIC_READ | GENERIC_WRITE,
&VolatileSettingsAttributes,
0L,
NULL,
REG_OPTION_VOLATILE,
NULL) != STATUS_SUCCESS) {
goto Cleanup;
}
if (RtlWriteRegistryValue(
RTL_REGISTRY_ABSOLUTE,
VolatileSettingsString.Buffer,
L"{5b45201d-f2f2-4f3b-85bb-30ff1f953599}",
REG_BINARY,
(PVOID)SymbolicLinkName.Buffer,
SymbolicLinkName.Length) == STATUS_SUCCESS) {
Success = TRUE;
}
Cleanup:
if (VolatileSettingsString.Buffer != NULL) {
ExFreePool(VolatileSettingsString.Buffer);
}
RtlFreeUnicodeString(&SymbolicLinkName);
Fallout:
if (Success) {
pVideoDebugPrint((Trace, "VideoPort - Device interface ok.\n"));
} else {
pVideoDebugPrint((Warn,
"VideoPort - Couldn't register, enable or save the device interface.\n"));
}
return Success;
} // VpEnableAdapterInterface
VOID
VpDisableAdapterInterface(
PFDO_EXTENSION fdoExtension
)
/*++
Routine Description:
This routine disables the display adapter interface.
Arguments:
fdoExtension - Pointer to the functional device object extension.
--*/
{
PWSTR SymbolicLinkList = NULL;
UNICODE_STRING SymbolicLinkName;
PAGED_CODE();
ASSERT ((fdoExtension != NULL) && IS_FDO(fdoExtension));
if (fdoExtension->PhysicalDeviceObject == NULL) {
//
// This fdo doesn't have a physical device object (e.g. vga ...).
// In this case, we didn't create any interface so there is
// nothing to disable.
//
return;
}
//
// There is no need to remove the InterfaceName from the registry
// as the parent key is volatile.
//
//
// Disable the interface
//
if (IoGetDeviceInterfaces(&GUID_DISPLAY_ADAPTER_INTERFACE,
fdoExtension->PhysicalDeviceObject,
0,
&SymbolicLinkList) != STATUS_SUCCESS) {
pVideoDebugPrint((Warn,
"VideoPort - Could not find any enabled device interfaces.\n"));
return;
}
RtlInitUnicodeString(&SymbolicLinkName, SymbolicLinkList);
if (SymbolicLinkName.Length > 0) {
if (IoSetDeviceInterfaceState(&SymbolicLinkName,
FALSE) != STATUS_SUCCESS) {
pVideoDebugPrint((Warn,
"VideoPort - Could not disable the device interface.\n"));
}
}
ExFreePool((PVOID)SymbolicLinkList);
} // VpDisableAdapterInterface
VOID
VpEnableNewRegistryKey(
PFDO_EXTENSION FdoExtension,
PDEVICE_SPECIFIC_EXTENSION DoSpecificExtension,
PUNICODE_STRING RegistryPath,
ULONG RegistryIndex
)
{
PKEY_VALUE_PARTIAL_INFORMATION GUIDBuffer = NULL;
ULONG GUIDLength = 0;
LPWSTR Buffer = NULL;
HANDLE GuidKey = NULL;
HANDLE NewDeviceKey = NULL;
HANDLE ServiceSubKey = NULL;
UNICODE_STRING UnicodeString, newGuidStr;
OBJECT_ATTRIBUTES ObjectAttributes;
ULONG Len;
GUID newGuid;
BOOLEAN IsLegacy = FALSE;
NTSTATUS ntStatus = STATUS_UNSUCCESSFUL;
BOOLEAN IsNewKey = FALSE;
PWSTR pService = NULL;
ULONG ServiceLen = 0;
ASSERT((DoSpecificExtension->DriverNewRegistryPath == NULL) &&
(DoSpecificExtension->DriverNewRegistryPathLength == 0));
ASSERT(DoSpecificExtension->DriverOldRegistryPath != NULL);
ASSERT(EnableNewRegistryKey);
newGuidStr.Buffer = NULL;
//
// Get the service name
//
pService = RegistryPath->Buffer +
(RegistryPath->Length / sizeof(WCHAR)) - 1;
while ((pService > RegistryPath->Buffer) &&
(*pService != L'\\')) {
pService--;
ServiceLen++;
}
ASSERT (*pService == L'\\');
pService++;
Buffer = (LPWSTR)ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
(ServiceLen + 1) * sizeof(WCHAR),
VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to allocate memory.\n"));
goto Fallout;
}
RtlZeroMemory(Buffer, (ServiceLen + 1) * sizeof(WCHAR));
wcsncpy(Buffer, pService, ServiceLen);
pService = Buffer;
Buffer = NULL;
//
// Try to open the PnP device key
//
if ((FdoExtension->PhysicalDeviceObject == NULL) ||
(IoOpenDeviceRegistryKey(FdoExtension->PhysicalDeviceObject,
PLUGPLAY_REGKEY_DEVICE,
KEY_READ | KEY_WRITE,
&GuidKey) != STATUS_SUCCESS)) {
//
// We failed to open the PnP device key.
// Try to open the service subkey instead.
//
if (!VpGetServiceSubkey(RegistryPath,
&GuidKey)) {
GuidKey = NULL;
goto Fallout;
}
IsLegacy = TRUE;
}
//
// Is the GUID there?
//
RtlInitUnicodeString(&UnicodeString, SZ_GUID);
ntStatus = ZwQueryValueKey(GuidKey,
&UnicodeString,
KeyValuePartialInformation,
(PVOID)NULL,
0,
&GUIDLength);
if ((ntStatus == STATUS_BUFFER_OVERFLOW) ||
(ntStatus == STATUS_BUFFER_TOO_SMALL)) {
//
// The GUID is there.
// Allocate a buffer large enough to contain the entire key data value.
//
GUIDBuffer = ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
GUIDLength,
VP_TAG);
if (GUIDBuffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to allocate memory.\n"));
goto Fallout;
}
//
// Get the GUID from the registry
//
ntStatus = ZwQueryValueKey(GuidKey,
&UnicodeString,
KeyValuePartialInformation,
GUIDBuffer,
GUIDLength,
&GUIDLength);
if (!NT_SUCCESS(ntStatus)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to get the GUID from the registry.\n"));
goto Fallout;
}
//
// Build the new registry path
//
Len = (wcslen(SZ_VIDEO_DEVICES) + 8) * sizeof(WCHAR) + GUIDBuffer->DataLength;
Buffer = (LPWSTR)ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
Len,
VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to allocate memory.\n"));
goto Fallout;
}
RtlZeroMemory(Buffer, Len);
wcscpy(Buffer, SZ_VIDEO_DEVICES);
wcscat(Buffer, L"\\");
wcsncpy(Buffer + wcslen(Buffer),
(LPWSTR)GUIDBuffer->Data,
GUIDBuffer->DataLength / sizeof(WCHAR));
ASSERT (RegistryIndex <= 9999);
swprintf(Buffer + wcslen(Buffer), L"\\%04d", RegistryIndex);
//
// Is the key already there?
//
if (RtlCheckRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
//
// Create the new key
//
if (RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create the new key.\n"));
goto Fallout;
}
//
// Initialize the key
//
if (IsLegacy) {
VpInitializeLegacyKey(DoSpecificExtension->DriverOldRegistryPath,
Buffer);
} else {
VpInitializeKey(FdoExtension->PhysicalDeviceObject,
Buffer);
}
}
} else {
//
// The GUID is not there so allocate a new one
//
// !!! Add special case for VGA, MNMDD & RDPCDD
//
ntStatus = ExUuidCreate(&newGuid);
if ((ntStatus != STATUS_SUCCESS) &&
(ntStatus != RPC_NT_UUID_LOCAL_ONLY)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create a new GUID.\n"));
goto Fallout;
}
if (RtlStringFromGUID(&newGuid, &newGuidStr) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to convert the GUID to a string.\n"));
newGuidStr.Buffer = NULL;
goto Fallout;
}
//
// Upcase the string
//
RtlUpcaseUnicodeString(&newGuidStr, &newGuidStr, FALSE);
//
// Build the new registry path
//
Len = (wcslen(SZ_VIDEO_DEVICES) +
wcslen(newGuidStr.Buffer) +
wcslen(SZ_COMMON_SUBKEY) + 8) * sizeof(WCHAR);
Buffer = (LPWSTR)ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION,
Len,
VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to allocate memory.\n"));
goto Fallout;
}
RtlZeroMemory(Buffer, Len);
wcscpy(Buffer, SZ_VIDEO_DEVICES);
if (RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create the new key.\n"));
goto Fallout;
}
wcscat(Buffer, L"\\");
wcscat(Buffer, newGuidStr.Buffer);
if (RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create the new key.\n"));
goto Fallout;
}
//
// Save the service name
//
Len = wcslen(Buffer);
wcscat(Buffer, L"\\");
wcscat(Buffer, SZ_COMMON_SUBKEY);
if (RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create the new key.\n"));
goto Fallout;
}
if (RtlWriteRegistryValue(
RTL_REGISTRY_ABSOLUTE,
Buffer,
SZ_SERVICE,
REG_SZ,
(PVOID)pService,
(ServiceLen + 1) * sizeof(WCHAR)) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to save the service name.\n"));
goto Fallout;
}
if (IsLegacy) {
ServiceSubKey = GuidKey;
} else {
if (!VpGetServiceSubkey(RegistryPath,
&ServiceSubKey)) {
ServiceSubKey = NULL;
}
}
if (ServiceSubKey != NULL) {
if (RtlWriteRegistryValue(RTL_REGISTRY_HANDLE,
ServiceSubKey,
SZ_SERVICE,
REG_SZ,
(PVOID)pService,
(ServiceLen + 1) * sizeof(WCHAR)) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to save the service name.\n"));
goto Fallout;
}
}
//
// Create the 000X subkey
//
Buffer[Len] = 0;
ASSERT (RegistryIndex == 0);
wcscat(Buffer, L"\\0000");
if (RtlCreateRegistryKey(RTL_REGISTRY_ABSOLUTE,
Buffer) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to create the new key.\n"));
goto Fallout;
}
//
// Save the new key under the PnP device key or the service subkey
//
if (RtlWriteRegistryValue(RTL_REGISTRY_HANDLE,
GuidKey,
SZ_GUID,
REG_SZ,
(PVOID)newGuidStr.Buffer,
(wcslen(newGuidStr.Buffer) + 1) * sizeof(WCHAR)) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpEnableNewRegistryKey: failed to write the new GUID to the registry.\n"));
goto Fallout;
}
//
// The key was not there, so initialize it.
//
if (IsLegacy) {
VpInitializeLegacyKey(DoSpecificExtension->DriverOldRegistryPath,
Buffer);
} else {
VpInitializeKey(FdoExtension->PhysicalDeviceObject,
Buffer);
}
}
pVideoDebugPrint((Info, "VIDEOPRT: VpEnableNewRegistryKey: %ws\n", Buffer));
//
// Initialize the new registry path fields
//
DoSpecificExtension->DriverNewRegistryPath = Buffer;
DoSpecificExtension->DriverNewRegistryPathLength = wcslen(Buffer) * sizeof(WCHAR);
Fallout:
//
// Cleanup
//
if (GUIDBuffer != NULL) {
ExFreePool(GUIDBuffer);
}
if (newGuidStr.Buffer != NULL) {
RtlFreeUnicodeString(&newGuidStr);
}
if ((DoSpecificExtension->DriverNewRegistryPath == NULL) &&
(Buffer != NULL)) {
ExFreePool(Buffer);
}
if (GuidKey != NULL) {
ZwClose(GuidKey);
}
if (!IsLegacy && (ServiceSubKey != NULL)) {
ZwClose(ServiceSubKey);
}
if (pService != NULL) {
ExFreePool(pService);
}
return;
} // VpEnableNewRegistryKey
VOID
VpInitializeKey(
PDEVICE_OBJECT PhysicalDeviceObject,
PWSTR NewRegistryPath
)
{
HANDLE NewDeviceKey = NULL;
HANDLE DriverKey = NULL;
HANDLE DriverSettingsKey = NULL;
UNICODE_STRING UnicodeString;
OBJECT_ATTRIBUTES ObjectAttributes;
ASSERT (PhysicalDeviceObject != NULL);
ASSERT (NewRegistryPath != NULL);
//
// Open the new key
//
RtlInitUnicodeString(&UnicodeString, NewRegistryPath);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
if (ZwOpenKey(&NewDeviceKey,
KEY_ALL_ACCESS,
&ObjectAttributes) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpInitializeKey: failed to open the new key.\n"));
NewDeviceKey = NULL;
goto Fallout;
}
//
// Open the PnP driver key
//
if (IoOpenDeviceRegistryKey(PhysicalDeviceObject,
PLUGPLAY_REGKEY_DRIVER,
KEY_READ | KEY_WRITE,
&DriverKey) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpInitializeKey: could not open the driver key.\n"));
DriverKey = NULL;
goto Fallout;
}
RtlInitUnicodeString(&UnicodeString, SZ_INITIAL_SETTINGS);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
DriverKey,
NULL);
//
// Open the "Settings" key.
// The class installer saved the initial settings there.
//
if (ZwOpenKey(&DriverSettingsKey,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpInitializeKey: failed to open the driver settings key.\n"));
DriverSettingsKey = NULL;
goto Fallout;
}
//
// Copy the settings
//
VpCopyRegistry(DriverSettingsKey,
NewDeviceKey,
NULL,
NULL);
Fallout:
if (DriverSettingsKey != NULL) {
ZwClose(DriverSettingsKey);
}
if (DriverKey != NULL) {
ZwClose(DriverKey);
}
if (NewDeviceKey != NULL) {
ZwClose(NewDeviceKey);
}
} // VpInitializeKey
VOID
VpInitializeLegacyKey(
PWSTR OldRegistryPath,
PWSTR NewRegistryPath
)
{
HANDLE NewDeviceKey = NULL;
HANDLE OldDeviceKey = NULL;
UNICODE_STRING UnicodeString;
OBJECT_ATTRIBUTES ObjectAttributes;
ASSERT (NewRegistryPath != NULL);
//
// Open the new key
//
RtlInitUnicodeString(&UnicodeString, NewRegistryPath);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
if (ZwOpenKey(&NewDeviceKey,
KEY_ALL_ACCESS,
&ObjectAttributes) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpInitializeLegacyKey: failed to open the new key.\n"));
NewDeviceKey = NULL;
goto Fallout;
}
//
// Open the old key
//
RtlInitUnicodeString(&UnicodeString,
OldRegistryPath);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
if (ZwOpenKey(&OldDeviceKey,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpInitializeLegacyKey: failed to open the old key.\n"));
OldDeviceKey = NULL;
goto Fallout;
}
//
// Copy the settings
//
VpCopyRegistry(OldDeviceKey,
NewDeviceKey,
NULL,
NULL);
Fallout:
if (NewDeviceKey != NULL) {
ZwClose(NewDeviceKey);
}
if (OldDeviceKey != NULL) {
ZwClose(OldDeviceKey);
}
} // VpInitializeLegacyKey
NTSTATUS
VpCopyRegistry(
HANDLE hKeyRootSrc,
HANDLE hKeyRootDst,
PWSTR SrcKeyPath,
PWSTR DstKeyPath
)
/*++
Routine Description:
This routine recursively copies a src key to a destination key.
Arguments:
hKeyRootSrc: Handle to root src key
hKeyRootDst: Handle to root dst key
SrcKeyPath: src root key relative path to the subkey which needs to be
recursively copied. if this is null SourceKey is the key
from which the recursive copy is to be done.
DstKeyPath: dst root key relative path to the subkey which needs to be
recursively copied. if this is null DestinationKey is the key
from which the recursive copy is to be done.
Return Value:
Status is returned.
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING UnicodeString;
HANDLE hKeySrc = NULL, hKeyDst = NULL;
ULONG ResultLength, Index;
PKEY_BASIC_INFORMATION KeyInfo;
PKEY_VALUE_FULL_INFORMATION ValueInfo;
PWSTR ValueName;
ULONG BufferSize = 512;
PVOID Buffer = NULL;
//
// Get a handle to the source key
//
if(SrcKeyPath == NULL) {
hKeySrc = hKeyRootSrc;
} else {
//
// Open the Src key
//
RtlInitUnicodeString(&UnicodeString, SrcKeyPath);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
hKeyRootSrc,
NULL);
Status = ZwOpenKey(&hKeySrc,
KEY_READ,
&ObjectAttributes);
if(!NT_SUCCESS(Status)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to open the source key.\n"));
hKeySrc = NULL;
goto Fallout;
}
}
//
// Get a handle to the destination key
//
if(DstKeyPath == NULL) {
hKeyDst = hKeyRootDst;
} else {
//
// Create the destination key.
//
RtlInitUnicodeString(&UnicodeString, DstKeyPath);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
hKeyRootDst,
NULL);
Status = ZwCreateKey(&hKeyDst,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes,
0,
NULL,
REG_OPTION_NON_VOLATILE,
NULL);
if (!NT_SUCCESS(Status)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to create the destination key.\n"));
hKeyDst = NULL;
goto Fallout;
}
}
//
// Enumerate all keys in the source key and recursively
// create all subkeys
//
for (Index = 0; ;Index++) {
do {
if (Buffer == NULL) {
Buffer = ExAllocatePoolWithTag(PagedPool,
BufferSize,
VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to allocate memory.\n"));
goto Fallout;
}
}
Status = ZwEnumerateKey(hKeySrc,
Index,
KeyBasicInformation,
Buffer,
BufferSize - sizeof(WCHAR),
&ResultLength);
if (Status == STATUS_BUFFER_TOO_SMALL) {
ExFreePool(Buffer);
Buffer = NULL;
BufferSize = ResultLength + sizeof(WCHAR);
}
} while (Status == STATUS_BUFFER_TOO_SMALL);
KeyInfo = (PKEY_BASIC_INFORMATION)Buffer;
if (!NT_SUCCESS(Status)) {
if(Status == STATUS_NO_MORE_ENTRIES) {
Status = STATUS_SUCCESS;
} else {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to enumerate the subkeys.\n"));
}
break;
}
//
// Zero-terminate the subkey name
//
KeyInfo->Name[KeyInfo->NameLength / sizeof(WCHAR)] = 0;
//
// Copy the subkey
//
Status = VpCopyRegistry(hKeySrc,
hKeyDst,
KeyInfo->Name,
KeyInfo->Name);
}
//
// Enumerate all values in the source key and create all the values
// in the destination key
//
for(Index = 0; ;Index++) {
do {
if (Buffer == NULL) {
Buffer = ExAllocatePoolWithTag(PagedPool,
BufferSize,
VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to allocate memory.\n"));
goto Fallout;
}
}
Status = ZwEnumerateValueKey(hKeySrc,
Index,
KeyValueFullInformation,
Buffer,
BufferSize,
&ResultLength);
if (Status == STATUS_BUFFER_TOO_SMALL) {
ExFreePool(Buffer);
Buffer = NULL;
BufferSize = ResultLength;
}
} while (Status == STATUS_BUFFER_TOO_SMALL);
ValueInfo = (PKEY_VALUE_FULL_INFORMATION)Buffer;
if(!NT_SUCCESS(Status)) {
if(Status == STATUS_NO_MORE_ENTRIES) {
Status = STATUS_SUCCESS;
} else {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to enumerate the values.\n"));
}
break;
}
//
// Process the value found and create the value in the destination key
//
ValueName = (PWSTR)
ExAllocatePoolWithTag(PagedPool,
ValueInfo->NameLength + sizeof(WCHAR),
VP_TAG);
if (ValueName == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to allocate memory.\n"));
goto Fallout;
}
wcsncpy(ValueName,
ValueInfo->Name,
(ValueInfo->NameLength)/sizeof(WCHAR));
ValueName[(ValueInfo->NameLength)/sizeof(WCHAR)] = 0;
RtlInitUnicodeString(&UnicodeString, ValueName);
Status = ZwSetValueKey(hKeyDst,
&UnicodeString,
ValueInfo->TitleIndex,
ValueInfo->Type,
(PVOID)((PUCHAR)ValueInfo + ValueInfo->DataOffset),
ValueInfo->DataLength);
if(!NT_SUCCESS(Status)) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpCopyRegistry: failed to set the value.\n"));
}
ExFreePool(ValueName);
}
Fallout:
//
// Cleanup
//
if (Buffer != NULL) {
ExFreePool(Buffer);
}
if ((hKeySrc != NULL) && (hKeySrc != hKeyRootSrc)) {
ZwClose(hKeySrc);
}
if ((hKeyDst != NULL) && (hKeyDst != hKeyRootDst)) {
ZwClose(hKeyDst);
}
return(Status);
} // VpCopyRegistry
BOOLEAN
VpGetServiceSubkey(
PUNICODE_STRING RegistryPath,
HANDLE* pServiceSubKey
)
{
LPWSTR Buffer = NULL;
USHORT Len = 0;
OBJECT_ATTRIBUTES ObjectAttributes;
UNICODE_STRING UnicodeString;
BOOLEAN bSuccess = FALSE;
Len = RegistryPath->Length + (wcslen(SZ_COMMON_SUBKEY) + 2) * sizeof(WCHAR);
Buffer = (LPWSTR)ExAllocatePoolWithTag(PagedPool | POOL_COLD_ALLOCATION, Len, VP_TAG);
if (Buffer == NULL) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpGetServiceSubkey: failed to allocate memory.\n"));
goto Fallout;
}
RtlZeroMemory(Buffer, Len);
wcsncpy(Buffer,
RegistryPath->Buffer,
RegistryPath->Length / sizeof(WCHAR));
wcscat(Buffer, L"\\");
wcscat(Buffer, SZ_COMMON_SUBKEY);
RtlInitUnicodeString(&UnicodeString, Buffer);
InitializeObjectAttributes(&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
NULL,
NULL);
if (ZwCreateKey(pServiceSubKey,
GENERIC_READ | GENERIC_WRITE,
&ObjectAttributes,
0,
NULL,
REG_OPTION_NON_VOLATILE,
NULL) != STATUS_SUCCESS) {
pVideoDebugPrint((Error,
"VIDEOPRT: VpGetServiceSubkey: could not create the service subkey.\n"));
goto Fallout;
}
bSuccess = TRUE;
Fallout:
if (Buffer != NULL) {
ExFreePool(Buffer);
}
return bSuccess;
} // VpGetServiceSubkey
VP_STATUS
VideoPortGetVersion(
PVOID HwDeviceExtension,
PVPOSVERSIONINFO pVpOsVersionInfo
)
{
RTL_OSVERSIONINFOEXW VersionInformation;
UNREFERENCED_PARAMETER(HwDeviceExtension);
if(pVpOsVersionInfo->Size < sizeof(VPOSVERSIONINFO)) {
return ERROR_INVALID_PARAMETER;
}
RtlZeroMemory ((PVOID)(&VersionInformation), sizeof(RTL_OSVERSIONINFOEXW));
VersionInformation.dwOSVersionInfoSize = sizeof(RTL_OSVERSIONINFOEXW);
if ( STATUS_SUCCESS !=
RtlGetVersion( (PRTL_OSVERSIONINFOW) (&VersionInformation)) ) {
return ERROR_INVALID_PARAMETER;
}
pVpOsVersionInfo->MajorVersion = VersionInformation.dwMajorVersion;
pVpOsVersionInfo->MinorVersion = VersionInformation.dwMinorVersion;
pVpOsVersionInfo->BuildNumber = VersionInformation.dwBuildNumber;
pVpOsVersionInfo->ServicePackMajor = VersionInformation.wServicePackMajor;
pVpOsVersionInfo->ServicePackMinor = VersionInformation.wServicePackMinor;
return NO_ERROR;
}