天天看点

WDF基本对象和句柄定义

1          对象和句柄定义

1.1         struct FX_OBJECT_INFO数组FxObjectsInfo:

Fxobjectinfokm.cpp中定义了一个结构体FX_OBJECT_INFO数组FxObjectsInfo[],WDF自定义的对象,如WDFDRIVER、WDFDEVICE等都在该数组中登记对应的信息。

typedef struct _FX_OBJECT_INFO {
    const CHAR* Name;  //WDF对象名
    const CHAR* HandleName; 
    USHORT Size; //对象大小
    USHORT ObjectType; //对象类型,enum类型值
} FX_OBJECT_INFO, *PFX_OBJECT_INFO;      

         WDF使用分别使用宏FX_INTERNAL_OBJECT_INFO_ENTRY和宏FX_INTERNAL_OBJECT_INFO_ENTRY初始化结构体FX_OBJECT_INFO数组FxObjectsInfo。

FX_OBJECT_INFO FxObjectsInfo[] = {
   FX_INTERNAL_OBJECT_INFO_ENTRY(FxObject, FX_TYPE_OBJECT),
FX_EXTERNAL_OBJECT_INFO_ENTRY(FxDriver,FX_TYPE_DRIVER, WDFDRIVER),
};      

用于初始化FxObjectsInfo数组的宏,其定义如下:

#define FX_INTERNAL_OBJECT_INFO_ENTRY(_obj,_type)  \
{ #_obj, NULL, sizeof(_obj), _type, }
 
#define FX_EXTERNAL_OBJECT_INFO_ENTRY(_obj,_type, _handletype) \
    {#_obj,                                                   \
     (const CHAR*) (_handletype) #_handletype,                 \
     sizeof(_obj),                                             \
     _type,                                                   \
}      

我们知道,##在宏定义中有连接两个操作符的作用,#在宏定义中起到将操作符字符化的作用,例如:

#define example(instr) printf("theinput string is:\t%s\n",#instr)
#define example1(instr) #instr      

当使用该宏定义时:

example(abc);  在编译时将会展开成:printf("theinput string is:\t%s\n","abc");
string str=example1(abc);  将会展成:stringstr="abc";      

参照上面的例子,将上面宏运算FX_INTERNAL_OBJECT_INFO_ENTRY(FxObject,FX_TYPE_OBJECT)展开,得到如下结果:

FX_OBJECT_INFOFxObjectsInfo[] = {
{ “FxObject”,NULL,sizeof(FxObject),FX_TYPE_OBJECT },
…
};
         最终,FxObjectsInfo[0].Name指向”FxObject”, FxObjectsInfo[0].HandleName为空,
FxObjectsInfo[0].Size=4 FxObjectsInfo[0].ObjectType=枚举值FX_TYPE_OBJECT。
         在继续展开FX_EXTERNAL_OBJECT_INFO_ENTRY(FxDriver,FX_TYPE_DRIVER, WDFDRIVER)宏之前,先看下WDFDRIVER的定义。
 
#ifdef STRICT
typedef void *HANDLE;
#if 0 && (_MSC_VER > 1000)
#define DECLARE_HANDLE(name) structname##__; typedef struct name##__ *name
#else
#define DECLARE_HANDLE(name) structname##__{int unused;}; typedef struct name##__ *name
#endif
 
Windows首先声明HANDLE为void*指针类型,然后,通过宏DECLARE_HANDLE定义了WDFDRIVER__结构体:
Struct WDFDRIVER__
{
         Intunused;
};      

紧接着,声明了WDFDRIVER为WDFDRIVER__*指针类型,对于指针而言,HANDLE和WDFDRIVER是等价的(可以相互转换)。这有点类似于MFC,它定义了各种句柄类型,如:

DECLARE_HANDLE(HMODULE); 

DECLARE_HANDLE(HINSTANCE); 

DECLARE_HANDLE(HGLOBAL); 

DECLARE_HANDLE(HDC); 

DECLARE_HANDLE(HWND); 

DECLARE_HANDLE(HMENU); 

在实际编码过程中,可能以以下形式

HWND hWnd

上面这段代码等价于:

struct HWND* hWnd = FindWindow(…);

如果做一个简单的转换,可以将struct HWND*以不损失精度为代价,直接转换为void*(即HANDLE)。所以,就本质而言,DECLARE_HANDLE宏仅仅为void*类型新创建一个类型名。

1.2         WDF对象和WDM对象的对应关系

虽然用WDF框架开发驱动,但隐约能看到背后WDM的影子。WDF对象亦是如此,就像我们会不自觉的把WDFDRIVER对象和DRIVER_OBJECT对象相互关联起来。证明两者之间确实存在某种关联性,我们还是需要从: FX_INTERNAL_OBJECT_INFO_ENTRY/FX_EXTERNAL_OBJECT_INFO_ENTRY两个宏入手。

         在这两个宏中都有一段计算结构体大小的代码:

#define FX_EXTERNAL_OBJECT_INFO_ENTRY(_obj,_type, _handletype) \
    {#_obj,                                                   \
     (const CHAR*) (_handletype) #_handletype,                 \
     sizeof(_obj),                                             \
     _type,                                                   \
}
 
#define FX_INTERNAL_OBJECT_INFO_ENTRY(_obj,_type)  \
{ #_obj, NULL, sizeof(_obj),_type, }      

他们都计算第一个宏参数_obj的大小。对于其他宏参数,不外乎是enum类型和HANDLE类型,因此,我们可以确定,第一个宏参数一定是WDF定义的各种对象类型,即FxObject、FxDriver才是真正的WDF对象。

         FxObject是WDF自定义的对象类型,没有对应的WDM对象,所以先来寻找FxDriver的定义以及对应的WDM对象。

class FxDriver : public FxNonPagedObject,public IFxHasCallbacks
{
friend class FxDevice;
friend class FxPackage;
friend class FxWmiIrpHandler;
 
private:
 
    MxDriverObjectm_DriverObject;
UNICODE_STRING m_RegistryPath;
….
};      

WDF定义了一个FxDriver类,类成员变量    MxDriverObject m_DriverObject和UNICODE_STRINGm_RegistryPath的名字很容易让我们联想到WDM中传给驱动入口函数DriverEntry的两个参数类型DRIVER_OBJECT和UNICODE_STRING。再跟进MxDriverObject的定义,它依然是一个WDF类:

class MxDriverObject
{
private:
MdDriverObjectm_DriverObject;
…
};