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.
529 lines
14 KiB
529 lines
14 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
agplib.h
|
|
|
|
Abstract:
|
|
|
|
Private header file for the common AGP library
|
|
|
|
Author:
|
|
|
|
John Vert (jvert) 10/22/1997
|
|
|
|
Revision History:
|
|
|
|
Elliot Shmukler (elliots) 3/24/1999 - Added support for "favored" memory
|
|
ranges for AGP physical memory allocation,
|
|
fixed some bugs.
|
|
|
|
--*/
|
|
#include "agp.h"
|
|
#include "wdmguid.h"
|
|
#include "wmilib.h"
|
|
#include <devioctl.h>
|
|
#include <acpiioct.h>
|
|
|
|
//
|
|
// regstr.h uses things of type WORD, which isn't around in kernel mode.
|
|
//
|
|
|
|
#define _IN_KERNEL_
|
|
|
|
#include "regstr.h"
|
|
|
|
#define AGP_HACK_FLAG_SUBSYSTEM 0x01
|
|
#define AGP_HACK_FLAG_REVISION 0x02
|
|
|
|
typedef struct _AGP_HACK_TABLE_ENTRY {
|
|
USHORT VendorID;
|
|
USHORT DeviceID;
|
|
USHORT SubVendorID;
|
|
USHORT SubSystemID;
|
|
ULONGLONG DeviceFlags;
|
|
UCHAR RevisionID;
|
|
UCHAR Flags;
|
|
} AGP_HACK_TABLE_ENTRY, *PAGP_HACK_TABLE_ENTRY;
|
|
|
|
extern PAGP_HACK_TABLE_ENTRY AgpDeviceHackTable;
|
|
extern PAGP_HACK_TABLE_ENTRY AgpGlobalHackTable;
|
|
|
|
typedef
|
|
ULONG_PTR
|
|
(*PCRITICALROUTINE)(
|
|
IN PVOID Extension,
|
|
IN PVOID Context
|
|
);
|
|
|
|
typedef struct _AGP_CRITICAL_ROUTINE_CONTEXT {
|
|
|
|
volatile LONG Gate;
|
|
volatile LONG Barrier;
|
|
|
|
PCRITICALROUTINE Routine;
|
|
PVOID Extension;
|
|
PVOID Context;
|
|
|
|
} AGP_CRITICAL_ROUTINE_CONTEXT, *PAGP_CRITICAL_ROUTINE_CONTEXT;
|
|
|
|
//
|
|
// Define common device extension
|
|
//
|
|
typedef enum _AGP_EXTENSION_TYPE {
|
|
AgpTargetFilter,
|
|
AgpMasterFilter
|
|
} AGP_EXTENSION_TYPE;
|
|
|
|
#define TARGET_SIG 'TpgA'
|
|
#define MASTER_SIG 'MpgA'
|
|
|
|
typedef struct _COMMON_EXTENSION {
|
|
ULONG Signature;
|
|
BOOLEAN Deleted;
|
|
AGP_EXTENSION_TYPE Type;
|
|
PDEVICE_OBJECT AttachedDevice;
|
|
BUS_INTERFACE_STANDARD BusInterface;
|
|
} COMMON_EXTENSION, *PCOMMON_EXTENSION;
|
|
|
|
// Structures to maintain a list of "favored" memory ranges
|
|
// for AGP allocation.
|
|
|
|
typedef struct _AGP_MEMORY_RANGE
|
|
{
|
|
PHYSICAL_ADDRESS Lower;
|
|
PHYSICAL_ADDRESS Upper;
|
|
} AGP_MEMORY_RANGE, *PAGP_MEMORY_RANGE;
|
|
|
|
typedef struct _AGP_FAVORED_MEMORY
|
|
{
|
|
ULONG NumRanges;
|
|
PAGP_MEMORY_RANGE Ranges;
|
|
} AGP_FAVORED_MEMORY;
|
|
|
|
//
|
|
// WMI data types
|
|
//
|
|
#define AGP_WMI_STD_DATA_GUID \
|
|
{ 0x8c27fbed, 0x1c7b, 0x47e4, 0xa6, 0x49, 0x0e, 0x38, 0x9d, 0x3a, 0xda, 0x4f }
|
|
|
|
typedef struct _AGP_STD_DATA {
|
|
PHYSICAL_ADDRESS ApertureBase;
|
|
ULONG ApertureLength;
|
|
ULONG AgpStatus;
|
|
ULONG AgpCommand;
|
|
} AGP_STD_DATA, *PAGP_STD_DATA;
|
|
|
|
typedef struct _TARGET_EXTENSION {
|
|
COMMON_EXTENSION CommonExtension;
|
|
PFAST_MUTEX Lock;
|
|
struct _MASTER_EXTENSION *ChildDevice;
|
|
PCM_RESOURCE_LIST Resources;
|
|
PCM_RESOURCE_LIST ResourcesTranslated;
|
|
AGP_FAVORED_MEMORY FavoredMemory;
|
|
PHYSICAL_ADDRESS GartBase;
|
|
ULONG GartLengthInPages;
|
|
PDEVICE_OBJECT PDO;
|
|
PDEVICE_OBJECT Self;
|
|
WMILIB_CONTEXT WmiLibInfo;
|
|
|
|
//
|
|
// This must be last, d'oh!
|
|
//
|
|
ULONGLONG AgpContext;
|
|
} TARGET_EXTENSION, *PTARGET_EXTENSION;
|
|
|
|
typedef struct _MASTER_EXTENSION {
|
|
COMMON_EXTENSION CommonExtension;
|
|
PTARGET_EXTENSION Target;
|
|
ULONG Capabilities;
|
|
ULONG InterfaceCount; // tracks the number of interfaces handed out
|
|
ULONG ReservedPages; // tracks the number of pages reserved in the aperture
|
|
BOOLEAN StopPending; // TRUE if we have seen a QUERY_STOP
|
|
BOOLEAN RemovePending; // TRUE if we have seen a QUERY_REMOVE
|
|
ULONG DisableCount; // non-zero if we are in a state where we cannot service requests
|
|
} MASTER_EXTENSION, *PMASTER_EXTENSION;
|
|
|
|
typedef struct _GLOBALS {
|
|
|
|
//
|
|
// Path to the driver's Services Key in the registry
|
|
//
|
|
UNICODE_STRING RegistryPath;
|
|
|
|
//
|
|
// Cached target capability settings
|
|
//
|
|
ULONG AgpStatus;
|
|
ULONG AgpCommand;
|
|
|
|
} GLOBALS;
|
|
|
|
|
|
extern GLOBALS Globals;
|
|
|
|
//
|
|
// The MBAT - used to retrieve "favored" memory ranges from
|
|
// the AGP northbridge via an ACPI BANK method
|
|
//
|
|
|
|
#include <pshpack1.h>
|
|
|
|
typedef struct _MBAT
|
|
{
|
|
UCHAR TableVersion;
|
|
UCHAR AgpBusNumber;
|
|
UCHAR ValidEntryBitmap;
|
|
AGP_MEMORY_RANGE DecodeRange[ANYSIZE_ARRAY];
|
|
} MBAT, *PMBAT;
|
|
|
|
#include <poppack.h>
|
|
|
|
#define MBAT_VERSION 1
|
|
|
|
#define MAX_MBAT_SIZE sizeof(MBAT) + ((sizeof(UCHAR) * 8) - ANYSIZE_ARRAY) \
|
|
* sizeof(AGP_MEMORY_RANGE)
|
|
|
|
#define CM_BANK_METHOD (ULONG)('KNAB')
|
|
|
|
//
|
|
// The highest memory address supported by AGP
|
|
//
|
|
|
|
#define MAX_MEM(_num_) _num_ = 1; \
|
|
_num_ = _num_*1024*1024*1024*4 - 1
|
|
|
|
#define RELEASE_BUS_INTERFACE(_ext_) (_ext_)->CommonExtension.BusInterface.InterfaceDereference((_ext_)->CommonExtension.BusInterface.Context)
|
|
|
|
//
|
|
// Macros for getting from the chipset-specific context to our structures
|
|
//
|
|
#define GET_TARGET_EXTENSION(_target_,_agp_context_) (_target_) = (CONTAINING_RECORD((_agp_context_), \
|
|
TARGET_EXTENSION, \
|
|
AgpContext)); \
|
|
ASSERT_TARGET(_target_)
|
|
#define GET_MASTER_EXTENSION(_master_,_agp_context_) { \
|
|
PTARGET_EXTENSION _targetext_; \
|
|
GET_TARGET_EXTENSION(_targetext_, (_agp_context_)); \
|
|
(_master_) = _targetext_->ChildDevice; \
|
|
ASSERT_MASTER(_master_); \
|
|
}
|
|
#define GET_TARGET_PDO(_pdo_,_agp_context_) { \
|
|
PTARGET_EXTENSION _targetext_; \
|
|
GET_TARGET_EXTENSION(_targetext_,(_agp_context_)); \
|
|
(_pdo_) = _targetext_->CommonExtension.AttachedDevice; \
|
|
}
|
|
|
|
#define GET_MASTER_PDO(_pdo_,_agp_context_) { \
|
|
PMASTER_EXTENSION _masterext_; \
|
|
GET_MASTER_EXTENSION(_masterext_, (_agp_context_)); \
|
|
(_pdo_) = _masterext_->CommonExtension.AttachedDevice; \
|
|
}
|
|
|
|
#define GET_AGP_CONTEXT(_targetext_) ((PVOID)(&(_targetext_)->AgpContext))
|
|
#define GET_AGP_CONTEXT_FROM_MASTER(_masterext_) GET_AGP_CONTEXT((_masterext_)->Target)
|
|
|
|
//
|
|
// Some debugging macros
|
|
//
|
|
#define ASSERT_TARGET(_target_) ASSERT((_target_)->CommonExtension.Signature == TARGET_SIG); \
|
|
ASSERT((_target_)->ChildDevice->CommonExtension.Signature == MASTER_SIG)
|
|
#define ASSERT_MASTER(_master_) ASSERT((_master_)->CommonExtension.Signature == MASTER_SIG); \
|
|
ASSERT((_master_)->Target->CommonExtension.Signature == TARGET_SIG)
|
|
|
|
//
|
|
// Locking macros
|
|
//
|
|
#define LOCK_MUTEX(_fm_) ExAcquireFastMutex(_fm_); \
|
|
ASSERT((_fm_)->Count == 0)
|
|
|
|
#define UNLOCK_MUTEX(_fm_) ASSERT((_fm_)->Count == 0); \
|
|
ExReleaseFastMutex(_fm_)
|
|
|
|
#define LOCK_TARGET(_targetext_) ASSERT_TARGET(_targetext_); \
|
|
LOCK_MUTEX((_targetext_)->Lock)
|
|
|
|
#define LOCK_MASTER(_masterext_) ASSERT_MASTER(_masterext_); \
|
|
LOCK_MUTEX((_masterext_)->Target->Lock)
|
|
|
|
#define UNLOCK_TARGET(_targetext_) ASSERT_TARGET(_targetext_); \
|
|
UNLOCK_MUTEX((_targetext_)->Lock)
|
|
|
|
#define UNLOCK_MASTER(_masterext_) ASSERT_MASTER(_masterext_); \
|
|
UNLOCK_MUTEX((_masterext_)->Target->Lock)
|
|
|
|
//
|
|
// Private resource type definition
|
|
//
|
|
typedef enum {
|
|
AgpPrivateResource = '0PGA'
|
|
} AGP_PRIVATE_RESOURCE_TYPES;
|
|
|
|
//
|
|
// Define function prototypes
|
|
//
|
|
|
|
//
|
|
// Driver and device initialization - init.c
|
|
//
|
|
NTSTATUS
|
|
AgpAttachDeviceRelations(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
|
|
//
|
|
// IRP Dispatch routines - dispatch.c
|
|
//
|
|
NTSTATUS
|
|
AgpDispatchPnp(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN OUT PIRP Irp
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpDispatchPower(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN OUT PIRP Irp
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpDispatchDeviceControl(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN OUT PIRP Irp
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpDispatchWmi(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN OUT PIRP Irp
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpSetEventCompletion(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PKEVENT Event
|
|
);
|
|
|
|
//
|
|
// Config space handling routines - config.c
|
|
//
|
|
NTSTATUS
|
|
ApQueryBusInterface(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
OUT PBUS_INTERFACE_STANDARD BusInterface
|
|
);
|
|
|
|
//
|
|
// Resource handing routines - resource.c
|
|
//
|
|
NTSTATUS
|
|
AgpFilterResourceRequirements(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpQueryResources(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpStartTarget(
|
|
IN PIRP Irp,
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
VOID
|
|
AgpStopTarget(
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
//
|
|
// AGP Interface functions
|
|
//
|
|
NTSTATUS
|
|
AgpInterfaceSetRate(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN ULONG AgpRate
|
|
);
|
|
|
|
VOID
|
|
AgpInterfaceReference(
|
|
IN PMASTER_EXTENSION Extension
|
|
);
|
|
|
|
VOID
|
|
AgpInterfaceDereference(
|
|
IN PMASTER_EXTENSION Extension
|
|
);
|
|
|
|
|
|
NTSTATUS
|
|
AgpInterfaceReserveMemory(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN ULONG NumberOfPages,
|
|
IN MEMORY_CACHING_TYPE MemoryType,
|
|
OUT PVOID *MapHandle,
|
|
OUT OPTIONAL PHYSICAL_ADDRESS *PhysicalAddress
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpInterfaceReleaseMemory(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN PVOID MapHandle
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpInterfaceCommitMemory(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN PVOID MapHandle,
|
|
IN ULONG NumberOfPages,
|
|
IN ULONG OffsetInPages,
|
|
IN OUT PMDL Mdl OPTIONAL,
|
|
OUT PHYSICAL_ADDRESS *MemoryBase
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpInterfaceFreeMemory(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN PVOID MapHandle,
|
|
IN ULONG NumberOfPages,
|
|
IN ULONG OffsetInPages
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpInterfaceGetMappedPages(
|
|
IN PMASTER_EXTENSION Extension,
|
|
IN PVOID MapHandle,
|
|
IN ULONG NumberOfPages,
|
|
IN ULONG OffsetInPages,
|
|
OUT PMDL Mdl
|
|
);
|
|
|
|
//
|
|
// Misc utils.c
|
|
//
|
|
BOOLEAN
|
|
AgpOpenKey(
|
|
IN PWSTR KeyName,
|
|
IN HANDLE ParentHandle,
|
|
OUT PHANDLE Handle,
|
|
OUT PNTSTATUS Status
|
|
);
|
|
|
|
BOOLEAN
|
|
AgpStringToUSHORT(
|
|
IN PWCHAR String,
|
|
OUT PUSHORT Result
|
|
);
|
|
|
|
ULONGLONG
|
|
AgpGetDeviceFlags(
|
|
IN PAGP_HACK_TABLE_ENTRY AgpHackTable,
|
|
IN USHORT VendorID,
|
|
IN USHORT DeviceID,
|
|
IN USHORT SubVendorID,
|
|
IN USHORT SubSystemID,
|
|
IN UCHAR RevisionID
|
|
);
|
|
|
|
ULONG_PTR
|
|
AgpExecuteCriticalSystemRoutine(
|
|
IN ULONG_PTR Context
|
|
);
|
|
|
|
//
|
|
// wmi.c
|
|
//
|
|
NTSTATUS
|
|
AgpSystemControl(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpWmiRegistration(
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpWmiDeRegistration(
|
|
IN PTARGET_EXTENSION Extension
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpSetWmiDataItem(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN ULONG GuidIndex,
|
|
IN ULONG InstanceIndex,
|
|
IN ULONG DataItemId,
|
|
IN ULONG BufferSize,
|
|
IN PUCHAR Buffer
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpSetWmiDataBlock(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN ULONG GuidIndex,
|
|
IN ULONG InstanceIndex,
|
|
IN ULONG BufferSize,
|
|
IN PUCHAR Buffer
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpQueryWmiDataBlock(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN ULONG GuidIndex,
|
|
IN ULONG InstanceIndex,
|
|
IN ULONG InstanceCount,
|
|
IN OUT PULONG InstanceLengthArray,
|
|
IN ULONG OutBufferSize,
|
|
OUT PUCHAR Buffer
|
|
);
|
|
|
|
NTSTATUS
|
|
AgpQueryWmiRegInfo(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
OUT PULONG RegFlags,
|
|
OUT PUNICODE_STRING InstanceName,
|
|
OUT PUNICODE_STRING *RegistryPath,
|
|
OUT PUNICODE_STRING MofResourceName,
|
|
OUT PDEVICE_OBJECT *Pdo
|
|
);
|
|
|
|
//
|
|
// AGP Physical Memory allocator
|
|
//
|
|
PMDL
|
|
AgpLibAllocatePhysicalMemory(
|
|
IN PVOID AgpContext,
|
|
IN ULONG TotalBytes);
|
|
|
|
//
|
|
// Handy structures for mucking about in PCI config space
|
|
//
|
|
|
|
//
|
|
// The PCI_COMMON_CONFIG includes the 192 bytes of device specific
|
|
// data. The following structure is used to get only the first 64
|
|
// bytes which is all we care about most of the time anyway. We cast
|
|
// to PCI_COMMON_CONFIG to get at the actual fields.
|
|
//
|
|
|
|
typedef struct {
|
|
ULONG Reserved[PCI_COMMON_HDR_LENGTH/sizeof(ULONG)];
|
|
} PCI_COMMON_HEADER, *PPCI_COMMON_HEADER;
|
|
|
|
|