uhcd.c
上传用户:qiulin1960
上传日期:2013-10-16
资源大小:2844k
文件大小:19k
源码类别:

Windows CE

开发平台:

Windows_Unix

  1. /*++
  2. THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
  3. ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
  4. THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
  5. PARTICULAR PURPOSE.
  6. Copyright (c) 1995-2000 Microsoft Corporation.  All rights reserved.
  7. Module Name:  
  8.     uhcd.c
  9.     
  10. Abstract:  
  11.     Platform dependant part of the USB Universal Host Controller Driver (UHCD).
  12. Notes: 
  13. --*/
  14. #include <windows.h>
  15. #include <nkintr.h>
  16. #include <oalintr.h>
  17. #include <ceddk.h>
  18. #include <uhcdddsi.h>
  19. // Registry key and value names
  20. #define UHCI_DRIVER_KEY         TEXT("Drivers\BuiltIn\UHCI")
  21. #define USE_EXISTING_VALUE_NAME TEXT("UseExistingSettings")
  22. #define IRQ_VALUE_NAME          TEXT("Irq")
  23. #define IOBASE_VALUE_NAME       TEXT("MemBase")
  24. // Amount of memory to use for HCD buffer
  25. static const DWORD gcTotalAvailablePhysicalMemory = 65536; // 64K
  26. static const DWORD gcHighPriorityPhysicalMemory = 0x4000; // 16K
  27. typedef struct _SUhcdPdd
  28. {
  29.     LPVOID lpvMemoryObject;
  30.     LPVOID lpvUhcdMddObject;
  31. } SUhcdPdd;
  32. #define UnusedParameter(x)  x = x
  33. /* UhcdPdd_DllMain
  34.  * 
  35.  *  DLL Entry point.
  36.  *
  37.  * Return Value:
  38.  */
  39. extern BOOL UhcdPdd_DllMain(HANDLE hinstDLL, DWORD dwReason, LPVOID lpvReserved)
  40. {
  41.     UnusedParameter(hinstDLL);
  42.     UnusedParameter(dwReason);
  43.     UnusedParameter(lpvReserved);
  44.     return TRUE;
  45. }
  46. /* GetRegistryConfig
  47.  *
  48.  *   Function to get the IRQ and I/O port range from the registry.  
  49.  *   Note: Will need to be changed to support multiple instances.
  50.  *
  51.  * Return Value:
  52.  *   TRUE for success, FALSE for error
  53.  */
  54. static BOOL
  55. GetRegistryConfig(
  56.     DWORD * lpdwUseExistingSettings, // OUT- Receives value that indicates whether to 
  57.                                      //      just use the resources assigned by the BIOS.  
  58.     DWORD * lpdwIrq,    // OUT- Receives IRQ value
  59.     DWORD * lpdwIoBase) // OUT- Receives I/O base
  60. {
  61.     HKEY hKey;
  62.     DWORD dwData;
  63.     DWORD dwSize;
  64.     DWORD dwType;
  65.     BOOL  fRet=FALSE;
  66.     DWORD dwRet;
  67.     dwRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE,UHCI_DRIVER_KEY,0,0,&hKey);
  68.     if (dwRet != ERROR_SUCCESS) {
  69.         DEBUGMSG(ZONE_ERROR,(TEXT("!UHCD:GetRegistryConfig RegOpenKeyEx(%s) failed %drn"),
  70.                              UHCI_DRIVER_KEY, dwRet));
  71.         return FALSE;
  72.     }
  73.     dwSize = sizeof(dwData);
  74.     dwRet = RegQueryValueEx(hKey,USE_EXISTING_VALUE_NAME,0,&dwType,(PUCHAR)&dwData,&dwSize);
  75.     if (dwRet != ERROR_SUCCESS) {
  76.         DEBUGMSG(ZONE_ERROR, (TEXT("!UHCD:GetRegistryConfig RegQueryValueEx(%s) failed %drn"),
  77.                               USE_EXISTING_VALUE_NAME, dwRet));
  78.         goto GetRegistryConfig_exit;
  79.     }
  80.     *lpdwUseExistingSettings = dwData;
  81.     dwSize = sizeof(dwData);
  82.     dwRet = RegQueryValueEx(hKey,IRQ_VALUE_NAME,0,&dwType,(PUCHAR)&dwData,&dwSize);
  83.     if (dwRet != ERROR_SUCCESS) {
  84.         DEBUGMSG(ZONE_ERROR, (TEXT("!UHCD:GetRegistryConfig RegQueryValueEx(%s) failed %drn"),
  85.                               IRQ_VALUE_NAME, dwRet));
  86.         goto GetRegistryConfig_exit;
  87.     }
  88.     *lpdwIrq = dwData;
  89.     dwSize = sizeof(dwData);
  90.     dwRet = RegQueryValueEx(hKey,IOBASE_VALUE_NAME,0,&dwType,(PUCHAR)&dwData,&dwSize);
  91.     if (dwRet != ERROR_SUCCESS) {
  92.         DEBUGMSG(ZONE_ERROR,(TEXT("!UHCD:GetRegistryConfig RegQueryValueEx(%s) failed %drn"),
  93.                              IOBASE_VALUE_NAME, dwRet));
  94.         goto GetRegistryConfig_exit;
  95.     }
  96.     *lpdwIoBase = dwData;
  97.     fRet = TRUE;
  98. GetRegistryConfig_exit:
  99.     RegCloseKey(hKey);
  100.     return fRet;
  101. }   // GetRegistryConfig
  102. /* SetRegistryConfig
  103.  *
  104.  *   Function to set the IRQ and I/O port range in the registry.
  105.  *   Note: Will need to be changed to support multiple instances.
  106.  *
  107.  * Return Value:
  108.  *   TRUE for success, FALSE for error
  109.  */
  110. static BOOL
  111. SetRegistryConfig(
  112.     DWORD dwIrq,    // IN - IRQ value
  113.     DWORD dwIoBase) // IN - I/O base
  114. {
  115.     HKEY hKey;
  116.     BOOL  fRet=FALSE;
  117.     DWORD dwRet;
  118.     dwRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE,UHCI_DRIVER_KEY,0,0,&hKey);
  119.     if (dwRet != ERROR_SUCCESS) {
  120.         DEBUGMSG(ZONE_ERROR,(TEXT("!UHCD:SetRegistryConfig RegOpenKeyEx(%s) failed %drn"),
  121.                              UHCI_DRIVER_KEY, dwRet));
  122.         return FALSE;
  123.     }
  124.     dwRet = RegSetValueEx(hKey,IRQ_VALUE_NAME,0,REG_DWORD,(PUCHAR)&dwIrq,sizeof(DWORD));
  125.     if (dwRet != ERROR_SUCCESS) {
  126.         DEBUGMSG(ZONE_ERROR, (TEXT("!UHCD:SetRegistryConfig RegQueryValueEx(%s) failed %drn"),
  127.                               IRQ_VALUE_NAME, dwRet));
  128.         goto SetRegistryConfig_exit;
  129.     }
  130.     dwRet = RegSetValueEx(hKey,IOBASE_VALUE_NAME,0,REG_DWORD,(PUCHAR)&dwIoBase,sizeof(DWORD));
  131.     if (dwRet != ERROR_SUCCESS) {
  132.         DEBUGMSG(ZONE_ERROR,(TEXT("!UHCD:SetRegistryConfig RegQueryValueEx(%s) failed %drn"),
  133.                              IOBASE_VALUE_NAME, dwRet));
  134.         goto SetRegistryConfig_exit;
  135.     }
  136.     fRet = TRUE;
  137. SetRegistryConfig_exit:
  138.     RegCloseKey(hKey);
  139.     return fRet;
  140. }   // SetRegistryConfig
  141. /* ConfigureUHCICard
  142.  * 
  143.  *    Configure UHCI controller I/O base and IRQ based on settings read from registry.
  144.  *    This should really call routines which access the IRQ routing and memory base through
  145.  *    the PCI BIOS and reconfigure the card.  Since we currently don't have the routines
  146.  *    to do this, use the following algorithm:
  147.  *        -- If fUseExisting flag is set, read the values from the PCI config registers on the
  148.  *           card, and update pioPortBase and pIrq.
  149.  *        -- If fUseExisting is not set, verify that the passed in values of pioPortBase and
  150.  *           pIrq match, and return FALSE if they do not. (change - don't fail if mem base
  151.  *           doesn't match, since this is determined by the PCI config mechanism, it will not
  152.  *           conflict with other cards.)
  153.  *
  154.  * Return Value
  155.  *    Return TRUE if successful, FALSE if card could not be detected, or if configuration
  156.  *    does not match what was specified in the registry, and fUseExistingSettings is not set.
  157.  */
  158. BOOL
  159. ConfigureUHCICard(
  160.     BOOL fUseExistingSettings,
  161.     PUCHAR *pioPortBase,   // IN - contains physical address of register base
  162.                            // OUT- contains virtual address of register base
  163.     ULONG *pIrq)           // IN - contains IRQ value
  164.                            // OUT- (if fUseExistingSettings) contains actual IRQ value
  165. {
  166.     PCI_SLOT_NUMBER     slotNumber;
  167.     PCI_COMMON_CONFIG   pciConfig;
  168.     int                 bus, device, function;
  169.     int                 length;
  170.     BOOL                bFoundIt = FALSE;
  171. // when detecting a PIIX4, these are the values which appear
  172. #define UHCD_VENDOR_ID_INTEL ((USHORT)0x8086)
  173. #define UHCD_DEVICE_ID_PIIX4 ((USHORT)0x7112)
  174. // PIIX4 is strange in that the port base address
  175. // is stored in BaseAddresses[ 4 ]. In fact, the
  176. // BaseAddresses[ 0-3 ] are all 0, for some reason.
  177. #define UHCD_PIIX4_ADDRESS_REG 4
  178.     for (bus = 0; bus < PCI_MAX_BUS; bus++) {
  179.         for (device = 0; device < PCI_MAX_DEVICES; device++) {
  180.             slotNumber.u.bits.DeviceNumber = device;
  181.             for (function = 0; function < PCI_MAX_FUNCTION; function++) {
  182.                 slotNumber.u.bits.FunctionNumber = function;
  183.                 length = HalGetBusData(PCIConfiguration, bus, slotNumber.u.AsULONG,
  184.                                        &pciConfig, sizeof(pciConfig) - 
  185.                                        sizeof(pciConfig.DeviceSpecific));
  186.                 if (length == 0 || pciConfig.VendorID == 0xFFFF) 
  187.                     break;
  188.                 // the only UHCI controller I've tested so far is the 
  189.                 // PIIX4. The rest of this code will need to be altered
  190.                 // to support other controllers.
  191.                 //
  192.                 // This DEBUGCHK says:
  193.                 // "halt if we find an Intel USB controller that is not a PIIX4"
  194.                 DEBUGCHK( !(pciConfig.BaseClass == PCI_CLASS_SERIAL_BUS_CTLR &&
  195.                             pciConfig.SubClass  == PCI_SUBCLASS_SB_USB &&
  196.                             pciConfig.VendorID == UHCD_VENDOR_ID_INTEL ) ||
  197.                           pciConfig.DeviceID == UHCD_DEVICE_ID_PIIX4 );
  198.                 if (pciConfig.BaseClass == PCI_CLASS_SERIAL_BUS_CTLR &&
  199.                     pciConfig.SubClass  == PCI_SUBCLASS_SB_USB &&
  200.                     pciConfig.VendorID == UHCD_VENDOR_ID_INTEL && 
  201.                     pciConfig.DeviceID == UHCD_DEVICE_ID_PIIX4 ) {
  202.                     RETAILMSG(1,(TEXT("UHCD: Found PCI USB UHCI controller, VendorId: %04X, DeviceId: %04Xrn"),
  203.                                  pciConfig.VendorID, pciConfig.DeviceID));
  204.                     bFoundIt = TRUE;
  205.                     DEBUGMSG( ZONE_INIT,
  206.                               (TEXT("UHCD: PCI config: Irq = 0x%X, Port Base = 0x%Xrn"), 
  207.                               pciConfig.u.type0.InterruptLine, pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ]) );
  208.                 
  209.                     // Update current config
  210.                     if (fUseExistingSettings) {
  211.                         *pIrq = pciConfig.u.type0.InterruptLine;
  212.                         *pioPortBase = (PUCHAR)pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ];
  213.                         // Save new values in registry
  214.                         SetRegistryConfig(*pIrq, (DWORD)*pioPortBase);
  215.                     }
  216.                     else if ((pciConfig.u.type0.InterruptLine != (UCHAR) *pIrq) ||
  217.                         (pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ] != (DWORD)*pioPortBase)) {
  218.                         RETAILMSG(1,(TEXT("!UHCD:  Configuration different from registry: IRQ %u, PortBase: 0x%X (reg: %u, 0x%X)rn"),
  219.                                      pciConfig.u.type0.InterruptLine,pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ],
  220.                                      (UCHAR)*pIrq,(DWORD)*pioPortBase));
  221.                         // If IRQ isn't what we expect, fail init (otherwise might conflict with ISA 
  222.                         // devices).  Shouldn't have a problem with conflicting memory addresses, so
  223.                         // don't fail if that doesn't match.
  224.                         if (pciConfig.u.type0.InterruptLine != (UCHAR) *pIrq)
  225.                             bFoundIt = FALSE;
  226.                     }
  227.                     break;
  228.                 }
  229.                 if (function == 0 && !(pciConfig.HeaderType & 0x80)) 
  230.                     break;
  231.             }
  232.             if (bFoundIt || length == 0)
  233.                 break;
  234.         }
  235.         if (bFoundIt || (length == 0 && device == 0))
  236.             break;
  237.     }
  238.     if (!bFoundIt) {
  239.         DEBUGMSG(ZONE_ERROR, (TEXT("UHCD: Error, can't find USB Controllerrn")));
  240.         return FALSE;
  241.     }
  242.     DEBUGMSG(ZONE_INIT, (TEXT("UHCD: Revision Stepping value = 0x%02Xrn"), pciConfig.RevisionID));
  243.     // "PCI System Architecture, 3rd Ed" by Shanley/Anderson,
  244.     // chapter 17, page 349 (I/O Base Address Register)
  245.     // states that for I/O mapping, the address register
  246.     // will end in 01. Page 348 shows that for memory mapping,
  247.     // the register ends in 0.
  248. #if 1
  249.     // so far, the only UHCI controller I've tested is the PIIX4, which
  250.     // is I/O mapped.
  251.     DEBUGCHK( pciConfig.VendorID == UHCD_VENDOR_ID_INTEL &&
  252.               pciConfig.DeviceID == UHCD_DEVICE_ID_PIIX4 );
  253.     // check for I/O mapping (see comment above)
  254.     DEBUGCHK( (((DWORD)*pioPortBase) & ~PCI_ADDRESS_IO_ADDRESS_MASK) == PCI_ADDRESS_IO_SPACE );
  255.     *pioPortBase = (PUCHAR)( ((DWORD)*pioPortBase) & PCI_ADDRESS_IO_ADDRESS_MASK ); 
  256.     DEBUGMSG(ZONE_INIT, (TEXT("UHCD: Using I/O Mapping. ioPortBase = 0x%Xn"), *pioPortBase)); 
  257. #else
  258.     if ( ((DWORD)*pioPortBase) & 0x1 ) {
  259.         DEBUGCHK( (((DWORD)*pioPortBase) & (1 << 1) ) == 0 );
  260.         *pioPortBase = (PUCHAR)(((DWORD)*pioPortBase) ^ 0x1); 
  261.         DEBUGMSG(ZONE_INIT, (TEXT("UHCD: Using I/O Mapping. ioPortBase = 0x%Xn"), *pioPortBase)); 
  262.     } else {
  263.         // Memory mapped
  264.         ULONG               portBase;
  265.         PHYSICAL_ADDRESS    ioPhysicalBase;
  266.         ULONG               inIoSpace = 0;
  267.         ULONG               portRange = 0x70; // cDefaultUhcdPortRange;
  268.         // So far, we haven't seen any memory mapped UHCI controllers
  269.         DEBUGCHK( FALSE );
  270.         // The bottom bit is 0 for memory-mapped
  271.         DEBUGCHK( !(pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ] & 0x1) );
  272.         portBase = pciConfig.u.type0.BaseAddresses[ UHCD_PIIX4_ADDRESS_REG ];
  273.     
  274.         ioPhysicalBase.LowPart = portBase;
  275.     
  276.         if (HalTranslateBusAddress(PCIBus, bus, ioPhysicalBase, &inIoSpace, 
  277.                                    &ioPhysicalBase)) {
  278.             if (!inIoSpace) {
  279.                 if ((*pioPortBase = (PUCHAR)MmMapIoSpace(ioPhysicalBase, 
  280.                                                          portRange, FALSE)) == NULL) {
  281.                     // We may as well not continue
  282.                     DEBUGMSG(ZONE_ERROR, (TEXT("UHCD: Error mapping I/O Ports.rn")));
  283.                     return FALSE;
  284.                 }
  285.             } else {
  286.                 *pioPortBase = (PUCHAR)ioPhysicalBase.LowPart;
  287.             }
  288.         } else {
  289.             DEBUGMSG(ZONE_ERROR, (TEXT("UHCD: Error translating I/O Ports.rn")));
  290.             return FALSE;
  291.         }
  292.     
  293.         DEBUGMSG(ZONE_INIT, 
  294.                  (TEXT("UHCD: ioPhysicalBase 0x%X, IoSpace 0x%Xrn"),
  295.                   ioPhysicalBase.LowPart, inIoSpace));
  296.         DEBUGMSG(ZONE_INIT, 
  297.                  (TEXT("UHCD: ioPortBase 0x%X, portBase 0x%Xrn"),
  298.                   *pioPortBase, portBase));
  299.     }
  300. #endif
  301.     return TRUE;
  302. }
  303. /* InitializeUHCI
  304.  *
  305.  *  Configure and initialize UHCI card
  306.  *
  307.  * Return Value:
  308.  *  Return TRUE if card could be located and configured, otherwise FALSE
  309.  */
  310. static BOOL 
  311. InitializeUHCI(
  312.     SUhcdPdd * pPddObject,    // IN - Pointer to PDD structure
  313.     LPCWSTR szDriverRegKey)   // IN - Pointer to active registry key string
  314. {
  315.     DWORD dwUseExistingSettings;
  316.     PUCHAR ioPortBase = NULL;
  317.     DWORD dwSysIntr, dwIRQ;
  318.     BOOL fResult = FALSE;
  319.     LPVOID pobMem = NULL;
  320.     LPVOID pobUhcd = NULL;
  321.     if (!GetRegistryConfig(&dwUseExistingSettings, &dwIRQ, (DWORD *)&ioPortBase)) {
  322.         RETAILMSG(1,(TEXT("!UHCD: Error reading registry settingsrn")));
  323.         return FALSE;
  324.     }
  325.     DEBUGMSG(ZONE_INIT,(TEXT("UHCD: Read config from registry: Use existing: %u, IRQ: %u, I/O base: %Xrn"),
  326.                         dwUseExistingSettings,dwIRQ,ioPortBase));
  327.     fResult   = ConfigureUHCICard(!!dwUseExistingSettings,&ioPortBase, &dwIRQ);
  328.     if(fResult)
  329.     {
  330.         dwSysIntr = MapIrq2SysIntr(dwIRQ);
  331.         DEBUGMSG(ZONE_INIT,(TEXT("UHCD: MapIrq2SysIntr(%u): %urn"),dwIRQ,dwSysIntr));
  332.         // The PDD can supply a buffer of contiguous physical memory here, or can let the 
  333.         // MDD try to allocate the memory from system RAM.  In our case, let the MDD do it.
  334.         pobMem = UhcdMdd_CreateMemoryObject(gcTotalAvailablePhysicalMemory, 
  335.                                             gcHighPriorityPhysicalMemory, NULL,NULL); 
  336.         if(pobMem)
  337.         {
  338.             pobUhcd = UhcdMdd_CreateUhcdObject(pPddObject, pobMem,
  339.                     szDriverRegKey, ioPortBase, dwSysIntr);
  340.             fResult = pobUhcd ? TRUE : FALSE;
  341.         }
  342.         else
  343.             fResult = FALSE;
  344.         if(!fResult)
  345.         {
  346.             if(pobUhcd)
  347.                 UhcdMdd_DestroyUhcdObject(pobUhcd);
  348.             if(pobMem)
  349.                 UhcdMdd_DestroyMemoryObject(pobMem);
  350.             pobUhcd = NULL;
  351.             pobMem = NULL;
  352.         }
  353.     }
  354.     pPddObject->lpvMemoryObject = pobMem;
  355.     pPddObject->lpvUhcdMddObject = pobUhcd;
  356.     return fResult;
  357. }
  358. /* UhcdPdd_Init
  359.  *
  360.  *   PDD Entry point - called at system init to detect and configure UHCI card.
  361.  *
  362.  * Return Value:
  363.  *   Return pointer to PDD specific data structure, or NULL if error.
  364.  */
  365. extern DWORD 
  366. UhcdPdd_Init(
  367.     DWORD dwContext)  // IN - Pointer to context value. For device.exe, this is a string 
  368.                       //      indicating our active registry key.
  369. {
  370.     SUhcdPdd *  pPddObject = malloc(sizeof(SUhcdPdd));
  371.     BOOL        fRet = FALSE;
  372.     fRet = InitializeUHCI(pPddObject, (LPCWSTR)dwContext);
  373.     if(!fRet)
  374.     {
  375.         free(pPddObject);
  376.         pPddObject = NULL;
  377.     }
  378.     return (DWORD)pPddObject;
  379. }
  380. /* UhcdPdd_CheckConfigPower
  381.  *
  382.  *    Check power required by specific device configuration and return whether it
  383.  *    can be supported on this platform.  For CEPC, this is trivial, just limit to
  384.  *    the 500mA requirement of USB.  For battery powered devices, this could be 
  385.  *    more sophisticated, taking into account current battery status or other info.
  386.  *
  387.  * Return Value:
  388.  *    Return TRUE if configuration can be supported, FALSE if not.
  389.  */
  390. extern BOOL UhcdPdd_CheckConfigPower(
  391.     UCHAR bPort,         // IN - Port number
  392.     DWORD dwCfgPower,    // IN - Power required by configuration
  393.     DWORD dwTotalPower)  // IN - Total power currently in use on port
  394. {
  395.     return ((dwCfgPower + dwTotalPower) > 500) ? FALSE : TRUE;
  396. }
  397. extern void UhcdPdd_PowerUp(DWORD hDeviceContext)
  398. {
  399.     SUhcdPdd * pPddObject = (SUhcdPdd *)hDeviceContext;
  400.     UhcdMdd_PowerUp(pPddObject->lpvUhcdMddObject);
  401.     return;
  402. }
  403. extern void UhcdPdd_PowerDown(DWORD hDeviceContext)
  404. {
  405.     SUhcdPdd * pPddObject = (SUhcdPdd *)hDeviceContext;
  406.     UhcdMdd_PowerDown(pPddObject->lpvUhcdMddObject);
  407.     return;
  408. }
  409. extern BOOL UhcdPdd_Deinit(DWORD hDeviceContext)
  410. {
  411.     SUhcdPdd * pPddObject = (SUhcdPdd *)hDeviceContext;
  412.     if(pPddObject->lpvUhcdMddObject)
  413.         UhcdMdd_DestroyUhcdObject(pPddObject->lpvUhcdMddObject);
  414.     if(pPddObject->lpvMemoryObject)
  415.         UhcdMdd_DestroyMemoryObject(pPddObject->lpvMemoryObject);
  416.     return TRUE;
  417. }
  418. extern DWORD UhcdPdd_Open(DWORD hDeviceContext, DWORD AccessCode,
  419.         DWORD ShareMode)
  420. {
  421.     UnusedParameter(hDeviceContext);
  422.     UnusedParameter(AccessCode);
  423.     UnusedParameter(ShareMode);
  424.     return 1; // we can be opened, but only once!
  425. }
  426. extern BOOL UhcdPdd_Close(DWORD hOpenContext)
  427. {
  428.     UnusedParameter(hOpenContext);
  429.     return TRUE;
  430. }
  431. extern DWORD UhcdPdd_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count)
  432. {
  433.     UnusedParameter(hOpenContext);
  434.     UnusedParameter(pBuffer);
  435.     UnusedParameter(Count);
  436.     return (DWORD)-1; // an error occured
  437. }
  438. extern DWORD UhcdPdd_Write(DWORD hOpenContext, LPCVOID pSourceBytes,
  439.         DWORD NumberOfBytes)
  440. {
  441.     UnusedParameter(hOpenContext);
  442.     UnusedParameter(pSourceBytes);
  443.     UnusedParameter(NumberOfBytes);
  444.     return (DWORD)-1;
  445. }
  446. extern DWORD UhcdPdd_Seek(DWORD hOpenContext, LONG Amount, DWORD Type)
  447. {
  448.     UnusedParameter(hOpenContext);
  449.     UnusedParameter(Amount);
  450.     UnusedParameter(Type);
  451.     return (DWORD)-1;
  452. }
  453. extern BOOL UhcdPdd_IOControl(DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn,
  454.         DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut)
  455. {
  456.     UnusedParameter(hOpenContext);
  457.     UnusedParameter(dwCode);
  458.     UnusedParameter(pBufIn);
  459.     UnusedParameter(dwLenIn);
  460.     UnusedParameter(pBufOut);
  461.     UnusedParameter(dwLenOut);
  462.     UnusedParameter(pdwActualOut);
  463.     return FALSE;
  464. }
  465. // Manage WinCE suspend/resume events
  466. // This gets called by the MDD's IST when it detects a power resume.
  467. // The CEPC does not have power suspend/resume functionality so
  468. // there's nothing we need to do beyond merely providing the entry point.
  469. extern void UhcdPdd_InitiatePowerUp (void)
  470. {
  471.     return;
  472. }