KEMBAR78
Chapter 6 - Developing Device Drivers | PDF | Device Driver | Application Programming Interface
0% found this document useful (0 votes)
472 views70 pages

Chapter 6 - Developing Device Drivers

MCTS I Exam 70-571 Windows Embedded CE 6. Preparation Kit Self-Paced Training Up to Date with Content Not for Resale. Device drivers are components that enable the operating system and user applications to interact with peripheral hardware. This chapter discusses best practices for writing device drivers with proper code structures, developing a secure and well-designed configuration user interface.

Uploaded by

Sunil Pai
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
472 views70 pages

Chapter 6 - Developing Device Drivers

MCTS I Exam 70-571 Windows Embedded CE 6. Preparation Kit Self-Paced Training Up to Date with Content Not for Resale. Device drivers are components that enable the operating system and user applications to interact with peripheral hardware. This chapter discusses best practices for writing device drivers with proper code structures, developing a secure and well-designed configuration user interface.

Uploaded by

Sunil Pai
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
You are on page 1/ 70

MCTS i

Exam 70-571

Windows Embedded CE 6.0

Preparation Kit Self-Paced Training

Up to Date
with

R2
Content

Not for Resale.


2 first top-level entry

Contents at a Glance
1 Customizing the Operating System Design
2 Building and Deploying the Run-Time Image
3 Performing System Programming
4 Debugging and Testing the System
5 Customizing a Board Support Package
6 Developing Device Drivers
Chapter 6
Developing Device Drivers
Device drivers are components that enable the operating system (OS) and user
applications to interact with peripheral hardware that is integrated or attached to a
target device, such as the Peripheral Component Interconnect (PCI) bus, keyboard,
mouse, serial ports, display, network adapter, and storage devices. Rather than
accessing the hardware directly, the operating system loads the corresponding device
drivers, and then uses the functions and input/output (I/O) services that these
drivers provide to carry out actions on the device. In this way, the Microsoft®
Windows® Embedded CE 6.0 R2 architecture remains flexible, extensible, and
independent of the underlying hardware details. The device drivers contain the
hardware-specific code, and you can implement custom drivers in addition to the
standard drivers that ship with CE to support additional peripherals. In fact, device
drivers are the largest part of the Board Support Package (BSP) for an OS design.
However, it is also important to keep in mind that poorly implemented drivers can
ruin an otherwise reliable system. When developing device drivers, it is imperative to
follow strict coding practices and test the components thoroughly in various system
configurations. This chapter discusses best practices for writing device drivers with
proper code structures, developing a secure and well-designed configuration user
interface, ensuring reliability even after prolonged use, and supporting multiple
power management features.

Exam objectives in this chapter:


■ Loading and using device drivers on Windows Embedded CE
■ Managing interrupts on the system
■ Understanding memory access and memory handling
■ Enhancing driver portability and system integration

241
242 Chapter 6 Developing Device Drivers

Before You Begin


■ To complete the lessons in this chapter, you must have the following:
■ At least some basic knowledge about Windows Embedded CE software
development, including fundamental concepts related to driver development,
such as I/O control (IOCTL) and Direct Memory Access (DMA).
■ An understanding of interrupt handling and how to respond to interrupts in a
device driver.
■ Familiarity with memory management in C and C++, as well as and knowledge
of how to avoid memory leaks.
■ A development computer with Microsoft Visual Studio® 2005 Service Pack 1 and
Platform Builder for Windows Embedded CE 6.0 R2 installed.
Lesson 1: Understanding Device Driver Basics 243

Lesson 1: Understanding Device Driver Basics


On Windows Embedded CE, a device driver is a dynamic-link library (DLL) that
provides a layer of abstraction between the underlying hardware, OS, and
applications running on the target device. The driver exposes a set of known
functions and provides the logic to initialize and communicate with the hardware.
Software developers can then call the driver’s functions in their applications to
interact with the hardware. If a device driver adheres to a well-known application
programming interface (API) such as Device Driver Interface (DDI), you can load the
driver as part of the operating system, such as a display driver or a driver for a storage
device. Without having to know details about the physical hardware, applications can
then call standard Windows API functions, such as ReadFile or WriteFile, to use the
peripheral device. You can support different types of peripherals by adding different
drivers to the OS design without having to reprogram your applications.

After this lesson, you will be able to:


■ Differentiate between native and stream drivers.
■ Describe the advantages and disadvantages of monolithic and layered driver archi-
tectures.
Estimated lesson time: 15 minutes.

Native and Stream Drivers


A Windows Embedded CE device driver is a DLL that exposes the standard DllMain
function as the entry point, so that a parent process can load the driver by calling
LoadLibrary or LoadDriver. Drivers loaded by means of LoadLibrary can be paged
out, but the operating system does not page out drivers loaded through LoadDriver.
While all drivers expose the DllMain entry point, Windows Embedded CE supports
two different types of drivers: native drivers and stream drivers. Native CE drivers
typically support input and output peripherals, such as display drivers, keyboard
drivers, and touchscreen drivers. The Graphics, Windowing, and Events Subsystem
(GWES) loads and manages these drivers directly. Native drivers implement specific
functions according to their purpose, which GWES can determine by calling the
GetProcAddress API. GetProcAddress returns a pointer to the desired function or
NULL if the driver does not support the function.
244 Chapter 6 Developing Device Drivers

Stream drivers, on the other hand, expose a well-known set of functions that enable
Device Manager to load and manage these drivers. For Device Manager to interact
with a stream driver, the driver must implement the Init, Deinit, Open, Close, Read,
Write, Seek, and IOControl functions. In many stream drivers, the Read, Write, and
Seek functions provide access to the stream content, yet not all peripherals are stream
devices. If the device has special requirements beyond Read, Write, and Seek, you can
use the IOControl function to implement the required functionality. The IOControl
function is a universal function that can accommodate any special needs of a stream
device driver. For example, you can extend a driver’s functionality by passing a
custom IOCTL command code and input and output buffers.

NOTE Native driver interface


Native drivers must implement different types of interfaces depending on the nature of the
driver. For complete information about the supported driver types, see the section “Windows
Embedded CE Drivers” in the Windows Embedded CE 6.0 Documentation, available on the
Microsoft MSDN® Web site at http://msdn2.microsoft.com/en-us/library/aa930800.aspx.

Monolithic vs. Layered Driver Architecture


Native and stream drivers only differ in terms of the APIs they expose. You can load
both types of drivers during system startup or on demand, and both types can use a
monolithic or layered design, as illustrated in Figure 6–1.
Lesson 1: Understanding Device Driver Basics 245

OS Applications

Graphics, Windowing and Events Subsystem


Device Manager
(GWES)

Device Driver Device Driver


Stream Interface Stream Interface
Interface (DDI) Interface (DDI)

Layered Driver Layered Stream Driver

Model Device Monolithic Model Device


Monolithic
Driver (MDD) Stream Driver (MDD)
Driver
Driver
(Single DLL)
(Single DLL)
Platform Device Platform Device
Driver (MDD) Driver (MDD)

Hardware

Figure 6-1 Monolithic and layered driver architectures

Monolithic Drivers
A monolithic driver relies on a single DLL to implement both the interface to the
operating system and applications, and the logic to the hardware. The development
costs for monolithic drivers are generally higher than for layered drivers, yet despite
this disadvantage, monolithic drivers also have advantages. The primary advantage is
a performance gain by avoiding additional function calls between separate layers in
the driver architecture. Memory requirements are also slightly lower in comparison to
layered drivers. A monolithic driver might also be the right choice for uncommon,
custom hardware. If no layered driver code exists that you could reuse, and if this is a
unique driver project, you might find it advantageous to implement a driver in a
monolithic architecture. This is especially true if reusable monolithic source code is
available.

Layered Drivers
In order to facilitate code reuse and lower development overhead and costs, Windows
Embedded CE supports a layered driver architecture based on model device driver
(MDD) and platform device driver (PDD). MDD and PDD provide an additional
abstraction layer for driver updates and to accelerate the development of device
246 Chapter 6 Developing Device Drivers

drivers for new hardware. The MDD layer contains the interface to the operating
system and the applications. On the hardware side, MDD interfaces with the PDD
layer. The PDD layer implements the actual functions to communicate with the
hardware.
When porting a layered driver to new hardware, you generally do not need to modify
the code in the MDD layer. It is also less complicated to duplicate an existing layered
driver and add or remove functionality than to create a new driver from scratch. Many
of the drivers included in Windows Embedded CE take advantage of the layered
driver architecture.

NOTE MDD/PDD architecture and driver updates


The MDD/PDD architecture can help driver developers save time during the development of
driver updates, such as providing quick fix engineering (QFE) fixes to customers. Restricting
modifications to the PDD layer increases development efficiencies.

Lesson Summary
Windows Embedded CE supports native and stream drivers. Native drivers are the
best choice for any devices that are not stream devices. For example, a display device
driver must be able to process data in arbitrary patterns and is therefore a good
candidate for a native driver. Other devices, such as storage hardware and serial ports,
are good candidates for stream drivers because these devices handle data primarily in
the form of ordered streams of bytes as if they were files. Both native and stream
drivers can use either the monolithic or layered driver design. In general, it is
advantageous to use the layered architecture based on MDD and PDD because it
facilitates code reuse and the development of driver updates. Monolithic drivers
might be the right choice if you want to avoid the additional function calls between
MDD and PDD for performance reasons.
Lesson 2: Implementing a Stream Interface Driver 247

Lesson 2: Implementing a Stream Interface Driver


On Windows Embedded CE, a stream driver is a device driver that implements the
stream interface API. Regardless of hardware specifics, all CE stream drivers expose
the stream interface functions to the operating system so the Device Manager of
Windows Embedded CE can load and manage these drivers. As the name implies,
stream drivers are suitable for I/O devices that act as sources or sinks of streams of
data, such as integrated hardware components and peripheral devices. It is also
possible for a stream driver to access other drivers to provide applications with more
convenient access to the underlying hardware. In any case, you need to know the
stream interface functions and how to implement them if you want to develop a fully
functional and reliable stream driver.

After this lesson, you will be able to:


■ Understand the purpose of Device Manager.
■ Identify stream driver requirements.
■ Implement and use a stream driver.
Estimated lesson time: 40 minutes.

Device Manager
The Windows Embedded CE Device Manager is the OS component that manages the
stream device drivers on the system. The OAL (Oal.exe) loads the kernel (Kernel.dll),
and the kernel loads Device Manager during the boot process. Specifically, the kernel
loads the Device Manager shell (Device.dll), which in turn loads the actual core
Device Manager code (Devmgr.dll), which again is in charge of loading, unloading,
and interfacing with stream drivers, as illustrated in Figure 6–2.
Stream drivers can be loaded as part of the operating system at boot time or on
demand when the corresponding hardware is connected in a Plug and Play fashion.
User applications can then use the stream drivers through file system APIs, such as
ReadFile and WriteFile, or by means of DeviceIoControl calls. Stream drivers that
Device Manager exposes through the file system appear to applications as regular file
resources with special file names. The DeviceIoControl function, on the other hand,
enables applications to perform direct input and output operations. However,
applications interact with the stream drivers indirectly through Device Manager in
both cases.
248 Chapter 6 Developing Device Drivers

User Applications

Obtains a Handle Via CreateFile Application Calls DeviceIoControl

File System
Device Manager

Calls XXX_stream Functions


CE Kernel

Device Driver
OEM Adaptation Layer (OAL) (Stream Interface)

Hardware

Figure 6-2 The Device Manager in Windows Embedded CE 6.0

Driver Naming Conventions


For an application to use a stream driver through the file system, the stream driver
must be presented as a file resource so that the application can specify the device file
in a CreateFile call to get a handle to the device. Having obtained the handle, the
application can then use ReadFile or WriteFile to perform I/O operations, which
Device Manager translates into corresponding calls to stream interface functions to
perform the desired read and write actions. For Windows Embedded CE 6.0 to
recognize stream device resources and redirect file I/O operations to the appropriate
stream drive, stream drivers must follow a special naming convention that
distinguishes these resources from ordinary files.
Windows Embedded CE 6.0 supports the following naming conventions for stream
drivers:
■ Legacy names The classical naming convention for stream drivers consists of
three upper case letters, a digit, and a colon. The format is XXX[0–9]:, where XXX
stands for the three-letter driver name and [0–9] is the index of the driver as
specified in the driver’s registry settings (see Lesson 3, “Configuring and
Loading a Driver”). Because the driver index has only one digit, legacy names
Lesson 2: Implementing a Stream Interface Driver 249

only support up to ten instances of a stream driver. The first instance


corresponds to index 1, the ninth instance uses index 9, and the tenth instance
refers to index 0. For example, CreateFile(L"COM1:"…) accesses the stream
driver for the first serial port by using the legacy name COM1:.

NOTE Legacy name limitation


The legacy naming convention does not support more than ten instances per stream driver.

■ Device names To access a stream driver with an index of ten or higher, you can
use the device name instead of the legacy name. The device name conforms to
the format \$device\XXX[index], where \$device\ is a namespace that
indicates that this is a device name, XXX stands for the three-letter driver name
and [index] is the index of the driver. The index can have multiple digits. For
example, CreateFile(L"\$device\COM11"…) would access the stream driver for
the eleventh serial port. Stream drivers with a legacy name can also be accessed,
such as CreateFile(L"\$device\COM1"…).

NOTE Legacy and device name access


