On Windows CE, stream drivers are opened just like files, and are opened using a unique three letter prefix (e.g. BKL,COM,TST,PWR).
The core stream interface entry points are:
XXX_Open (Device Manager)
XXX_Close (Device Manager)
XXX_Read (Device Manager)
XXX_Write (Device Manager)
1. Choose a unique three letter identifier for your driver. lets choose it as TST.
2. name the driver - here we will name it as TestDriver
3. Next, you will need to implement the required entry points.
lets see required functions for a complete stream driver:
//TestDriver.c by Umesh Jagga
#include
//This function initializes a device. It is called by Device Manager.
//This function is required by drivers loaded by ActivateDeviceEx, ActivateDevice, or
//RegisterDevice.
DWORD TST_Init(LPCTSTR pContext, LPCVOID lpvBusContext);
//This function de-initializes a device. It is called by Device Manager
BOOL TST_Deinit( DWORD hDeviceContext );
//This function opens a device for reading, writing, or both. An application indirectly invokes
//this function when it calls the CreateFile function to open special device file names.
DWORD TST_Open( DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode );
//This function closes a device context created by the hOpenContext parameter.
//An application calls the CloseHandle function to stop using a stream interface driver. The hFile
//parameter specifies the handle associated with the device context. In response to
//CloseHandle, the operating system invokes XXX_Close.
BOOL TST_Close( DWORD hOpenContext );
//This function sends a command to a device.
//An application uses the DeviceIoControl function to specify an operation to perform. The
//operating system, in turn, invokes the XXX_IOControl function
BOOL TST_IOControl( DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut );void USB_PowerUp( DWORD hDeviceContext );
void TST_PowerDown( DWORD hDeviceContext );
//This function reads data from the device identified by the open context
//an application calls the ReadFile function to read from the device, the operating system
//invokes this function.
DWORD TST_Read( DWORD hOpenContext, LPVOID pBuffer, DWORD Count );
//This function writes data to the device.
//After an application uses the WriteFile function to write to the device, the operating system,
//invokes this function.
DWORD TST_Write( DWORD hOpenContext, LPCVOID pBuffer, DWORD Count );
//This function moves the data pointer in the device.
//After an application calls the SetFilePointer function to move the data pointer in the device,
//the operating system invokes this function.
//If your device is capable of opening more than once, this function modifies only the data
//pointer for the instance specified by hOpenContext.
DWORD TST_Seek( DWORD hOpenContext, long Amount, WORD Type );
//If the function is used, it is called by the system when processes and threads are initialized
//and terminated, or on calls to the LoadLibrary and FreeLibrary functions.
//Direct entry point for a DLL.
//If you use the C Runtime in your application, it is responsible for performing any initialization
//of the C Runtime at process attach, and for deinitializing the C Runtime at process detach
// ----------------------------------------------------
BOOL APIENTRY DllMain( HANDLE hModule, DWORD ul_reason_for_call, LPVOID pReserved )
{
switch(ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
OutputDebugString(L"TestDriver - DLL_PROCESS_ATTACH\n");
break;
case DLL_PROCESS_DETACH:
OutputDebugString(L"TestDriver - DLL_PROCESS_DETACH\n");
break;
case DLL_THREAD_ATTACH:
OutputDebugString(L"TestDriver - DLL_THREAD_ATTACH\n");
break;
case DLL_THREAD_DETACH:
OutputDebugString(L"TestDriver - DLL_THREAD_DETACH\n");
break;
default:
break;
}
return TRUE;
}
// Driver Init...
DWORD TST_Init( LPCTSTR pContext, LPCVOID lpvBusContext)
{
OutputDebugString(L"TestDriver - TST_Init - Context: ");
OutputDebugString(pContext);
OutputDebugString(L"TestDriver - ~ TST_Init\n");
return 0x1234;
}
BOOL TST_Deinit( DWORD hDeviceContext )
{
OutputDebugString(L"TestDriver - TST_Deinit\n");
OutputDebugString(L"TestDriver - ~ TST_Deinit\n");
return TRUE;
}
// Driver Open
DWORD TST_Open( DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode )
{
OutputDebugString(L"TestDriver - TST_Open\n");
OutputDebugString(L"hDeviceContext - ");
OutputDebugString(L"TestDriver - ~ TST_Open\n");
return 0x5678;
}
BOOL TST_Close( DWORD hOpenContext )
{
OutputDebugString(L"TestDriver - TST_Close\n");
OutputDebugString(L"hOpenContext - ");
OutputDebugString(L"\n");
OutputDebugString(L"TestDriver - ~ TST_Close\n");
return TRUE;
}
BOOL TST_IOControl( DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut )
{
OutputDebugString(L"TestDriver - TST_IOControl\n");
OutputDebugString(L"hOpenContext - ");
OutputDebugString(L"\n");
switch (dwCode) {
case IOCTL_DRIVER_TST:
{
OutputDebugString(L"DRIVER TST IOCTL...\n");
}
break;
default:
OutputDebugString(L"Unknown IOCTL\n");
break;
}
OutputDebugString(L"TestDriver - ~ TST_IOControl\n");
return TRUE;
}
void TST_PowerUp( DWORD hDeviceContext )
{
OutputDebugString(L"TestDriver - TST_PowerUp\n");
OutputDebugString(L"hDeviceContext - ");
OutputDebugString(L"\n");
OutputDebugString(L"TestDriver - ~ TST_PowerUp\n");
}
void TST_PowerDown( DWORD hDeviceContext )
{
OutputDebugString(L"TestDriver - TST_PowerDown\n");
OutputDebugString(L"hDeviceContext - ");
OutputDebugString(L"\n");
OutputDebugString(L"TestDriver - ~ TST_PowerDown\n");
}
DWORD TST_Read( DWORD hOpenContext, LPVOID pBuffer, DWORD Count )
{
OutputDebugString(L"TestDriver - TST_Read\n");
OutputDebugString(L"hOpenContext - ");
OutputDebugString(L"\n");
OutputDebugString(L"TestDriver - ~ TST_Read\n");
return 1;
}
DWORD TST_Write( DWORD hOpenContext, LPCVOID pBuffer, DWORD Count )
{
OutputDebugString(L"TestDriver - TST_Write\n");
OutputDebugString(L"hOpenContext - ");
OutputDebugString(L"\n");
OutputDebugString(L"TestDriver - ~ TST_Write\n");
return 1;
}
DWORD TST_Seek( DWORD hOpenContext, long Amount, WORD Type )
{
OutputDebugString(L"TestDriver - TST_Seek\n");
OutputDebugString(L"hOpenContext - ");
OutputDebugString(L"TestDriver - ~ TST_Seek\n");
return 0;
}
//end of TestDriver.c by umesh Jagga
now lets write a testdriver.def file. This is the definition file that is passed
to the linker. This file contains the exported function names.
Note: the DLL filename and def filename should be same.
LIBRARY TestDriver
EXPORTS
TST_Init
TST_Deinit
TST_Open
TST_Close
TST_IOControl
TST_PowerUp
TST_PowerDown
TST_Read
TST_Write
TST_Seek
now lets write a sources file:
TARGETNAME=TestDriver
TARGETTYPE=DYNLINK
TARGETLIBS=$(_COMMONSDKROOT)\lib\$(_CPUINDPATH)\coredll.lib
SOURCES=TestDriver.c
now we need to put the registry and bib file entries.
you will need to create the registry entries for this driver so that thedriver can be loaded by Device.exe.
lets put this in platform.bib and platform.reg files.
addin platform.reg file:
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\TestDriver]
"Dll" = "TestDriver.Dll"
"Prefix" = "TST"
"Order" = dword:0
"FriendlyName" = "Umesh TestDriver"
"Ioctl" = dword:0
add the below in platform.bib file
TestDriver.dll $(_FLATRELEASEDIR)\TestDriver.dll NK SH
Just for reference:
Device Driver Loading Process:
Kernel---loads---> Device.exe (I/O Resource Manager)-------Loads------>
-------->REGENUM.DLL(registryenumerator,its reentrant)
-----------------loads-----------
REGENUM.DLL(ISA) PCIBUS.DLL
Now we know the concept of device driver lets proceed to step2...
making a custom device driver for your device for customized functionality........
No comments:
Post a Comment