Although legacy names and device names differ in format and supported range of driver
instances, CreateFile returns the same handle with access to the same stream driver in
both cases.

■ Bus name Stream drivers for devices on a bus, such as Personal Computer
Memory Card International Association (PCMCIA) or Universal Serial Bus
(USB), correspond to bus names that the relevant bus driver passes to Device
Manager when enumerating the drivers available on the bus. Bus names relate to
the underlying bus structure. The general format is \$bus\BUSNAME_[bus
number]_[device number]_[function number], where \$bus\ is a namespace
that indicates that this is a bus name, BUSNAME refers to the name or type of the
bus, and [bus number], [device number], and [function number] are bus-specific
identifiers. For example, CreateFile(L"\$ bus\PCMCIA_0_0_0"…) accesses
device 0, function 0, on PCMCIA bus 0, which might correspond to a serial port.

NOTE Bus name access


Bus names are primarily used to obtain handles for unloading and reloading bus drivers
and for power management, but not for data read and write operations.
250 Chapter 6 Developing Device Drivers

Stream Interface API


For Device Manager to load and manage a stream driver successfully, the stream
driver must export a common interface, generally referred to as the stream interface.
The stream interface consists of 12 functions to initialize and open the device, read
and write data, power up or down the device, and close and de-initialize the device, as
summarized in Table 6–1.
Table 6-1 Stream interface functions
Function Name Description
XXX_Init Device Manager calls this function to load a driver during the
boot process or in answer to calls to ActivateDeviceEx in
order to initialize the hardware and any memory structures
used by the device.
XXX_PreDeinit Device Manager calls this function before calling
XXX_Deinit to give the driver a chance to wake sleeping
threads and invalidate open handles in order to accelerate
the de-initialization process. Applications do not call this
function.
XXX_Deinit Device Manager calls this function to de-initialize and de-
allocate memory structures and other resources in response
to a DeActivateDevice call after deactivating and unloading a
driver.
XXX_Open Device Manager calls this function when an application
requests access to the device by calling CreateFile for
reading, writing, or both.
XXX_PreClose Device Manager calls this function to give the driver a chance
to invalidate handles and wake sleeping threads in order to
accelerate the unloading process. Applications do not call
this function.
XXX_Close Device Manager calls this function when an application
closes an open instance of the driver, such as by calling the
CloseHandle function. The stream driver must de-allocate all
memory and resources allocated during the previous
XXX_Open call.
Lesson 2: Implementing a Stream Interface Driver 251

Table 6-1 Stream interface functions (Continued)


Function Name Description
XXX_Read Device Manager calls this function in response to a ReadFile
call to read data from the device and pass it to the caller.
Even if the device does not provide any data to read, the
stream device driver must implement this function to be
compatible with Device Manager.
XXX_Write Device Manager calls this function in response to a WriteFile
call to pass data from the caller to the device. Similar to
XXX_Read, XXX_Write is mandatory but can be empty if the
underlying device does not support write operations, such
as an input-only communications port.
XXX_Seek Device Manager calls this function in response to a
SetFilePointer call to move the data pointer to a particular
point in the data stream for reading or writing. Similar to
XXX_Read and XXX_Write, this function can be empty but
must be exported to be compatible with Device Manager.
XXX_IOControl Device Manager calls this function in response to a
DeviceIoControl call to perform device-specific control
tasks. For example, power management features rely on
DeviceIoControl calls if the driver advertises a power
management interface, such as to query power capabilities
and manage the power status through the IOCTLs
IOCTL_POWER_CAPABILITIES, IOCTL_POWER_QUERY,
and IOCTL_POWER_SET. Applications can also use the
DeviceIoControl function to read and write data in device
drivers that do not use XXX_Write and XXX_Read. This is a
common approach in many stream drivers.
252 Chapter 6 Developing Device Drivers

Table 6-1 Stream interface functions (Continued)


Function Name Description
XXX_PowerUp Device Manager calls this function when the operating
system comes out of a low power mode. Applications do not
call this function. This function executes in kernel mode, can
therefore not call external APIs, and must not be paged out
because the operating system is running in single-threaded,
non-paged mode. Microsoft recommends that drivers
implement power management based on Power Manager
and power management IOCTLs for suspend and resume
functionality in a driver.
XXX_PowerDown Device Manager calls this function when the operating
system transitions into suspend mode. Similar to XXX_
PowerUp, this function executes in kernel mode, can
therefore not call external APIs, and must not be paged out.
Applications do not call this function. Microsoft
recommends that drivers implement power management
based on Power Manager and power management IOCTLs.

NOTE XXX prefix


In the function names, the prefix XXX is a placeholder that refers to the three-letter driver name.
You need to replace this prefix with the actual name in the driver code, such as COM_Init for a
driver called COM, or SPI_Init for a Serial Peripheral Interface (SPI) driver.

Device Driver Context


Device Manager supports context management based on device context and open
context parameters, which Device Manager passes as DWORD values to the stream
driver with each function call. Context management is vital if a driver must allocate
and deallocate instance-specific resources, such as blocks of memory. It is important
to keep in mind that device drivers are DLLs, which implies that global variables and
other memory structures defined or allocated in the driver are shared by all driver
instances. Deallocating the wrong resources in response to an XXX_Close or
XXX_Deinit call can lead to memory leaks, application failures, and general system
instabilities.
Lesson 2: Implementing a Stream Interface Driver 253

Stream drivers can manage context information per device driver instance based on
the following two levels:
1. Device context The driver initializes this context in the XXX_Init function.
This context is therefore also called the Init Context. Its primary purpose is to
help the driver manage resources related to hardware access. Device Manager
passes this context information to the XXX_Init, XXX_Open, XXX_PowerUp,
XXX_PowerDown, XXX_PreDeinit and XXX_Deinit functions.
2. Open context The driver initializes this second context in the XXX_Open
function. Each time an application calls CreateFile for a stream driver, the stream
driver creates a new open context. The open context then enables the stream
driver to associate data pointers and other resources with each opened driver
instance. Device Manager passes the device context to the stream driver in the
XXX_Open function so that the driver can store a reference to the device context
in the open context. In this way, the driver can retain access to the device context
information in subsequent calls, such as XXX_Read, XXX_Write, XXX_Seek,
XXX_IOControl, XXX_PreClose and XXX_Close. Device Manager only passes
the open context to these functions in the form of a DWORD parameter.
The following code listing illustrates how to initialize a device context for a sample
driver with the driver name SMP (such as SMP1:):
DWORD SMP_Init(LPCTSTR pContext, LPCVOID lpvBusContext)
{
T_DRIVERINIT_STRUCTURE *pDeviceContext = (T_DRIVERINIT_STRUCTURE *)
LocalAlloc(LMEM_ZEROINIT|LMEM_FIXED, sizeof(T_DRIVERINIT_STRUCTURE));

if (pDeviceContext == NULL)
{
DEBUGMSG(ZONE_ERROR,(L" SMP: ERROR: Cannot allocate memory "

+ "for sample driver’s device context.\r\n"));

// Return 0 if the driver failed to initialize.

return 0;
}

// Perform system intialization...

pDeviceContext->dwOpenCount = 0;

DEBUGMSG(ZONE_INIT,(L"SMP: Sample driver initialized.\r\n"));

return (DWORD)pDeviceContext;
}
254 Chapter 6 Developing Device Drivers

Building a Device Driver


To create a device driver, you can add a subproject for a Windows Embedded CE DLL
to your OS design, but the most common way to do it is to add the device driver’s
source files inside the Drivers folder of the Board Support Package (BSP). For detailed
information about configuring Windows Embedded CE subprojects, see Chapter 1,
“Customizing the Operating System Design.”
A good starting point for a device driver is A Simple Windows Embedded CE DLL
Subproject, which you can select on the Auto-Generated Subproject Files page in the
Windows Embedded CE Subproject Wizard. It automatically creates a source code
file with a definition for the DllMain entry point for the DLL, various parameter files,
such as empty module-definition (.def) and registry (.reg) files, and preconfigures the
Sources file to build the target DLL. For more detailed information about parameter
files and the Sources file, see Chapter 2, “Building and Deploying a Run-Time Image.”

Implementing Stream Functions


Having created the DLL subproject, you can open the source code file in Visual Studio
and add the required functions to implement the stream interface and required driver
functionality. The following code listing shows the definition of the stream interface
functions.
// SampleDriver.cpp : Defines the entry point for the DLL application.
//

#include "stdafx.h"

BOOL APIENTRY DllMain(HANDLE hModule,


DWORD ul_reason_for_call,
LPVOID lpReserved)
{
return TRUE;
}

DWORD SMP_Init(LPCTSTR pContext, LPCVOID lpvBusContext)


{
// Implement device context initialization code here.
return 0x1;
}

BOOL SMP_Deinit(DWORD hDeviceContext)


{
// Implement code to close the device context here.
return TRUE;
}
Lesson 2: Implementing a Stream Interface Driver 255

DWORD SMP_Open(DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode)


{
// Implement open context initialization code here.
return 0x2;
}

BOOL SMP_Close(DWORD hOpenContext)


{
// Implement code to close the open context here.
return TRUE;
}

DWORD SMP_Write(DWORD hOpenContext, LPCVOID pBuffer, DWORD Count)


{
// Implement the code to write to the stream device here.
return Count;
}

DWORD SMP_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count)


{
// Implement the code to read from the stream device here.
return Count;
}

BOOL SMP_IOControl(DWORD hOpenContext, DWORD dwCode,

PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut,

DWORD dwLenOut, PDWORD pdwActualOut)


{
// Implement code to handle advanced driver actions here.
return TRUE;
}

void SMP_PowerUp(DWORD hDeviceContext)


{
// Implement power management code here or use IO Control.
return;
}

void SMP_PowerDown(DWORD hDeviceContext)


{
// Implement power management code here or use IO Control.
return;
}

Exporting Stream Functions


Making the stream functions in the driver DLL accessible to external applications
requires the linker to export the functions during the build process. C++ provides
several options to accomplish this, yet for driver DLLs compatible with Device
256 Chapter 6 Developing Device Drivers

Manager, you must export the functions by defining them in the .def file of the DLL
subproject. The linker uses the .def file to determine which functions to export and
how to do so. For a standard stream driver, you must export the stream interface
functions using the prefix that you specify in the driver’s source code and registry
settings. Figure 6–3 shows a sample .def file for the stream interface skeleton listed in
the previous section.

Figure 6-3 A sample .def file for a stream driver

Sources File
Prior to building the newly created stream driver, you should also check the Sources
file in the root folder of the DLL subproject to ensure that it includes all necessary files
in the build process. As mentioned in Chapter 2, the Sources file configures the
compiler and linker to build the desired binary files. Table 6–2 summarizes the most
important Sources file directives for device drivers.
Lesson 2: Implementing a Stream Interface Driver 257

Table 6-2 Important Sources file directives for device drivers


Directive Description
WINCEOEM=1 Causes additional header files and import
libraries from the %_WINCEROOT%\Public tree
to be included to enable the driver to make
platform-dependent function calls, such as
KernelIoControl, InterruptInitialize, and
InterruptDone.
TARGETTYPE=DYNLINK Instructs the Build tool to create a DLL.
DEFFILE=<Driver Def File References the module-definition file that defines
Name>.def the exported DLL functions.
DLLENTRY=<DLL Main Entry Specifies the function that is called when
Point> processes and threads attach and detach to and
from the driver DLL (Process Attach, Process
Detach, Thread Attach, and Thread Detach).

Opening and Closing a Stream Driver by Using the File API


To access a stream driver, an application can use the CreateFile function and specify
the desired device name. The following example illustrates how to open a driver
called SMP1: for reading and writing. It is important to note, however, that Device
Manager must already have loaded the driver, such as during the boot process. Lesson
3 later in this chapter provides detailed information about configuring and loading
device drivers.
// Open the driver, which results in a call to the SMP_Open function
hSampleDriver = CreateFile(L"SMP1:",
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);

if (hSampleDriver == INVALID_HANDLE_VALUE )
{
ERRORMSG(1,(TEXT("Unable to open the driver.\r\n"));
return FALSE;
}
258 Chapter 6 Developing Device Drivers

// Access the driver and perform read,


// write, and seek operations as required.

// Close the driver


CloseHandle(hSampleDriver);

Dynamically Loading a Driver


As mentioned earlier in this lesson, an application can also communicate with a
stream device driver after calling the ActivateDevice or ActivateDeviceEx function.
ActivateDeviceEx offers more flexibility than ActivateDevice, yet both functions cause
Device Manager to load the stream driver and call the driver’s XXX_Init function. In
fact, ActivateDevice calls ActivateDeviceEx. Note, however, that ActivateDeviceEx
does not provide access to an already loaded driver. The primary purpose of the
ActivateDeviceEx function is to read a driver-specific registry key specified in the
function call to determine the DLL name, device prefix, index, and other values, add
the relevant values to the active device list, and then load the device driver into the
Device Manager process space. The function call returns a handle that the application
can later use to unload the driver in a call to the DeactivateDevice function.
ActivateDeviceEx replaces the older RegisterDevice function as a method to load a
driver on demand, as illustrated in the following code sample:
// Ask Device Manager to load the driver for which the definition
// is located at HKLM\Drivers\ Sample in the registry.
hActiveDriver = ActivateDeviceEx(L"\\Drivers\\Sample", NULL, 0, NULL);
if (hActiveDriver == INVALID_HANDLE_VALUE)
{
ERRORMSG(1, (L"Unable to load driver"));
return -1;
}

// Once the driver is lodaded, applications can open the driver


hDriver = CreateFile (L"SMP1:",
GENERIC_READ| GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);

if (hDriver == INVALID_HANDLE_VALUE)
{
ERRORMSG(1, (TEXT("Unable to open Sample (SMP) driver")));
return 0;
}
Lesson 2: Implementing a Stream Interface Driver 259

//Insert code that uses the driver here

// Close the driver when access is no longer needed


if (hDriver != INVALID_HANDLE_VALUE)
{
bRet = CloseHandle(hDriver);
if (bRet == FALSE)
{
ERRORMSG(1, (TEXT("Unable to close SMP driver")));
}
}

// Manually unload the driver from the system using Device Manager
if (hActiveDriver != INVALID_HANDLE_VALUE)
{
bRet = DeactivateDevice(hActiveDriver);
if (bRet == FALSE)
{
ERRORMSG(1, (TEXT("Unable to unload SMP driver ")));
}
}

NOTE Automatic vs. dynamic loading of drivers


Calling ActivateDeviceEx to load a driver has the same result as loading the driver automatically
during the boot process through parameters defined in the HKEY_LOCAL_MACHINE\Driv-
ers\BuiltIn key. The BuiltIn registry key is covered in more detail in Lesson 3 later in this chapter.

Lesson Summary
Stream drivers are Windows Embedded CE drivers that implement the stream
interface API. The stream interface enables Device Manager to load and manage these
drivers, and applications can use standard file system functions to access these
drivers and perform I/O operations. To present a stream driver as a file resource
accessible through a CreateFile call, the name of the stream driver must follow a
special naming convention that distinguishes the device resource from ordinary files.
Legacy names (such as COM1:) have a limitation of ten instances per driver because
they include only a single-digit instance identification. If you must support more than
ten driver instances on a target device, use a device name instead (such as
\$device\COM1).
Because Device Manager can load a single driver multiple times to satisfy the requests
from different processes and threads, stream drivers must implement context
management. Windows Embedded CE knows two context levels for device drivers,
device context and open context, which the operating system passes in each
260 Chapter 6 Developing Device Drivers

appropriate function call to the driver so that the driver can associate internal
resources and allocated memory regions with each caller.
The stream interface consists of 12 functions: XXX_Init, XXX_Open, XXX_Read,
XXX_Write, XXX_Seek, XXX_IOControl, XXX_PowerUp, XXX_PowerDown,
XXX_PreClose, XXX_Close, XXX_PreDeinit, and XXX_Deinit. Not all functions are
mandatory (such as XXX_PreClose and XXX_PreDeinit), yet any functions that the
stream device driver implements must be exposed from the driver DLL to Device
Manager. To export these functions, you must define them in the .def file of the DLL
subproject. You should also adjust the DLL subproject’s Sources file to ensure that the
driver DLL can make platform-dependent function calls.
Lesson 3: Configuring and Loading a Driver 261

Lesson 3: Configuring and Loading a Driver


In general, you have two options to load a stream driver under Windows Embedded
CE 6.0. You can instruct Device Manager to load the driver automatically during the
boot sequence by configuring driver settings in the HKEY_LOCAL_MACHINE
\Drivers\BuiltIn registry key, or you can load the driver dynamically through a direct
call to ActivateDeviceEx. Either way, Device Manager can load the device driver with
the same registry flags and settings. The key difference is that you receive a handle to
the driver when using ActivateDeviceEx, which you can use later in a call to
DeactivateDevice. Especially during the development stage, it might be advantageous
to load a driver dynamically through ActivateDeviceEx so that you can unload the
driver, install an updated version, and then reload the driver without having to restart
the operating system. You can also use DeactivateDevice to unload drivers loaded
automatically based on entries under the BuiltIn registry key, but you cannot reload
them without calling ActivateDeviceEx directly.

After this lesson, you will be able to:


■ Identify the mandatory registry settings for a device driver.
■ Access the registry settings from within a driver.
■ Load a driver at startup or on demand in an application.
■ Load a driver in user space or kernel space.
Estimated lesson time: 25minutes.

Device Driver Load Procedure


Whether you load a device driver statically or dynamically, the ActivateDeviceEx
function is always involved. A dedicated driver named the Bus Enumerator
( B u s E n u m ) c a l l s A c t i v a t e D e v i c e E x fo r e ve r y d r i ve r re g i s t e re d u n d e r
HKEY_LOCAL_MACHINE\Drivers\BuiltIn just as you can call ActivateDeviceEx
directly, passing in an alternate registry path for the driver settings in the lpszDevKey
parameter.
Device Manager uses the following procedure to load device drivers at boot time:
1. Device Manager reads the HKEY_LOCAL_MACHINE\Drivers\RootKey entry
to determine the location of the device driver entries in the registry. The default
value of the RootKey entry is Drivers\BuiltIn.
262 Chapter 6 Developing Device Drivers

2. Device Manager reads the Dll registry value at the location specified in the
RootKey location (HKEY_LOCAL_MACHINE\Drivers\BuiltIn) to determine
the enumerator DLL to load. By default, t his is t he bus enumerator
(BusEnum.dll). The bus enumerator is a stream driver that exports the Init and
Deinit functions.
3. The bus enumerator runs at startup to scan the RootKey registry location for
subkeys that refer to additional buses and devices. It can be run again later with
a different RootKey to load more drivers. The bus enumerator examines the
Order value in each subkey to determine the load order.
4. Starting with the lowest Order values, the bus enumerator iterates through the
subkeys and calls ActivateDeviceEx passing in the current driver’s registry path
(that is, HKEY_LOCAL_MACHINE\Drivers\BuiltIn\<DriverName>).
5. ActivateDeviceEx loads the driver DLL registered in the DLL value located in the
driver’s subkey, and t hen creates a subkey for the driver under t he
HKEY_LOCAL_MACHINE\Drivers\Active registry key to keep track of all
currently loaded drivers.
Figure 6–4 shows a typical registration under the HKEY_LOCAL_MACHINE
\Drivers\BuiltIn registry key for an audio device driver.

Figure 6-4 An audio device driver registration


Lesson 3: Configuring and Loading a Driver 263

Registry Settings to Load Device Drivers


If you use ActivateDeviceEx to load your driver dynamically, you are not required to
place the driver’s registry settings in a subkey under HKEY_LOCAL_MACHINE
\Drivers\BuiltIn. You can use an arbitrary path, such as HKEY_LOCAL_MACHINE
\SampleDriver. However, the registry values for the driver are the same in both cases.
Table 6–3 lists general registry entries that you can specify for a device driver in the
driver’s registry subkey (see Figure 6–4 for sample values).
Table 6-3 General registry entries for device drivers
Registry Entry Type Description
Prefix REG_SZ A string value that contains the driver’s
three-letter name. This is the value that
replaces XXX in the stream interface
functions. Applications also use this prefix
to open a context of the driver through
CreateFile.
Dll REG_SZ This is the name of the DLL that Device
Manager loads to load the driver.
Note that this is the only mandatory registry
entry for a driver.
Index REG_DWORD This is the number appended to the driver
prefix to create the driver’s file name. For
example, if this value is 1, applications can
access this driver through a call to
CreateFile(L"XXX1:"…) or
CreateFile(L"\$device\XXX1"…).
Note that this value is optional. If you do not
define it, Device Manager assigns the next
available index value to the driver.
264 Chapter 6 Developing Device Drivers

Table 6-3 General registry entries for device drivers (Continued)


Registry Entry Type Description
Order REG_DWORD This is the order in which Device Manger
loads the driver. If this value is not specified,
the driver will be loaded last, at the same
time as other drivers with no order
specified. Drivers with the same Order value
start concurrently.
You should only configure this value to
enforce a sequential load order. For
example, a Global Positioning System (GPS)
driver might require a Universal
Asynchronous Receiver/Transmitter
(UART) driver to get access to the GPS data
through a serial port. In this case, it is
important to assign the UART driver a lower
Order value than the GPS driver so that the
UART driver starts first. This will enable the
GPS driver to access the UART driver during
its initialization.
IClass REG_MULTI_SZ This value can specify predefined device
interface globally unique identifiers
(GUIDs). To advertise an interface to Device
Manager, such as to support the Plug and
Play notification system and power
management capabilities, add the corres-
ponding interface GUIDs to the IClass value
or call AdvertiseInterface in the driver.
Lesson 3: Configuring and Loading a Driver 265

Table 6-3 General registry entries for device drivers (Continued)


Registry Entry Type Description
Flags REG_DWORD This value can contain the following flags:
■ DEVFLAGS_UNLOAD (0x0000 0001)
The driver unloads after a call to
XXX_Init.
■ DEVFLAGS_NOLOAD (0x0000 0004)
The driver cannot be loaded.
■ DEVFLAGS_NAKEDENTRIES
(0x0000 0008) The entry points of
the driver are Init, Open, IOControl,
and so forth, without any prefixes.
■ DEVFLAGS_BOOTPHASE_1
(0x0000 1000) The driver is loaded
during system phase 1 for systems with
multiple boot phases. This prevents the
driver from being loaded more than
one time during the boot process.
■ DEVFLAGS_IRQ_EXCLUSIVE
(0x0000 0100) The bus driver loads
this driver only when it has exclusive
access to the interrupt request (IRQ)
specified by the IRQ value.
■ DEVFLAGS_LOAD_AS_USERPROC
(0x0000 0010) Loads the driver in
user mode.
266 Chapter 6 Developing Device Drivers

Table 6-3 General registry entries for device drivers (Continued)


Registry Entry Type Description
UserProcGroup REG_DWORD Associates a driver marked with the
DEVFLAGS_LOAD_AS_USERPROC
(0x0000 0010) flag to load in user mode
with a user-mode driver host process group.
User-mode drivers that belong to the same
group are loaded by Device Manager in the
same host process instance. If this registry
entry does not exist, Device Manager loads
the user-mode driver into a new host
process instance.

NOTE Flags
For details about the Flags registry value, see the section “ActivateDeviceEx” in the Windows
Embedded CE 6.0 Documentation, available on the Microsoft MSDN Web site at http://
msdn2.microsoft.com/en-us/library/aa929596.aspx.

Registry Keys Related to Loaded Device Drivers


Apart from configurable registry entries in driver-specific subkeys, Device Manager
also maintains dynamic registry information in subkeys for loaded drivers under the
HKEY_LOC AL_MACHINE\Drivers\Active key. The subkeys correspond to
numerical values that the operating system assigns dynamically and increments for
each driver until the system is restarted. The number does not signify a particular
driver. For example, if you unload and reload a device driver, the operating system
assigns the next number to the driver and does not reuse the previous subkey.
Because you cannot ensure a reliable association between the subkey number and a
p ar ti cul a r de vi ce dr i ve r, yo u s ho u ld n o t edi t t h e d ri ve r e ntr ie s i n t he
HKEY_LOCAL_MACHINE\Drivers\Active key manually. However, you can create,
read, and write driver-specific registry keys at load time in a driver’s XXX_Init
function because Device Manager passes the path to the current Drivers\Active
subkey to the stream driver as the first parameter. The driver can open this registry
key using OpenDeviceKey.
Table 6–4 lists typical entries that the subkeys under Drivers\Active can contain.
Lesson 3: Configuring and Loading a Driver 267

Registry entries for device drivers under the


Table 6-4
HKEY_LOCAL_MACHINE\Drivers\Active key
Registry Type Description
Entry
Hnd REG_DWORD The handle value for the loaded device driver. You
can obtain this DWORD value from the registry and
pass it in a call to DeactivateDevice in order to
unload the driver.
BusDriver REG_SZ The name of the driver’s bus.
BusName REG_SZ The name of the device’s bus.
DevID A unique device identifier from Device Manager.
FullName REG_SZ The name of the device if used with the $device
namespace.
Name REG_SZ The driver’s legacy device file name including
index, if a prefix is specified (not present for drivers
that do not specify a prefix).
Order REG_DWORD The same order value as in the driver’s registry key.
Key REG_SZ The registry path to the driver’s registry key.
PnpId REG_SZ The Plug and Play identifier string for PCMCIA
drivers.
Sckt REG_DWORD For PCMCIA drivers, describes the current socket
and function of the PC card.

NOTE Checking the Active key


By call ing t he RequestDeviceNoti ficati ons function wit h a devic e inter face GUID of
DEVCLASS_STREAM_GUID, an application can receive messages from Device Manager to iden-
tify loaded stream drivers programmatically. RequestDeviceNotifications supersedes the Enum-
Devices function.
268 Chapter 6 Developing Device Drivers

Kernel-Mode and User-Mode Drivers


Drivers can either run in the kernel memory space or in user memory space. In kernel
mode, drivers have full access to the hardware and kernel memory, although function
calls are generally limited to kernel APIs. Windows Embedded CE 6.0 runs drivers in
kernel mode, by default. On the other hand, drivers in user mode do not have direct
access to kernel memory. There are some performance penalties when running in
user mode, yet the advantage is that a driver failure in user mode only affects the
current process, whereas the failure of a kernel-mode driver can impair the entire
operating system. The system can generally recover more gracefully from the failure of
a user-mode driver.

NOTE Kernel driver restrictions


Kernel drivers cannot display a user interface directly in CE 6.0 R2. To use any user interface ele-
ments, developers must create a companion DLL that will be loaded into user-mode, then call
into this DLL with CeCallUserProc. For more information on CeCallUserProc, see the MSDN Web
page at http://msdn2.microsoft.com/en-us/library/aa915093.aspx.

User-Mode Drivers and the Reflector Service


In order to communicate with the underlying hardware and perform useful tasks,
user-mode drivers must be able to access system memory and privileged APIs
unavailable to standard user-mode processes. To facilitate this, Windows Embedded
CE 6.0 features a Reflector service, which runs in kernel mode, performs buffer
marshaling, and calls privileged memory management APIs on behalf of the user-
mode drivers. The Reflector service is transparent so that user-mode drivers can work
almost the same way as kernel-mode drivers without modifications. An exception to
this rule is a driver that uses kernel APIs that are not available in user mode. It is not
possible to run these types of kernel-mode drivers in user mode.
When an application calls ActivateDeviceEx, Device Manager loads the driver either
directly in kernel space or passes the request to the Reflector service, which in turn
starts a user mode driver host process (Udevice.exe) through a CreateProcess call.
The Flags registry entry in the driver’s registry key determines whether a driver
should run in user mode (DEVFLAGS_LOAD_AS_USERPROC flag). Having started
the required instance of Udevice.exe and user-mode driver, the Reflector service
forwards the XXX_Init call from Device Manager to the user-mode driver and returns
the return code from the user-mode driver back to Device Manager, as indicated in
Figure 6–5. The same proxy principle also applies to all other stream functions.
Lesson 3: Configuring and Loading a Driver 269

User Space

Udevice.exe Udevice.exe

User-Mode Driver User-Mode Driver


DLL DLL

Kernel Space
Device Manager

Reflector Service Kernel-Mode


Driver DLL

Figure 6-5 User-mode drivers, kernel-mode drivers, and the Reflector service

User-Mode Drivers Registry Settings


On Windows Embedded CE 6.0, you can run multiple user-mode drivers in a single
host process or have multiple host processes enabled on the system. Drivers grouped
in a single Udevice.exe instance share the same process space, which is particularly
useful for drivers that depend on each other. However, drivers in the same process
space can affect each other’s stability. For example, if a user-mode driver causes the
host process to fail, all drivers in that host process fail. The system continues to
function except for the affected drivers and applications accessing these drivers, yet it
is possible to recover from this situation by reloading the drivers, if the applications
support it. If you isolate a critical driver in a separate user mode driver host process,
you can increase the overall system stability. By using the registry entries listed in
Table 6–5, you can define individual host process groups.
Table 6-5 Registry entries for user-mode driver host processes
Registry Entry Type Description
HKEY_LOCAL_MACHINE REG_KEY Defines a three-digit group ID
\ Drivers\ProcGroup_### (###) for a user-mode driver host
process, such as ProcGroup_003,
which you can then specify in the
UserProcGroup entry in a driver’s
registry key, such as
UserProcGroup =3.
270 Chapter 6 Developing Device Drivers

Table 6-5 Registry entries for user-mode driver host processes (Continued)
Registry Entry Type Description
ProcName REG_SZ The process that the Reflector
service starts to host the user-mode
driver, such as ProcName=
Udevice.exe.
ProcVolPrefix REG_SZ Specifies the file system volume
that the Reflector service mounts
for the user-mode driver host
process, such as ProcVolPrefix =
$udevice. The specified
ProcVolPrefix replaces the $device
volume in driver device names.

Having defined the desired host process groups, you can associate each user-mode
driver with a particular group by adding the UserProcGroup registry entry to the
device driver’s registry subkey (see Table 6–3 earlier in this lesson). By default, the
UserProcGroup registry entry does not exist, which corresponds to a configuration in
which Device Manager loads every user-mode driver into a separate host process
instance.

Binary Image Builder Configuration


As explained in Chapter 2, “Building and Deploying a Run-Time Image,” the Windows
Embedded CE build process relies on binary image builder (.bib) files to generate the
content of the run-time image and to define the final memory layout of the device.
Among other things, you can specify a combination of flags for a driver’s module
definition. Issues can arise if .bib file settings and registry entries do not match for a
device driver. For example, if you specify the K flag for a device driver module in a .bib
file and also set the DEVFLAGS_LOAD_AS_USERPROC flag in the driver’s registry
subkey to load the driver into the user-mode driver host process, the driver fails to
load because the K flag instructs Romimage.exe to load the module in kernel space
above the memory address 0x80000000. To load a driver in user mode, be sure to
load the module into user space below 0x80000000, such as into the NK memory
region defined in the Config.bib file for the BSP.
Lesson 3: Configuring and Loading a Driver 271

The following .bib file entry demonstrates how to load a user-mode driver into the NK
memory region:
driver.dll $(_FLATRELEASEDIR)\driver.dll NK SHQ

The S and H flags indicate that Driver.dll is both a system file and a hidden file,
located in the flat release directory. The Q flag specifies that the system can load this
module concurrently into both kernel and user space. It adds two copies of the DLL
to the run-time image, one with and one without the K flag, and doubles in this way
ROM and RAM space requirements for the driver. Use the Q flag sparingly.
Extending the above example, the Q flag is equivalent to the following:
driver.dll $(_FLATRELEASEDIR)\driver.dll NK SH
driver.dll $(_FLATRELEASEDIR)\driver.dll NK SHK

Lesson Summary
Windows Embedded CE can load drivers into kernel space or user space. Drivers
running in kernel space have access to system APIs and kernel memory and can affect
the stability of the system if failures occur. However, properly implemented kernel-
mode drivers exhibit better performance than user-mode drivers, due to reduced
context switching between kernel and user mode. On the other hand, the advantage
of user-mode drivers is that failures primarily affect the current user-mode process.
User-mode drivers are also less privileged, which can be an important aspect in
respect to non-trusted drivers from third-party vendors.
To integrate a driver running in user mode with Device Manager running in kernel
mode, Device Manager uses a Reflector service that loads the driver in a user-mode
driver host process and forwards the stream function calls and return values between
the driver and Device Manager. In this way, applications can continue to use familiar
file system APIs to access the driver, and the driver does not need code changes
regarding the stream interface API to remain compatible with Device Manager. By
default, user-mode drivers run in separate host processes, but you can also configure
host process groups and associate drivers with these groups by adding a
corresponding UserProcGroup registry entry to a driver’s registry subkey. Driver
subkeys can reside in any registry location, yet if you want to load the drivers at boot
time automatically, you must place the subkeys into Device Manager’s RootKey,
which by default is HKEY_LOCAL_MACHINE\Drivers\BuiltIn. Drivers that have
their subkeys in different locations can be loaded on demand by calling the
ActivateDeviceEx function.
272 Chapter 6 Developing Device Drivers

Lesson 4: Implementing an Interrupt Mechanism


in a Device Driver
Interrupts are notifications generated either in hardware or software to inform the
CPU that an event has occurred that requires immediate attention, such as timer
events or keyboard events. In response to an interrupt, the CPU stops executing the
current thread, jumps to a trap handler in the kernel to respond to the event, and then
resumes executing the original thread after the interrupt is handled. In this way,
integrated and peripheral hardware components, such as system clock, serial ports,
network adapters, keyboards, mouse, touchscreen, and other devices, can get the
attention of the CPU and have the kernel exception handler run appropriate code in
interrupt service routines (ISRs) within the kernel or in associated device drivers. To
implement interrupt processing in a device driver efficiently, you must have a detailed
understanding of Windows Embedded CE 6.0 interrupt handling mechanisms,
including the registration of ISRs in the kernel and the execution of interrupt service
threads (ISTs) within the Device Manager process.

After this lesson, you will be able to:


■ Implement an interrupt handler in the OEM adaptation layer (OAL).
■ Register and handle interrupts in a device driver interrupt service thread (IST).
Estimated lesson time: 40 minutes.

Interrupt Handling Architecture


Windows Embedded CE 6.0 is a portable operating system that supports different
CPU types with varying interrupt schemes by implementing a flexible interrupt
handling architecture. Most importantly, the interrupt handling architecture takes
advantage of interrupt-synchronization capabilities in the OAL and thread-
synchronization capabilities of Windows Embedded CE to split the interrupt
processing into ISRs and ISTs, as illustrated in Figure 6–6.
Windows Embedded CE 6.0 interrupt handling is based on the following concepts:
1. During the boot process, the kernel calls the OEMInit function in the OAL to
register all available ISRs built into the kernel with their corresponding
hardware interrupts based on their interrupt request (IRQ) values. IRQ values
are numbers that identify the source of the interrupt in the processor interrupt
controller registers.
Lesson 4: Implementing an Interrupt Mechanism in a Device Driver 273

2. Device drivers can dynamically install ISRs implemented in ISR DLLs by calling
the LoadIntChainHandler function. LoadIntChainHandler loads the ISR DLL
into kernel memory space and registers the specified ISR routine with the
specified IRQ value in the kernel’s interrupt dispatch table.
3. An interrupt occurs to notify the CPU that an event requires suspending the
current thread of execution and transferring control to a different routine.
4. In response to the interrupt, the CPU stops executing the current thread and
jumps to the kernel exception handler as the primary target of all interrupts.
5. The exception handler masks off all interrupts of an equal or lower priority and
then calls the appropriate ISR registered to handle the current interrupt. Most
hardware platforms use interrupt masks and interrupt priorities to implement
hardware–based interrupt synchronization mechanisms.
6. The ISR performs any necessary tasks, such as masking the current interrupt so
that the current hardware device cannot trigger further interrupts, which would
interfere with the current processing, and then returns a SYSINTR value to the
exception handler. The SYSINTR value is a logical interrupt identifier.
7. The exception handler passes the SYSINTR value to the kernel’s interrupt
support handler, which determines the event for the SYSINTR value, and, if
found, signals that event for any waiting ISTs for the interrupt.
8. The interrupt support handler unmasks all interrupts, with the exception of the
interrupt currently in processing. Keeping the current interrupt masked off
explicitly prevents the current hardware device from triggering another
interrupt while the IST runs.
9. The IST runs in response to the signaled event to perform and finish the
interrupt handling without blocking other devices on the system.
10. The IST calls the InterruptDone function to inform the kernel’s interrupt
support handler that the IST has finished its processing and is ready for another
interrupt event.
11. The interrupt support handler calls the OEMInterruptDone function in the OAL
to complete the interrupt handling process and reenable the interrupt.
274 Chapter 6 Developing Device Drivers

Interrupt IRQ SYSINTR


ISR IST
Occurs

IST calls InterruptDone to reenable the IRQ.


Figure 6-6 IRQs, ISRs, SYSINTRs, and ISTs

Interrupt Service Routines


In general, ISRs are small blocks of code that run in response to a hardware interrupt.
Because the kernel exception handler masks off all interrupts of equal or lesser
priority while this ISR runs, it is important to complete the ISR and return a SYSINTR
value as quickly as possible so that the kernel can re-enable (unmask) all IRQs with
minimal delay (except the currently processed interrupt). System performance can
suffer significantly if too much time is spent in ISRs, leading to missed interrupts or
overrun buffers on some devices. Another important aspect is that the ISR runs in
kernel mode and does not have access to higher-level operating system APIs. For
these reasons, ISRs usually perform no more than the most basic tasks, such as
quickly copying data from hardware registers to memory buffers. On Windows
Embedded CE, time-consuming interrupt processing is usually performed in an IST.
The primary task of the ISR is to determine the interrupt source, mask off or clear the
interrupt at the device, and then return a SYSINTR value for the interrupt to notify the
kernel about an IST to run. In the simplest case, the ISR returns SYSINTR_NOP to
indicate that no further processing is necessary. Accordingly, the kernel does not
signal an event for an IST to handle the interrupt. On the other hand, if the device
driver uses an IST to handle the interrupt, the ISR passes the logical interrupt
identifier to the kernel, the kernel determines and signals the interrupt event, and the
IST typically resumes from a WaitForSingleObject call and executes the interrupt
processing instructions in a loop. The latency between the ISR and the IST depends
on the priority of the thread and other threads running in the system, as explained in
Chapter 3, “Performing System Programming.” Typically, ISTs run with a high thread
priority.

Interrupt Service Threads


An IST is a regular thread that performs additional processing in response to an
interrupt, after the ISR has completed. The IST function typically includes a loop and
a WaitForSingleObject call to block the thread infinitely until the kernel signals the
specified IST event, as illustrated in the following code snippet. However, before you
Lesson 4: Implementing an Interrupt Mechanism in a Device Driver 275

can use the IST event, you must call InterruptInitialize with the SYSINTR value and an
event handle as parameters so that the CE kernel can signal this event whenever an
ISR returns the SYSINTR value. Chapter 3 provides detailed information about multi-
threaded programming and thread synchronization based on events and other kernel
objects.
CeSetThreadPriority(GetCurrentThread(), 200);

// Loop until told to stop


while(!pIst->stop)
{
// Wait for the IST event.
WaitForSingleObject(pIst->hevIrq, INFINITE)

// Handle the interrupt.


InterruptDone(pIst->sysIntr);
}

When the IST has completed processing an IRQ, it should call InterruptDone to
inform the system that the interrupt was processed, that the IST is ready to handle the
next IRQ, and that the interrupt can be reenabled by means of an OEMInterruptDone
call. Table 6–6 lists the OAL functions that the system uses to interact with the
interrupt controller to manage interrupts.
Table 6-6 OAL functions for interrupt management
Function Description
OEMInterruptEnable This function is called by the kernel in response to
InterruptInitialize and enables the specified
interrupt in the interrupt controller.
OEMInterruptDone This function is called by the kernel in response to
InterruptDone and should unmask the interrupt
and acknowledge the interrupt in the interrupt
controller.
OEMInterruptDisable This function disables the interrupt in the interrupt
controller and is called in response to the
InterruptDisable function.
OEMInterruptHandler For ARM processors only, this function identifies
the interrupt SYSINTR that occurs by looking at the
status of the interrupt controller.
276 Chapter 6 Developing Device Drivers

Table 6-6 OAL functions for interrupt management (Continued)


Function Description
HookInterrupt For processors other than ARM, this function
registers a callback function for a specified interrupt
ID. This function must be called in the OEMInit
function to register mandatory interrupts.
OEMInterruptHandlerFIQ For ARM processors, used to handle interrupts for
the Fast Interrupt (FIQ) line.

CAUTION WaitForMultipleObjects restriction


Do not use the WaitForMultipleObjects function to wait for an interrupt event. If you must wait
for multiple interrupt events, you should create an IST for each interrupt.

Interrupt Identifiers (IRQ and SYSINTR)


Each hardware interrupt line corresponds to an IRQ value in the interrupt controller
registers. Each IRQ value can be associated with only one ISR, but an ISR can map to
multiple IRQs. The kernel does not need to maintain the IRQs. It just determines and
signals events associated with the SYSINTR values returned from the ISR in response
to the IRQ. The ability to return varying SYSINTR values from an ISR provides the
basis to support multiple devices that use the same shared interrupt.

NOTE OEMInterruptHandler and HookInterrupt


Target devices that only support a single IRQ, such as ARM–based systems, use the OEMInter-
ruptHandler function as the ISR to identify the embedded peripheral that triggered the interrupt.
Original equipment manufacturers (OEMs) must implement this function as part of the OAL. On
platforms that support multiple IRQs, such as Intel x86–based systems, you can associate the
IRQs with individual ISRs by calling HookInterrupt.

Static Interrupt Mappings


For the ISR to determine a correct SYSINTR return value there must be a mapping
between the IRQ and the SYSINTR, which can be hardcoded into the OAL. The
Bsp_cfg.h file for the Device Emulator BSP demonstrates how to define a SYSINTR
value in the OAL for a target device relative to the SYSINTR_FIRMWARE value. If you
want to define additional identifiers in your OAL for a custom target device, keep in
Lesson 4: Implementing an Interrupt Mechanism in a Device Driver 277

mind that the kernel reserves all values below SYSINTR_FIRMWARE for future use
and the maximum value should be less than SYSINTR_MAXIMUM.
To add a mapping of static SYSINTR values to IRQs on a target device, you can call the
OALIntrStaticTranslate function during system initialization. For example, the Device
Emulator BSP calls OALIntrStaticTranslate in the BSPIntrInit function to register a
custom SYSINTR value for the built-in Open Host Controller Interface (OHCI) in the
kernel’s interrupt mapping arrays (g_oalSysIntr2Irq and g_oalIrq2SysIntr). However,
static SYSINTR values and mappings are not a common way to associate IRQs with
SYSINTRs because it is difficult and requires OAL code changes to implement custom
interrupt handling. Static SYSINTR values are typically used for core hardware
components of a target device where there is no explicit device driver and the ISR
resides in the OAL.

Dynamic Interrupt Mappings


The good news is that you do not need to hardcode SYSINTR values into the OAL if
you call KernelIoControl in your device drivers with an IO control code of
IOCTL_HAL_REQUEST_SYSINTR to register IRQ/SYSINTR mappings. The call
eventually ends in the OALIntrRequestSysIntr function, which dynamically allocates
a new SYSINTR for the given IRQ, and then registers the IRQ and SYSINTR mappings
in the kernel’s interrupt mapping arrays. Locating a free SYSINTR value up to
SYSINTR_MAXUMUM is more flexible than static SYSINTR assignments because
this mechanism does not require any modifications to the OAL when you add new
drivers to the BSP.
When calling KernelIoControl with IOCTL_HAL_REQUEST_SYSINTR, you
establish a 1:1 relationship between IRQ and SYSINTR. If the IRQ-SYSINTR mapping
table already has an entry for the specified IRQ, OALIntrRequestSysIntr will not
create a second entry. To remove an entry from the interrupt mapping tables, such as
when unloading a driver, call Ker nelIoControl with an IO control code of
IOCTL_HAL_REQUEST_SYSINTR. IOCTL_HAL_RELEASE_SYSINTR dissociates
the IRQ from the SYSINTR value.
The following code sample illustrates the use of IOCTL_HAL_REQUEST_SYSINTR
and IOCTL_HAL_RELEASE_SYSINTR. It takes a custom value (dwLogintr) and
passes this value to the OAL to be translated into a SYSINTR value, and then
associates this SYSINTR with an IST event.
278 Chapter 6 Developing Device Drivers

DWORD dwLogintr = IRQ_VALUE;


DWORD dwSysintr = 0;
HANDLE hEvent = NULL;
BOOL bResult = TRUE;

// Create event to associate with the interrupt


m_hEvent = CreateEvent(NULL,FALSE,FALSE,NULL);
if (m_hDetectionEvent == NULL)
{
return ERROR_VALUE;
}

// Ask the kernel (OAL) to associate an SYSINTR value to an IRQ


bResult = KernelIoControl(IOCTL_HAL_REQUEST_SYSINTR,
&dwLogintr, sizeof(dwLogintr),
&dwSysintr, sizeof(dwSysintr),
0);
if (bResult == FALSE)
{
return ERROR_VALUE;
}

// Initialize interrupt and associate the SYSINTR value with the event.
bResult = InterruptInitialize(dwSysintr, hEvent,0,0);

if (bResult == FALSE)
{
return ERROR_VALUE;
}

// Interrupt management loop


while(!m_bTerminateDetectionThread)
{
// Wait for the event associated to the interrupt
WaitForSingleObject(hEvent,INFINITE);

// Add actual IST processing here

// Acknowledge the interrupt


InterruptDone(m_dwSysintr);
}

// Deinitialize interrupts will mask the interrupt


bResult = InterruptDisable(dwSysintr);

// Unregister SYSINTR
bResult = KernelIoControl(IOCTL_HAL_RELEASE_SYSINTR,
&dwSysintr, sizeof(dwSysintr),
NULL,0,
0);

// Close the event object


CloseHandle(hEvent);
Lesson 4: Implementing an Interrupt Mechanism in a Device Driver 279

Shared Interrupt Mappings


The 1:1 relationship between IRQ and SYSINTR implies that you cannot register
multiple ISRs for an IRQ directly to implement interrupt sharing, but you can map
multiple ISRs indirectly. The interrupt mapping tables only map an IRQ to one static
ISR, yet within this ISR, you can call the NKCallIntChain function to iterate through
the chain of ISRs, registered dynamically through LoadIntChainHandler.
NKCallIntChain goes through the ISRs registered for the shared interrupt and returns
the first SYSINTR value that is not equal to SYSINTR_CHAIN. Having determined the
appropriate SYSINTR for the current interrupt source, the static ISR can pass this
logical interrupt identifier to the kernel to signal the corresponding IST event. The
LoadIntChainHandler function and installable ISRs are covered in more detail later in
this lesson.

Communication between an ISR and an IST


Because ISR and IST run at different times and in different contexts, you must take
extra care of physical and virtual memory mappings if an ISR must pass data to an IST.
For example, an ISR might copy individual bytes from a peripheral device into an
input buffer, returning SYSINTR_NOP until the buffer is full. The ISR returns the
actual SYSINTR value only when the input buffer is ready for the IST. The kernel
signals the corresponding IST event and the IST runs to copy the data into a process
buffer.
One way to accomplish this data transfer is to reserve a physical memory section in a
.bib file. Config.bib contains several examples for the serial and debug drivers. The
ISR can then call the OALPAtoVA function to translate the physical address of the
reserved memory section into a virtual address. Because the ISR runs in kernel mode,
the ISR can access the reserved memory to buffer data from the peripheral device. The
IST, on the other hand, calls MmMapIoSpace outside the kernel to map the physical
memory to a process-specific virtual address. MmMapIoSpace uses VirtualAlloc and
VirtualCopy to map the physical memory into virtual memory, yet you can also call
VirtualAlloc and VirtualCopy directly if you need more control over the address
mapping process.
Another option to pass data from an ISR to an IST is to allocate physical memory in
SDRAM dynamically by using the AllocPhysMem function in the device driver, which
is particularly useful for installable ISRs loaded into the kernel on an as-needed basis.
AllocPhysMem allocates a physically contiguous memory area and returns the
physical address (or fails if the allocation size is not available). The device driver can
280 Chapter 6 Developing Device Drivers

communicate the physical address to the ISR in a call to KernelIoControl based on a


user-defined IO control code. The ISR then uses the OALPAtoVA function to translate
the physical address into a virtual address. The IST uses MmMapIoSpace or the
VirtualAlloc and VirtualCopy functions, as already explained for statically reserved
memory regions.

Installable ISRs
When adapting Windows Embedded CE to a custom target device, it is important to
keep the OAL as generic as possible. Otherwise, you have to modify the code every
time you add a new component to the system. To provide flexibility and adaptability,
Windows Embedded CE supports installable ISRs (IISR), which a device driver can
load into kernel space on demand, such as when new peripheral devices are
connected in a Plug and Play fashion. Installable ISRs also provide a solution to
process interrupts when multiple hardware devices share the same interrupt line. The
ISR architecture relies on lean DLLs that contain the code of the installable ISR and
export the entry points summarized in Table 6–7.
Table 6-7 Exported installable ISR DLL functions
Function Description
ISRHandler This function contains the installable interrupt handler.
The return value is the SYSINTR value for the IST that you
want to run in response to the IRQ registered for the
installable ISR in the call to the LoadIntChainHandler
function. The OAL must support chaining on at least that
IRQ, which means that an unhandled interrupt can be
chained to another handler (which is the installed ISR in
this case) when an interrupt occurs.
CreateInstance This function is called when an installable ISR is loaded
by using the LoadIntChainHandler function. It returns an
instance identifier for the ISR.
DestroyInstance This function is called when an installable ISR is
unloaded by using the FreeIntChainHandler function.
IOControl This function supports communication from the IST to
the ISR.
Lesson 4: Implementing an Interrupt Mechanism in a Device Driver 281

NOTE Generic installable ISR (GIISR)


To help you implement installable ISRs, Microsoft provides a generic installable ISR sample that
covers the most typical needs for many devices. You can find the source code in the following
folder: %_WINCEROOT%\Public\Common\Oak\Drivers\Giisr.

Registering an IISR
The LoadIntChainHandler function expects three parameters that you must specify
to load and register an installable ISR. The first parameter (lpFilename) specifies the
filename of the ISR DLL to load. The second parameter (lpszFunctionName)
identifies the name of the interrupt handler function, and the third parameter (bIRQ)
defines the IRQ number for which you want to register the installable ISR. In response
to hardware disconnect, a device driver can also unload an installable ISR by calling
the FreeIntChainHandler function.

External Dependencies and Installable ISRs


It is important to keep in mind that LoadIntChainHandler loads ISR DLLs into kernel
space, which means that the installable ISR cannot call high-level operating system
APIs and cannot import or implicitly link to other DLLs. If the DLL has explicit or
implicit links to other DLLs, or if it uses the C run-time library, the DLL will not be
able to load. The installable ISR must be completely self-sufficient.
To ensure that an installable ISR does not link to the C run-time libraries or any DLLs,
you must add the following lines to the Sources file in your DLL subproject:
NOMIPS16CODE=1
NOLIBC=1

The NOLIBC=1 directive ensures that the C run-time libraries are not linked and the
NOMIPS16CODE=1 option enables the compiler option /QRimplicit-import, which
prevents implicit links to other DLLs. Note that this directive has absolutely no
relationship to the Microprocessor without Interlocked Pipeline Stages (MIPS) CPU.
282 Chapter 6 Developing Device Drivers

Lesson Summary
Windows Embedded CE relies on ISRs and ISTs to respond to interrupt requests
triggered by internal and external hardware components that require the attention of
the CPU outside the normal code execution path. ISRs are typically compiled directly
into the kernel or implemented in device drivers loaded at boot time and registered
with corresponding IRQs through HookInterrupt calls. You can also implement
installable ISRs in ISR DLLs, which device drivers can load on demand and associate
with an IRQ by calling LoadIntChainHandler. Installable ISRs also enable you to
support interrupt sharing. For example, on a system with only a single IRQ, such as
an ARM–based device, you can modify the OEMInterruptHandler function, which is
a static ISR that loads further installable ISRs depending on the hardware component
that triggered the interrupt.
Apart from the fact that ISR DLLs must not have any dependencies on external code,
ISRs and installable ISRs have many similarities. The primary task of an interrupt
handler is to determine the interrupt source, mask off or clear the interrupt at the
device, and then return a SYSINTR value for the IRQ to notify the kernel about an IST
to run. Windows Embedded CE maintains interrupt mapping tables that associate
IRQs with SYSINTR values. You can define static SYSINTR values in the source code
or use dynamic SYSINTR values that you can request from the kernel at run time. By
using dynamic SYSINTR values, you can increase the portability of your solutions.
According to the SYSINTR value, the kernel can signal an IST event which enables the
corresponding interrupt service thread to resume from a WaitForSingleObject call. By
performing most of the work to handle the IRQ in the IST instead of the ISR, you can
achieve optimal system performance because the system blocks interrupt sources
with lower or equal priority only during ISR execution. The kernel unmasks all
interrupts when the ISR finishes, with the exception of the interrupt currently in
processing. The current interrupt source remains blocked so that a new interrupt
from the same device cannot interfere with the current interrupt handling procedure.
When the IST has finished its work, the IST must call InterruptDone to inform the
kernel that it is ready for a new interrupt so that the kernel’s interrupt support
handler can reenable the IRQ in the interrupt controller.
Lesson 5: Implementing Power Management for a Device Driver 283

Lesson 5: Implementing Power Management


for a Device Driver
As mentioned in Chapter 3, power management is important for Windows
Embedded CE devices. The operating system includes a Power Manager (PM.dll),
which is a kernel component that integrates with Device Manager to enable devices to
manage their own power states and applications to set power requirements for certain
devices. The primary purpose of Power Manager is to optimize power consumption
and to provide an API to system components, drivers, and applications for power
notifications and control. Although Power Manager does not impose strict
requirements on power consumption or capabilities in any particular power state, it is
beneficial to add power management features to a device driver so that you can
manage the state of your hardware components in a way that is consistent with the
power state of the target device. For more information about Power Manager, device
and system power states, and power management features supported in Windows
Embedded CE 6.0, read Chapter 3, “Performing System Programming.”

After this lesson, you will be able to:


■ Identify the power management interface for device drivers.
■ Implement power management in a device driver.
Estimated lesson time: 30 minutes.

Power Manager Device Drivers Interface


Power Manager interacts with power management–enabled drivers through the
XXX_PowerUp, XXX_PowerDown, and XXX_IOControl functions. For example, the
device driver itself can request a change of the device power level from Power Manager
by calling the DevicePowerNotify function. In response, Power Manager calls
XXX_IOControl with an IO control code of IOCTL_POWER_SET passing in the
requested device power state. It might seem overcomplicated for a device driver to
change the power state of its own device through Power Manager, yet this procedure
ensures a consistent behavior and positive end-user experience. If an application
requested a specific power level for the device, Power Manager might not call the
IOCTL_POWER_SET handler in response to DevicePowerNotify. Accordingly, device
drivers should not assume that successful calls to DevicePowerNotify result in a call to
the IOCTL_POWER_SET handler or that any calls to IOCTL_POWER_SET are the
result of a DevicePowerNotify call. Power Manager can send notifications to a device
284 Chapter 6 Developing Device Drivers

driver in many situations, such as during a system power state transition. To receive
power management notifications, device drivers must advertise that they are power
management enabled, either statically through the IClass registry entry in the driver’s
registry subkey or dynamically by using the AdvertiseInterface function.

XXX_PowerUp and XXX_PowerDown


You can use the XXX_PowerUp and XXX_PowerDown stream interface functions to
implement suspend and resume functionality. The kernel calls XXX_PowerDown
right before powering down the CPU and XXX_PowerUp right after powering it up. It
is important to note that the system operates in single-threaded mode during these
stages with most system calls disabled. For this reason, Microsoft recommends using
the XXX_IOControl function instead of XXX_PowerUp and XXX_PowerDown to
implement power management features, including suspend and resume functionality.

CAUTION Power management restrictions


If you implement suspend and resume functionality based on the XXX_PowerUp and
XXX_PowerDown functions, avoid calling system APIs, particularly thread-blocking APIs, such as
WaitForSingleObject. Blocking the active thread in single-threaded mode causes an unrecover-
able system lockup.

IOControl
The best way to implement power manager in a stream driver is to add support for
power management I/O control codes to the driver’s IOControl function. When you
notify Power Manager about your driver’s power management capabilities through an
IClass registry entry or the AdvertiseInterface function, your driver receives
corresponding notification messages.
Table 6–8 lists the IOCTLs that Power Manager can send to a device driver to perform
power management–related tasks.
Table 6-8 Power management IOCTLs
Function Description
IOCTL_POWER_CAPABILITIES Requests information on what
power states the driver
supports. Note that the driver
can still be set to other power
states (D0 through D4).
Lesson 5: Implementing Power Management for a Device Driver 285

Table 6-8 Power management IOCTLs (Continued)


Function Description
IOCTL_POWER_GET Requests the current power
state of the driver.
IOCTL_POWER_SET Sets the power state of the
driver. The driver maps the
received power state number to
actual settings and changes the
device state. The new device
driver power state should be
returned to Power Manager in
the output buffer.
IOCTL_POWER_QUERY Power Manger checks to see if
the driver is able to change the
state of the device. This
function is deprecated.
IOCTL_REGISTER_POWER_RELATIONSHIP Enables a device driver to
register as the proxy for another
device driver, so that Power
Manager passes all power
requests to this device driver.

IClass Power Management Interfaces


Power Manager supports an IClass registry entry that you can configure in the driver’s
registry subkey to associate your driver with one or more device class values. An
IClass value is a globally unique identifier (GUID) referring to an interface, defined
under the HKEY_LOC AL_MACHINE\System\Cur rentControlSet\Control
\Power\Interfaces registry key. The most important interface for driver developers is
the interface for generic power management–enabled devices, associated with the
GUID {A32942B7-920C-486b-B0E6-92A702A99B35}. By adding this GUID to the
IClass registry entry of your device driver, you can inform Power Manager to send
your driver IOCTLs for power management notifications, as illustrated in Figure 6–7.
286 Chapter 6 Developing Device Drivers

Figure 6-7 Configuring the IClass registry entry to receive power management notifications.

MORE INFO Registry settings for power management


You can also use registry settings and device classes to configure the default power states for a
device, as explained in Lesson 5, “Implementing Power Management,” of Chapter 3.
Lesson 5: Implementing Power Management for a Device Driver 287

Lesson Summary
To ensure reliable power management on Windows Embedded CE, device drivers
should not change their own internal power state without the involvement of Power
Manager. Operating system components, drivers, and applications can call the
DevicePowerNotify function to request a power state change. Accordingly, Power
Manager sends a power state change request to the driver if the power state change is
consistent with the current state of the system. The recommended way to add power
management capabilities to a stream driver is to add support for power management
IOCTLs to the XXX_IOControl function. The XXX_PowerUp and XXX_PowerDown
functions only provide limited capability because Power Manager calls these
functions at a time when the system operates in single-thread mode. If the device
advertises a power management interface through an IClass registry entry or calls the
AdvertiseInterface to announce supported IOCTL interfaces dynamically, Power
Manager will send IOCTLs to the device driver in response to power-related events.
288 Chapter 6 Developing Device Drivers

Lesson 6: Marshaling Data across Boundaries


In Windows Embedded CE 6.0, each process has its own separate virtual memory
space and memory context. Accordingly, marshaling data from one process to
another requires either a copying process or a mapping of physical memory sections.
Windows Embedded CE 6.0 handles most of the details and provides system
functions, such as OALPAtoVA and MmMapIoSpace, to map physical memory
addresses to virtual memory addresses in a relatively straightforward way. However,
driver developers must understand the details of data marshaling to ensure a reliable
and secure system. It is imperative to validate embedded pointers and properly
handle asynchronous buffer access so that a user application cannot exploit a kernel-
mode driver to manipulate memory regions that the application should not be able to
access. Poorly implemented kernel-mode drivers can open a back door for malicious
applications to take over the entire system.

After this lesson, you will be able to:


■ Allocate and use buffers in device drivers.
■ Use embedded pointers in an application.
■ Verify the validity of embedded pointers in a device driver.
Estimated lesson time: 30 minutes.

Understanding Memory Access


Windows Embedded CE works in a virtual memory context and hides the physical
memory, as illustrated in Figure 6–8. The operating system relies on the Virtual
Memory Manager (VMM) and the processor’s Memory Management Unit (MMU) for
the translation of the virtual addresses into physical addresses and for other memory
access management tasks.
Lesson 6: Marshaling Data across Boundaries 289

Kernel Space Kernel Space


0xFFFF FFFF 0xFFFF FFFF
System Trap Area System Trap Area
0xF000 0000 0xF000 0000

Kernel Virtual Memory Kernel Virtual Memory


0xD000 0000 0xD000 0000

Object Store Object Store


0xC800 0000 0xC800 0000

Kernel XIP DLLs Kernel XIP DLLs


0xC000 0000 0xC000 0000

Statically Mapped Virtual Statically Mapped Virtual


Addresses : Uncached Memory Page Addresses : Uncached
0xA000 0000 0xA000 0000

Statically Mapped Virtual Memory Page Statically Mapped Virtual


Addresses : Cached Addresses : Cached
0x8000 0000 0x8000 0000

User Space User Space


0x7FFF FFFF 0x7FFF FFFF
Shared System Heap Shared System Heap
0x7000 0000 0x7000 0000
Memory Page
RAM-Backed Mapfiles RAM-Backed Mapfiles

0x6000 0000 Memory Page 0x6000 0000


Shared User DLLs Shared User DLLs
(Code and Data) Memory Page (Code and Data)

0x4000 0000 0x4000 0000


Process Space Physical Process Space
(Executable Code and Data Memory (Executable Code and Data
VM Allocation VM Allocation
File-Backed Mapfiles) File-Backed Mapfiles)
0x0000 0000 0x0000 0000

Virtual Memory Virtual Memory


Figure 6-8 Virtual memory regions in kernel space and user space

Physical addresses are not directly addressable by the CPU except during
initialization before the kernel has enabled the MMU, yet this does not imply that the
physical memory is no longer accessible. In fact, every fully allocated virtual memory
page must map to some actual physical page on the target device. Processes in
separate virtual address spaces only require a mechanism to map the same physical
memory areas into an available virtual memory region to share data. The physical
address is the same across all processes running on the system. Only the virtual
addresses differ. By translating the physical address per process into a valid virtual
290 Chapter 6 Developing Device Drivers

address, processes can access the same physical memory region and share data across
process boundaries.
As mentioned earlier in this chapter, kernel-mode routines, such as ISRs, can call
OALPAtoVA to map a physical address (PA) into a cached or uncached virtual address
(VA). Because OALPAtoVA maps the physical address to a virtual address in the kernel
space, user-mode processes, such as ISTs, cannot use this function. The kernel space
is inaccessible in user mode. However, threads in user-mode processes, such as ISTs,
can call the MmMapIoSpace function to map a physical address to a nonpaged,
cached or uncached virtual address in the user space. The MmMapIoSpace call results
in the creation of a new entry in the MMU Table (TBL) if no match was found or it
returns an existing mapping. By calling the MmUnmapIoSpace function, the user-
mode process can release the memory again.

NOTE Physical memory access restrictions


Applications and user-mode drivers cannot access physical device memory directly. User-mode
processes must call HalTranslateBusAddress to map the physical device memory range for the
bus to a physical system memory address before calling MmMapIoSpace. To convert a bus
address into a virtual address in a single function call, use the TransBusAddrToVirtual function,
which in turn calls HalTranslateBusAddress and MmMapIoSpace.

Allocating Physical Memory


It’s possible to allocate a portion of memory so you can use it in a driver or the kernel.
There are two ways to do this:
■ Dynamically, by calling the AllocPhysMem function AllocPhysMem
allocates contiguous physical memory in one or more pages that you can map to
virtual memory in the user space by calling MmMapIoSpace or OALPAtoVA,
depending on whether the code is running in user mode or kernel mode.
Because physical memory is allocated in units of memory pages, it is not
possible to allocate less than a page of physical memory. The size of the memory
page depends on the hardware platform. A typical page size is 64 KB.
■ Statically, by creating a RESERVED section in the Config.bib file You can
statically reserve physical memory by using the MEMORY section of a run-time
image’s BIB file, such as Config.bib in the BSP folder. Figure 6–9 illustrates this
approach. The names of the memory regions are for informational purposes and
are only used to identify the different memory areas defined on the system. The
important pieces of information are the address definitions and the RESERVED
Lesson 6: Marshaling Data across Boundaries 291

keyword. According to these settings, Windows Embedded CE excludes the


reserved regions from system memory so that they can be used for DMA by
peripherals and data transfers. There is no risk of access conflicts because the
system does not use reserved memory areas.

Figure 6-9 Definition of reserved memory regions in a Config.bib file

Application Caller Buffers


In Windows Embedded CE 6.0, applications and device drivers run in different
process spaces. For example, Device Manager loads stream drivers into the kernel
process (Nk.exe) or into the user-mode driver host process (Udevice.exe), whereas
each application runs in its own individual process space. Because pointers to virtual
memory addresses in one process space are invalid in other process spaces, you must
map or marshal pointer parameters if separate processes are supposed to access the
same buffer region in physical memory for communication and data transfer across
process boundaries.
292 Chapter 6 Developing Device Drivers

Using Pointer Parameters


A pointer parameter is a pointer that a caller can pass as a parameter to a function. The
DeviceIoControl parameters lpInBuf and lpOutBuf are perfect examples. Applications
can use DeviceIoControl to perform direct input and output operations. A pointer to
an input buffer (lpInBuf) and a pointer to an output buffer ( lpOutBuf) enable data
transfer between the application and the driver. DeviceIoControl is declared in
Winbase.h as follows:
WINBASEAPI BOOL WINAPI DeviceIoControl (HANDLE hDevice,
DWORD dwIoControlCode,
__inout_bcount_opt(nInBufSize)LPVOID lpInBuf,
DWORD nInBufSize,
__inout_bcount_opt(nOutBufSize) LPVOID lpOutBuf,
DWORD nOutBufSize,
__out_opt LPDWORD lpBytesReturned,
__reserved LPOVERLAPPED lpOverlapped);

Pointer parameters are convenient to use in Windows Embedded CE 6.0 because the
kernel automatically performs full access checks and marshaling on these
parameters. In the DeviceIoControl declaration above, you can see that the buffer
parameters lpInBuf and lpOutBuf are defined as in/out parameters of a specified size,
while lpBytesReturned is an out-only parameter. Based on these declarations, the
kernel can ensure that an application does not pass in an address to read-only
memory (such as a shared heap, which is read-only to user-mode processes, but
writable to the kernel) as an in/out or out-only buffer pointer or it will trigger an
exception. In this way, Windows Embedded CE 6.0 ensures that an application
cannot gain elevated access permissions to a memory region through a kernel-mode
driver. Accordingly, on the driver’s side, you do not have to perform any access checks
for the pointers passed in through the XXX_IOControl stream interface function
(pBufIn and pBufOut).

Using Embedded Pointers


Embedded pointers are pointers that a caller passes to a function indirectly through
a memory buffer. For example, an application can store a pointer inside the input
buffer passed in to DeviceIoControl through the parameter pointer lpInBuf. The
kernel will automatically check and marshal the parameter pointer lpInBuf, yet the
system has no way to identify the embedded pointer inside the input buffer. As far as
the kernel is concerned, the memory buffer simply contains binary data. Windows
Embedded CE 6.0 provides no mechanisms to specify explicitly that this block of
memory contains pointers.
Lesson 6: Marshaling Data across Boundaries 293

Because embedded pointers bypass the kernel’s access checks and marshaling
helpers, you must perform access checks and marshaling of embedded pointers in
device drivers manually before you can use them. Otherwise, you might create
vulnerabilities that malicious user-mode code can exploit to perform illegal actions
and compromise the entire system. Kernel-mode drivers enjoy a high level of
privileges and can access system memory that user-mode code should not be able to
access.
To verify that the caller process has the required access privileges, marshal the
pointer, and access the buffer, you should call the CeOpenCallerBuffer function.
CeOpenCallerBuffer checks access privileges based on whether the caller is running
in kernel-mode or user-mode, allocates a new virtual address for the physical memory
of the caller’s buffer, and optionally allocates a temporary heap buffer to create a copy
of the caller’s buffer. Because the mapping of the physical memory involves allocating
a new virtual address range inside the driver, do not forget to call CeCloseCallerBuffer
when the driver has finished its processing.

Handling Buffers
Having performed implicit (parameter pointers) or explicit (embedded pointers)
access checks and pointer marshaling, the device driver is ready to access the buffer.
However, access to the buffer is not exclusive. While the device driver reads data from
and writes data to the buffer, the caller might also read and write data concurrently, as
illustrated in Figure 6–10. Security issues can arise if a device driver stores marshaled
pointers in the caller's buffer. A second thread in the application could then
manipulate the pointer to access a protected memory region through the driver. For
this reason, drivers should always make secure copies of the pointers and buffer size
values they receive from a caller and copy embedded pointers to local variables to
prevent asynchronous modification.

IMPORTANT Asynchronous buffer handling


Never use pointers in the caller's buffer after they have been marshaled, and do not use the
caller's buffer to store marshaled pointers or other variables required for driver processing. For
example, copy buffer size values to local variables so that callers cannot manipulate these values
to cause buffer overruns. One way to prevent asynchronous modification of a buffer by the
caller is to call CeOpenCallerBuffer with the ForceDuplicate parameter set to TRUE to copy the
data from the caller’s buffer to a temporary heap buffer.
294 Chapter 6 Developing Device Drivers

User-Mode Application Kernel-Mode Driver

0x8000 0000

0x1FFF FFFF
Memory Buffer CeOpenCallerBuffer
0x7C00 0000

Figure 6-10 Manipulating a marshaled pointer in a shared buffer

Synchronous Access
Synchronous memory access is synonymous with non-concurrent buffer access. The
caller’s thread waits until the function call returns, such as DeviceIoControl, and
there are no other threads in the caller process that access the buffer while the driver
performs its processing tasks. In this scenario, the device driver can use parameter
pointers and embedded pointers (after a call to CeOpenCallerBuffer) without
additional precautions.
The following is an example of accessing a buffer from an application synchronously.
This sample source code is an exerpt from an XXX_IOControl function of a stream
driver:
BOOL SMP_IOControl(DWORD hOpenContext, DWORD dwCode,
PBYTE pBufIn, DWORD dwLenIn,
PBYTE pBufOut, DWORD dwLenOut,
PDWORD pdwActualOut)
{
BYTE *lpBuff = NULL;

...

if (dwCode == IOCTL_A_WRITE_FUNCTION)
{
// Check parameters
if ( pBufIn == NULL || dwLenIn != sizeof(AN_INPUT_STRUCTURE))
{
DEBUGMSG(ZONE_IOCTL, (TEXT("Bad parameters\r\n")));
return FALSE;
}

// Access input buffer


hrMemAccessVal = CeOpenCallerBuffer((PVOID) &lpBuff,
(PVOID) pBufIn,
Lesson 6: Marshaling Data across Boundaries 295

dwLenIn,
ARG_I_PTR,
FALSE);

// Check hrMemAccessVal value


// Access the pBufIn through lpBuff

...

// Close the buffer when it is no longer needed


CeCloseCallerBuffer((PVOID)lpBuff, (PVOID)pBufOut,
dwLenOut, ARG_I_PTR);

...

Asynchronous Access
Asynchronous buffer access assumes that multiple caller and driver threads access
the buffer sequentially or concurrently. Both scenarios present challenges. In the
sequential access scenario, the caller thread might exit before the driver thread has
f inis hed its pro cessing. B y calli ng t he ma rsha ling hel per function
CeAllocAsynchronousBuffer, you must re-marshal the buffer after it was marshaled by
CeOpenCallerBuffer to ensure in the driver that the buffer remains available even if
t he caller’s address space is unavailable. Do not forget to call
CeFreeAsynchronousBuffer after the driver has finished its processing.
To ensure that your device driver works in kernel and user mode, use the following
approach to support asynchronous buffer access:
■ Pointer parameters Pass pointer parameters as scalar DWORD values and
then call CeOpenCallerBuffer and CeAllocAsynchronousBuffer to perform
access check s and marshaling. Note t hat you cannot call
CeAllocAsynchronousBuffer on a pointer parameter in user-mode code or
perform asynchronous write-back of O_PTR or IO_PTR values.
■ Embedded pointers Pass embedded pointers to CeOpenCallerBuffer and
CeAllocAsynchronousBuffer to perform access checks and marshaling.
To address the second scenario of concurrent access, you must create a secure copy of
the buffer after marshaling, as mentioned earlier. Calling CeOpenCallerBuffer with
the ForceDuplicate parameter set to TRUE and CeCloseCallerBuffer is one option.
Another is to call CeAllocDuplicateBuffer and CeFreeDuplicateBuffer for buffers
296 Chapter 6 Developing Device Drivers

referenced by parameter pointers. You can also copy a pointer or buffer into a stack
variable or allocate heap memory by using VirtualAlloc and then use memcpy to copy
the caller’s buffer. Keep in mind that if you do not create a secure copy, you’re leaving
in a vulnerability that a malicious application could use to take control of the system.

Exception Handling
Another important aspect that should not be ignored in asynchronous buffer access
scenarios revolves around the possibility that embedded pointers might not point to
valid memory addresses. For example, an application can pass a pointer to a driver
that refers to an unallocated or reserved memory region, or it could asynchronously
free the buffer. To ensure a reliable system and prevent memory leaks, you should
enclose buffer-access code in a __try frame and any cleanup code to free memory
allocations in a __finally block or an exception handler. For more information about
exception handling, see Chapter 3, “Performing System Programming.”

Lesson Summary
Windows Embedded CE 6.0 facilitates inter-process communication between
applications and device drivers through kernel features and marshaling helper
functions that hide most of the complexities from driver developers. For parameter
pointers, the kernel performs all checks and pointer marshaling automatically. Only
embedded pointers require extra care because the kernel cannot evaluate the content
of application buffers passed to a driver. Validating and marshaling an embedded
pointer in a synchronous access scenario involves a straightfor ward call to
CeOpenCallerBuffer. Asynchronous access scenarios, however, require an additional
call to CeAllocAsynchronousBuffer to marshal the pointer one more time. To ensure
that your driver does not introduce system vulnerabilities, make sure you handle
buffers correctly, create a secure copy of the buffer content so that callers cannot
manipulate the values, and do not use pointers or buffer size values in the caller's
buffer after they have been marshaled. Never store marshaled pointers or other
variables required for driver processing in the caller's buffer.
Lesson 7: Enhancing Driver Portability 297

Lesson 7: Enhancing Driver Portability


Device drivers help to increase the flexibility and portability of the operating system.
Ideally, they require no code changes to run on different target devices with varying
communication requirements. There are several relatively straightforward techniques
that you can use to make your drivers portable and reusable. One common approach
is to maintain configuration settings in the registry instead of hardcoding the
parameters into the OAL or the driver. Windows Embedded CE also supports a
layered architecture based on MDD and PDD that you can leverage in your device
driver design, and there are further techniques that you can use to implement drivers
in a bus-agnostic way to support peripheral devices regardless of the bus type to
which they are connected.

After this lesson, you will be able to:


■ Describe how to use registry settings to increase the portability and reusability of a
device driver.
■ Implement a device driver in a bus-agnostic way.
Estimated lesson time: 15 minutes.

Accessing Registry Settings in a Driver


To increase the portability and reusability of a device driver, you can configure registry
entries, which you should add to the driver’s registry subkey. For example, you can
define I/O-mapped memory addresses or settings for installable ISRs that the device
driver loads dynamically. To access the entries in a device driver’s registry key, the
driver has to identify where its own settings are located. This is not necessarily the
HKEY_LOC AL_MACHINE\Drivers\BuiltIn key. However, the correct path
in fo r m at io n is a va i la bl e i n t h e Ke y va l ue t h a t yo u ca n f in d un der t h e
HKEY_LOCAL_MACHINE\Drivers\Active key in the loaded driver’s subkey. Device
Manager passes the path to the driver’s Drivers\Active subkey to the XXX_Init
function in the LPCTSTR pContext parameter. The device driver can then use this
LPCTSTR value in a call to OpenDeviceKey to obtain a handle to the device’s registry
key. It is not necessary to read the Key values from the driver’s Drivers\Active subkey
directly. The handle returned by OpenDeviceKey points to the driver’s registry key,
which you can use like any other registry handle. Most importantly, do not forget to
close the handle when it is no longer needed.
298 Chapter 6 Developing Device Drivers

TIP XXX_Init function and driver settings


The XXX_Init function is the best place to determine all configuration settings for a driver
defined in the registry. Rather than accessing the registry repeatedly in subsequent stream func-
tion calls, it is good practice to store the configuration information in the device context created
and returned to Device Manager in response to the XXX_Init call.

Interrupt-Related Registry Settings


If your device driver must load an installable ISR for a device and you want to increase
the portability of your code, you can register the ISR handler, IRQ, and SYSINTR
values in registry keys, read these values from the registry when initializing the driver,
verify that the IRQ and SYSINTR values are valid, and then install the specified ISR by
using the LoadIntChainHandler function.
Table 6–9 lists the registry entries that you can configure for this purpose. By calling
the DDKReg_GetIsrInfo function, you can then read these values and pass them to
the LoadIntChainHandler function dynamically. For more information about
interrupt handing in device drivers, see Lesson 4, “Implementing an Interrupt
Mechanism in a Device Driver,” earlier in this chapter.
Table 6-9 Interrupt-related registry entries for device drivers
Registry Type Description
Entry
IRQ REG_DWORD Specifies the IRQ used to request a SYSINTR for
setting up an IST within the driver.
SYSINTR REG_DWORD Specifies a SYSINTR value to use for setting up an
IST within the driver.
IsrDll REG_SZ The filename of the DLL containing the installable
ISR.
IsrHandler REG_SZ Specifies the entry point for the installable ISR that
the specified DLL exposes.
Lesson 7: Enhancing Driver Portability 299

Memory-Related Registry Settings


Memory-related registry values enable you to configure a device through the registry.
Table 6–10 lists the memory-related registry information that a driver can obtain in a
DDKWINDOWINFO structure by calling DDKReg_GetWindowInfo. By using the
BusTransBusAddrToVirtual function, you can map the bus addresses of memory-
mapped windows to physical system addresses you can then translate into virtual
addresses by using MnMapIoSpace.
Table 6-10 Memory-related registry entries for device drivers
Registry Type Description
Entry
IoBase REG_DWORD A bus-relative base of a single memory-mapped
window used by the device.
IoLen REG_DWORD Specifies the length of the memory-mapped
window defined in IoBase.
MemBase REG_MULTI_SZ A bus-relative base of multiple memory-mapped
windows used by the device.
MemLen REG_MULTI_SZ Specifies the length of the memory-mapped
memory windows defined in MemBase.

PCI-Related Registry Settings


Another registry helper function that you can use to populate a DDKPCIINFO
structure with the standard PCI device instance information is DDKReg_GetPciInfo.
Table 6–11 lists the PCI-related settings you can configure in a driver's registry subkey.
Table 6-11 PCI-related registry entries for device drivers
Registry Entry Type Description
DeviceNumber REG_DWORD The PCI device number.
FunctionNumber REG_DWORD The PCI function number of the device,
which indicates a single function device
on a multifunction PCI card.
InstanceIndex REG_DWORD The instance number of the device.
300 Chapter 6 Developing Device Drivers

Table 6-11 PCI-related registry entries for device drivers (Continued)


Registry Entry Type Description
DeviceID REG_DWORD The type of the device
ProgIF REG_DWORD A register-specific programming
interface, for example, USB OHCI or
UHCI.
RevisionId REG_DWORD The revision number of the device.
Subclass REG_DWORD The basic function of the device; for
example, an IDE controller.
SubSystemId REG_DWORD The type of card or subsystem that uses
the device.
SubVendorId REG_DWORD The vendor of the card or subsystem that
uses the device.
VendorId REG_MULTI_SZ The manufacturer of the device.

Developing Bus-Agnostic Drivers


Similar to settings for installable ISRs, memory-mapped windows, and PCI device
instance information, you can maintain any GPIO numbers or timing configurations
in the registry and achieve in this way a bus-agnostic driver design. The underlying
idea of a bus-agnostic driver is the support of multiple bus implementations for the
same hardware chipset, such as PCI or PCMCIA, without requiring code
modifications.
To implement a bus-agnostic driver, use the following approach:
1. Maintain all necessary configuration parameters in the driver’s registry subkey,
and use t he Windows Embedded CE registr y helper functions
DDKReg_GetIsrInfo, DDKReg_GetWindowInfo, and DDKReg_GetPciInfo to
retrieve these settings during driver initialization.
2. Call HalTranslateBusAddress to translate bus-specific addresses to system
physical addresses and then call MmMapIoSpace to map the physical addresses
to virtual addresses.
Lesson 7: Enhancing Driver Portability 301

3. Reset the hardware, mask the interrupt, and load an installable ISR by calling the
L o a d I n t C h a i n H a n d l e r f u n c t i o n w i t h i n fo r m a t i o n o b t a i n e d f ro m
DDKReg_GetIsrInfo.
4. Load any initialization settings for the installable ISR from the registry by using
RegQueryValueEx and pass the values to the installable ISR in a call to
KernelLibIoControl with a user-defined IOCTL. For example, the Generic
Installable Interrupt Service Routine (GIISR) included in Windows Embedded
CE uses an IOCTL_GIISR_INFO handler to initialize instance information that
enables GIISR to recognize when the device’s interrupt bit is set and return the
cor responding SYSINTR value. You can f ind t he source code in t he
C:\Wince600\Public\Common\Oak\Drivers\Giisr folder.
5. Begin the IST by calling the CreateThread function and unmask the interrupt.

Lesson Summary
To increase the portability of a device driver, you can configure the registry entries in
the driver’s registry subkey. Windows Embedded CE provides several registry helper
f u n c t i o n s t h a t yo u c a n t h e n u s e t o r e t r i e v e t h e s e s e t t i n g s , s u c h a s
DDKReg_GetIsrInfo, DDKReg_GetWindowInfo, and DDKReg_GetPciInfo. These
helper functions query specific information for installable ISRs, memory-mapped
w i n d ow s , a n d P C I d e v i c e i n s t a n c e i n fo r m a t i o n , ye t yo u c a n a l s o c a l l
RegQueryValueEx to retrieve values from other registry entries. However, to use any
of these registry functions, you must first obtain a handle to the driver’s registry
subkey by calling OpenDeviceKey. OpenDeviceKey expects a registry path, which
Device Manager passes to the driver in the XXX_Init function call. Do not forget to
close the registry handle when it is no longer needed.
302 Chapter 6 Developing Device Drivers

Lab 6: Developing Device Drivers


In this lab, you implement a stream driver that stores and retrieves a string of 128
Unicode characters in memory. A base version of this driver is available in the
companion material for this book. You only need to add the code as a subproject to an
OS design. You then configure .bib file and registry settings to load this driver
automatically during boot time and create a WCE Console Application to test the
driver’s functionality. In a last step, you add power management support to the string
driver.

NOTE Detailed step-by-step instructions


To help you successfully master the procedures presented in this Lab, see the document
“Detailed Step-by-Step Instructions for Lab 6” in the companion material for this book.

 Add a Stream Interface Driver to a Run-Time Image


1. Clone the Device Emulator BSP and create an OS design based on this BSP as
outlined in Lab 2, “Building and Deploying a Run-Time Image.”
2. Copy the string driver source code that you can find on the companion CD in
the \Labs\StringDriver\String folder into your BSP folder in the path
%_WINCEROOT%\Platform\<BSPName>\Src\Drivers. This should result in a
folder named String in your platform in the Drivers folder, and immediately
inside this folder you should have the files from the driver on the companion
CD, such as sources, string.c, string.def. It’s possible to write a driver from
scratch, but starting from a working example such as this is much quicker.
3. Add an entry to the Dirs file in the Drivers folder above your new String folder to
include the string driver into the build process.

CAUTION Include In Build option


Do not use the Include In Build option in Solution Explorer to include the string driver into the
build process. Solution Explorer removes important CESYSGEN directives from the Dirs file.

4. Add an entry to Platform.bib to add the built string driver, contained in


$(_FLATRELEASEDIR), to the run-time image. Mark the driver module as a
hidden system file.
5. Add the following line to Platform.reg to include the string driver’s .reg file into
the run-time image’s registry:
Lab 6: Developing Device Drivers 303

#include "$(_TARGETPLATROOT)\SRC\DRIVERS\String\string.reg"
6. Build the string driver by right clicking it in Solution Explorer and selecting
Build.
7. Make a new run-time image in Debug mode.

NOTE Building the run-time image in Release mode


If you want to work with the Release version of the run-time image, you must change the
DEBUGMSG statements in the driver code to RETAILMSG statements to output the driver
messages.

8. Open the generated Nk.bin in the flat release directory to verify that it contains
String.dll and registry entries in the HKEY_LOCAL_MACHINE\Drivers
\BuiltIn\String subkey to load the driver at startup.
9. Load the generated image on the Device Emulator.
10. Open the Modules window after the image starts by pressing CTRL+ALT+U, or
open the Debug menu in Visual Studio, point to Windows, and then select
Modules. Verify that the system has loaded string.dll, as illustrated in Figure 6–11.

Figure 6-11 Modules window with loaded string driver

 Access the Driver from an Application


1. Create a new WCE Console Application subproject as part of your OS design by
using the Windows Embedded CE Subproject Wizard. Select WCE Console
Application and the template A Simple Windows Embedded CE Console
Application.
304 Chapter 6 Developing Device Drivers

2. Modify the subproject image settings to exclude the subproject from the image
by right clicking the OS design name in the solution view and selecting
Properties.
3. Include <windows.h> and <winioctl.h>
4. Add code to the application to open an instance of the driver by using CreateFile.
Fo r t h e s e c o n d C re a t eF i le p a r a m et e r ( dw D e s i red A c ce s s ) , p a s s i n
GENERIC_READ|GENERIC_WRITE. For t he f ift h parameter
(dwCreationDisposition), pass in OPEN_EXISTING. If you open the driver with
the $device naming convention, be sure to escape the slashes and not include
the colon at the end of the filename.
HANDLE hDrv = CreateFile(L"\\$device\\STR1",
GENERIC_READ|GENERIC_WRITE,
0, 0, OPEN_EXISTING, 0, 0);

5. Copy the IOCTL header file (String_ioctl.h) from the string driver folder to the
new application’s folder and include it in the source code file.
6. Declare an instance of a PARMS_STRING structure defined in
String_iocontrol.h, included along with the rest of the sample string driver, to
enable applications to store a string in the driver using the following code:
PARMS_STRING stringToStore;
wcscpy_s(stringToStore.szString,
STR_MAX_STRING_LENGTH,
L"Hello, driver!");

7. Use a DeviceIoControl call with an I/O control code of IOCTL_STRING_SET to


store this string in the driver.
8. Run the application by building it and selecting Run Programs from the Target
menu.
9. The Debug window should show the message Stored String "Hello, driver!"
Successfully when you run the application, as shown in Figure 6–12.

Figure 6-12 A debug message from the string driver


Lab 6: Developing Device Drivers 305

 Adding Power Management Support


1. Turn off and detach from the Device Emulator.
2. Add the IClass for generic power management devices with the following line to
the string driver’s registry key in String.reg:
"IClass"=multi_sz:"{A32942B7-920C-486b-B0E6-92A702A99B35}"

3. Add the power management code that you can find in the
StringDriverPowerCode.txt file under \Labs\StringDriver\Power on the
companion CD to the string driver’s IOControl function to support
I O C T L _ P OW E R _ G E T, I O C T L _ P OW E R _ S E T, and
IOCTL_POWER_CAPABILITIES.
4. Add code to the string driver’s device context so it stores its current power state:
CEDEVICE_POWER_STATE CurrentDx;

5. Add the header <pm.h> to the application, and add calls to SetDevicePower with
the name of the string driver and different power states; for example:
SetDevicePower(L"STR1:", POWER_NAME, D2);

6. Run the application again, and observe the power state–related debug messages
when Power Manager changes the string driver’s power state, as shown in Figure
6–13.

Figure 6-13 Power management–related debug messages from the string driver
306 Chapter 6 Review

Chapter Review
Windows Embedded CE 6.0 is extraordinarily modular in its design and supports
ARM-, MIPS-, SH4-, and x86–based boards in a multitude of hardware configurations.
The CE kernel contains the core OS code and the platform-specific code resides in the
OAL and in device drivers. In fact, device drivers are the largest part of the BSP for an
OS design. Rather than accessing the hardware directly, the operating system loads
the corresponding device drivers, and then uses the functions and I/O services that
these drivers provide.
Windows Embedded CE device drivers are DLLs that adhere to a well-known API so
that the operating system can load them. Native CE drivers interface with GWES
while stream drivers interface with Device Manager. Stream drivers implement the
stream interface API so that their resources can be exposed as special file system
resources. Applications can use the standard file system APIs to interact with these
drivers. The stream interface API also includes support for IOCTL handlers, which
come in handy if you want to integrate a driver with Power Manager. For example,
Po w e r M a n a ge r c a l l s X X X _ I O C o n t r o l w i t h a n I O C o n t r o l c o d e o f
IOCTL_POWER_SET passing in the requested device power state.
Native and stream drivers can feature a monolithic or layered design. The layered
design splits the device driver logic in an MDD and a PDD part, which helps to
increase the reusability of the code. The layered design also facilitates driver updates.
Windows Embedded CE also features a flexible interrupt-handling architecture based
on ISRs and ISTs. The ISR's main task is to identify the interrupt source and notify the
kernel with a SYNTINR value about the IST to run. The IST performs the majority of
the processing, such as time-consuming buffer copying processes.
In general, you have two options to load a driver under Windows Embedded CE 6.0.
You can add the driver’s registry settings to the BuiltIn registry key to start the driver
automatically during the boot process or you load the driver automatically in a call to
ActivateDeviceEx. Depending on the driver’s registry entries, you can run a driver in
kernel mode or user mode. Windows Embedded CE 6.0 includes a user-mode driver
host process and a Reflector service that enables most kernel-mode drivers to run in
user mode without code modifications. Because device drivers run in different
process spaces than applications on Windows Embedded CE 6.0, you must marshal
the data in either a mapping of physical memory sections or copying process to
facilitate communication. It is imperative to validate and marshal embedded pointers
by calling CeOpenCallerBuffer and CeAllocAsynchronousBuffer and properly
Chapter 6 Review 307

handling asynchronous buffer access so that a user application cannot exploit a


kernel-mode driver to take over the system.

Key Terms
Do you know what these key terms mean? You can check your answers by looking up
the terms in the glossary at the end of the book.
■ IRQ
■ SYSINTR
■ IST
■ ISR
■ User mode
■ Marshaling
■ Stream interface
■ Native interface
■ PDD
■ MDD
■ Monolithic
■ Bus-agnostic

Suggested Practice
To help you successfully master the exam objectives presented in this chapter,
complete the following tasks:

Enhance Power Management Features


Continue to develop the string driver’s power management code.
■ Clear the string buffer Modify the string driver to delete the contents of the
string buffer when the device driver switches into the power state D3 or D4.
■ Change the power capabilities See what happens when you return a different
POWER_CAPABILITIES value to Power Manager.
308 Chapter 6 Review

More IOCTLs
Expand on the features of the string driver by adding more IOCTL handlers.
■ Reverse the stored string Add an IOCTL to reverse the contents of the string
in the buffer.
■ Concatenate a string Add an IOCTL that concatenates a second string to the
string stored without overrunning the buffer.
■ Embedded pointers Replace the string parameter with a pointer to a string
and access it with CeOpenCallerBuffer.

Installable ISR
Learn more about installable ISRs by reading the product documentation.
■ Learn more about installable ISRs Read the section “Installable ISRs and
Device Drivers” in the Windows Embedded CE 6.0 Documentation, available on
the Microsoft MSDN Web site at http://msdn2.microsoft.com/en-us/library/
aa929596.aspx to learn more about installable ISRs.
■ Find an example of an installable ISR Find an example of an installable ISR
and study its structure. A good starting point is the GIISR code that you can find
in the %_WINCEROOT%\Public\Common\Oak\Drivers\Giisr folder.

You might also like