// Subproject1.cpp : Defines the entry point for the DLL application.
//
#include "stdafx.h"
#include <windows.h>
#include "ceddk.h"
void DBGout(DWORD dwValue);
//==============================================================================================================================
BOOL APIENTRY DllMain( HANDLE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
}
//==========================================================================================================
DWORD_GPI_Read(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count)
{
return 0;
}
DWORD_GPI_Write(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count)
{
return 0;
}
DWORD_GPI_IOControl(DWORD hOpenContext,DWORD dwCode,PBYTE pBufIn,DWORD dwLenIn,PBYTE pBufOut,DWORD dwLenOut,PDWORD pdwActualOut)
{
return true;
}
写了个GPIO口得驱动函数内容都是空的,怎么编译还是过不了呢?
下面是爆出的错误。
:ERRORE] f:\wince600\platform\at10x0bd\src\drivers\subproject1\subproject1.cpp(18) : error C4716: 'DllMain' : must return a value
如果我在第18行加上return true;
BOOL APIENTRY DllMain( HANDLE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
//18行
}
编译结果出现这些错误
BUILD: [01:0000000065:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol DllEntry
BUILD: [01:0000000066:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Close
BUILD: [01:0000000067:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Deinit
BUILD: [01:0000000068:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_IOControl
BUILD: [01:0000000069:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Init
BUILD: [01:0000000070:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Open
BUILD: [01:0000000071:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_PowerDown
BUILD: [01:0000000072:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_PowerUp
BUILD: [01:0000000073:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Read
BUILD: [01:0000000074:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Seek
BUILD: [01:0000000075:ERRORE] Subproject1.def : error LNK2001: unresolved external symbol GPI_Write
BUILD: [01:0000000076:ERRORE] F:\WINCE600\platform\AT10X0BD\lib\ARMV4I\retail\Subproject1.lib : fatal error LNK1120: 11 unresolved externals
11 个解决方案
#1
有sources文件没啊?
#2
DWORD_GPI_Write 中间是不是多了个下划线?
#3
第一,检查下你的写的流驱动的过程对不对,是否对注册表、*.bib文件 *.def 文件进行了修改,最好不要落下哪一步,那样的话会出现莫名的问题(当然有的可能无关你的问题)
第二,“error LNK2001: unresolved external symbol DllEntry“很有可能是你的导出的接口函数没有使用编辑器的扩展关键字_declspec(dllexport)字样
我写过的类似的代码:
#include "stdafx.h"
#include<windows.h>
//#include<C:\WINCE600\PUBLIC\COMMON\DDK\INC\ceddk.h>
#include<ceddk.h>
#pragma comment(lib,"CEDDK.lib")
#include<Ddkreg.h>
#include<dbgapi.h>
//#include<oalintr.h>
#include<Nkintr.h>
_declspec(dllexport) extern DWORD AUD_Init(LPCTSTR pContext,LPCVOID lpvBusContext);
_declspec(dllexport) extern BOOL AUD_Deinit(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Open(DWORD hDeviceContext,DWORD AccessCode,DWORD ShareMode);
_declspec(dllexport) extern BOOL AUD_Close(DWORD hOpenContext);
_declspec(dllexport) extern BOOL AUD_IOControl( DWORD hOpenContext,DWORD dwCode,//用于描述这次IOControl操作的语意
PBYTE pBufIn,//指向需要传送给驱动程序使用的数据
DWORD dwLenIn,//要传送给驱动程序使用的数据长度
PBYTE pBufOut,//指向驱动程序传给应用程序使用的数据
DWORD dwLenOut,//要传送给应用程序使用的数据长度
PDWORD pdwActualOut//返回实际处理数据的长度
);
_declspec(dllexport) extern void AUD_PowerUp(DWORD hDeviceContext);
_declspec(dllexport) extern void AUD_PowerDown(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Read(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Write(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Seek(DWORD hOpenContext,LONG Amount,WORD Type);
在函数使用前进行了一个声明,虽然比较笨,但是可行
哈哈 楼主 给分哦
第二,“error LNK2001: unresolved external symbol DllEntry“很有可能是你的导出的接口函数没有使用编辑器的扩展关键字_declspec(dllexport)字样
我写过的类似的代码:
#include "stdafx.h"
#include<windows.h>
//#include<C:\WINCE600\PUBLIC\COMMON\DDK\INC\ceddk.h>
#include<ceddk.h>
#pragma comment(lib,"CEDDK.lib")
#include<Ddkreg.h>
#include<dbgapi.h>
//#include<oalintr.h>
#include<Nkintr.h>
_declspec(dllexport) extern DWORD AUD_Init(LPCTSTR pContext,LPCVOID lpvBusContext);
_declspec(dllexport) extern BOOL AUD_Deinit(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Open(DWORD hDeviceContext,DWORD AccessCode,DWORD ShareMode);
_declspec(dllexport) extern BOOL AUD_Close(DWORD hOpenContext);
_declspec(dllexport) extern BOOL AUD_IOControl( DWORD hOpenContext,DWORD dwCode,//用于描述这次IOControl操作的语意
PBYTE pBufIn,//指向需要传送给驱动程序使用的数据
DWORD dwLenIn,//要传送给驱动程序使用的数据长度
PBYTE pBufOut,//指向驱动程序传给应用程序使用的数据
DWORD dwLenOut,//要传送给应用程序使用的数据长度
PDWORD pdwActualOut//返回实际处理数据的长度
);
_declspec(dllexport) extern void AUD_PowerUp(DWORD hDeviceContext);
_declspec(dllexport) extern void AUD_PowerDown(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Read(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Write(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Seek(DWORD hOpenContext,LONG Amount,WORD Type);
在函数使用前进行了一个声明,虽然比较笨,但是可行
哈哈 楼主 给分哦
#4
#5
非常感谢楼上的细心回答啊,呵呵!我早上在网上找了个比较细的步骤,按它的方法修改了一下,光是编译我加进去那个流驱动的目录不会出错,但是在把目标文件拷到板子的时候,串口打印信息没我驱动里那条信息。我就把目标文件全删了,整个重新编译了一遍,结果冒出一千多个错,这回光编译我那个就不行了,救命哦,55~~
#6
这种问题大多是重复定义,或者两个cpp用公共变量出现的问题。看看头文件和全局变量
#7
这个因为我也是在网上找的别人的帖子学的,应该没那个问题吧?
我现在编译似乎能通过了, 但是烧录到板子还是看不到我的打印信息啊?
我现在编译似乎能通过了, 但是烧录到板子还是看不到我的打印信息啊?
#8
除了流驱动程序的内容外,我几乎是照抄这个方法,那不会不适合我的工程吧?
1. DWORD XXX_Init(LPCTSTR pContext, DWORD dwBusContext):
该函数用于初始化一个流设备驱动,在设备被加载的时候调用,调用成功后会返回一个句柄。 pContext:在Active注册表键路径下的一个字符串 dwBusContext:不常用,这里可以设为0 2. BOOL XXX_Deinit(DWORD hDeviceContext): 卸载一个设备驱动。 hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
3. DWORD XXX_Open(DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode): 打开一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
AccessCode:访问权限代码,一般是只读或者只写或者读写
ShareMode:共享模式,是否支持共享或者独享
4. BOOL XXX_Close(DWORD hOpenContext):
关闭一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
5. DWORD XXX_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count):
从设备上面读取数据。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:读取数据的字节数
6. DWORD XXX_Write(DWORD hOpenContext, LPCVOID pBuffer, DWORD Count):
写数据到设备上面。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:写入数据的字节数
7. DWORD XXX_Seek(DWORD hOpenContext, long Amount, WORD Type):
移动设备中的数据指针。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
Amount:移动的字节数
Type:FILE_BEGIN表示从头移动
FILE_CURRENT表示从当前位置移动
FILE_END表示从末尾往前移动
8. void XXX_PowerUp(DWORD hOpenContext):
打开设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
9. void XXX_PowerDown(DWORD hOpenContext):
关闭设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
10. BOOL XXX_IOControl(DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut):
设备IO控制操作函数。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
dwCode:操作码
pBufIn:输入Buffer
dwLenIn:输入Buffer的size
pBufOut:输出Buffer
dwLenOut:输出Buffer的size
pdwActualOut:实际输出的字节数
11. BOOL XXX_PreClose(DWORD hOpenContext):
标记一个正要关闭的句柄为无效,并唤醒所有正在休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
12. BOOL XXX_PreDeinit(DWORD hDeviceContext):
标记一个设备实例为无效,并唤醒所有休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
上面这些函数就是流设备驱动的所有接口函数,理解起来应该不难。下面介绍一个实际的流设备驱动的例子,是基于WinCE6.0的(和WinCE5.0比一些配置文件稍有不同)。这里介绍的是一个操作GPIO的流设备驱动并介绍具体添加流设备驱动的步骤:
(1) 更改BSP工程文件,添加GPIO驱动的选项:
在BSP目录下面的”CATALOG”文件夹下面找到”BspName.pbcxml”并用记事本打开,然后添加GPIO驱动的选项,首先找到<BSP>…</BSP>并在里面添加下面一行:
<BspItemId>Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx</BspItemId>
然后在< CatalogFile >…</CatalogFile>中添加下面的驱动描述:
<Item Id="Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx">
<Title>GPIO</Title>
<Description>GPIO Driver</Description>
<Type>BspSpecific</Type>
<Variable>BSP_EP94XX_GPIO</Variable>
<Location>Device Drivers</Location>
<SourceCode>
<Title>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Title>
<Path>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Path>
</SourceCode>
</Item>
上面实际上添加了GPIO驱动,环境变量为BSP_EP94XX_GPIO,源代码位于路径”\Platform\EP94XX\SRC\DRIVERS\gpio”下面。
(2) 创建GPIO驱动文件夹并更改dir文件:
进入”\Platform\EP94XX\SRC\DRIVERS\”目录,创建一个名为”gpio”的文件夹,这个文件夹包含GPIO驱动。然后打开dirs文件,在末尾添加”gpio”。
(3) 开发GPIO驱动:
进入”\Platform\EP94XX\SRC\Drivers\gpio”并创建gpio.c文件,在文件中封装相应的流设备接口函数,如下:
GPI_Init(..)
GPI_DeInit(..)
GPI_Read(..)
GPI_Write(..)
…
可以在GPI_Read函数中读取GPIO的状态,在GPI_Write函数中设置GPIO的状态,当然也可以通过GPI_IoControl函数来实现。
然后在该路径下面创建makefile文件,并在里面包含下面一行就可以了:
!INCLUDE $(_MAKEENVROOT)\makefile.def
接下来创建模块导出文件gpio.def,具体内容如下:
LIBRARY GPIO_LIB
EXPORTS
DllEntry
GPI_Init
GPI_Deinit
GPI_Open
GPI_Close
GPI_Read
GPI_Write
GPI_Seek
GPI_IOControl
GPI_PowerDown
GPI_PowerUp
最后创建用于编译的sources文件,具体内容如下:
!ifndef BSP_EP94XX_GPIO
SKIPBUILD=1
!endif
TARGETNAME=gpio
RELEASETYPE=PLATFORM
TARGETTYPE=DYNLINK
TARGETLIBS= \
$(_SYSGENSDKROOT)\lib\$(_CPUINDPATH)\coredll.lib
DLLENTRY=DllEntry
SOURCES= gpio.c
(4) 添加GPIO驱动的注册表配置:
打开”\PLATFORM\EP94XX\files\”目录下的platform.reg文件,添加下面的配置:
IF BSP_EP94XX_GPIO
; Add these entries to your registry to enable the gpio device
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\GPIO]
prefix"="GPI"
"Dll"="gpio.dll"
"Order"=dword:1
ENDIF
(5) 添加驱动模块到NK中
打开”\PLATFORM\EP94XX\files\”目录下的platform.bib文件,添加如下内容:
IF BSP_EP94XX_GPIO
gpio.dll $(_FLATRELEASEDIR)\gpio.dll NK SHK
ENDIF
1. DWORD XXX_Init(LPCTSTR pContext, DWORD dwBusContext):
该函数用于初始化一个流设备驱动,在设备被加载的时候调用,调用成功后会返回一个句柄。 pContext:在Active注册表键路径下的一个字符串 dwBusContext:不常用,这里可以设为0 2. BOOL XXX_Deinit(DWORD hDeviceContext): 卸载一个设备驱动。 hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
3. DWORD XXX_Open(DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode): 打开一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
AccessCode:访问权限代码,一般是只读或者只写或者读写
ShareMode:共享模式,是否支持共享或者独享
4. BOOL XXX_Close(DWORD hOpenContext):
关闭一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
5. DWORD XXX_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count):
从设备上面读取数据。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:读取数据的字节数
6. DWORD XXX_Write(DWORD hOpenContext, LPCVOID pBuffer, DWORD Count):
写数据到设备上面。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:写入数据的字节数
7. DWORD XXX_Seek(DWORD hOpenContext, long Amount, WORD Type):
移动设备中的数据指针。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
Amount:移动的字节数
Type:FILE_BEGIN表示从头移动
FILE_CURRENT表示从当前位置移动
FILE_END表示从末尾往前移动
8. void XXX_PowerUp(DWORD hOpenContext):
打开设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
9. void XXX_PowerDown(DWORD hOpenContext):
关闭设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
10. BOOL XXX_IOControl(DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut):
设备IO控制操作函数。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
dwCode:操作码
pBufIn:输入Buffer
dwLenIn:输入Buffer的size
pBufOut:输出Buffer
dwLenOut:输出Buffer的size
pdwActualOut:实际输出的字节数
11. BOOL XXX_PreClose(DWORD hOpenContext):
标记一个正要关闭的句柄为无效,并唤醒所有正在休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
12. BOOL XXX_PreDeinit(DWORD hDeviceContext):
标记一个设备实例为无效,并唤醒所有休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
上面这些函数就是流设备驱动的所有接口函数,理解起来应该不难。下面介绍一个实际的流设备驱动的例子,是基于WinCE6.0的(和WinCE5.0比一些配置文件稍有不同)。这里介绍的是一个操作GPIO的流设备驱动并介绍具体添加流设备驱动的步骤:
(1) 更改BSP工程文件,添加GPIO驱动的选项:
在BSP目录下面的”CATALOG”文件夹下面找到”BspName.pbcxml”并用记事本打开,然后添加GPIO驱动的选项,首先找到<BSP>…</BSP>并在里面添加下面一行:
<BspItemId>Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx</BspItemId>
然后在< CatalogFile >…</CatalogFile>中添加下面的驱动描述:
<Item Id="Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx">
<Title>GPIO</Title>
<Description>GPIO Driver</Description>
<Type>BspSpecific</Type>
<Variable>BSP_EP94XX_GPIO</Variable>
<Location>Device Drivers</Location>
<SourceCode>
<Title>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Title>
<Path>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Path>
</SourceCode>
</Item>
上面实际上添加了GPIO驱动,环境变量为BSP_EP94XX_GPIO,源代码位于路径”\Platform\EP94XX\SRC\DRIVERS\gpio”下面。
(2) 创建GPIO驱动文件夹并更改dir文件:
进入”\Platform\EP94XX\SRC\DRIVERS\”目录,创建一个名为”gpio”的文件夹,这个文件夹包含GPIO驱动。然后打开dirs文件,在末尾添加”gpio”。
(3) 开发GPIO驱动:
进入”\Platform\EP94XX\SRC\Drivers\gpio”并创建gpio.c文件,在文件中封装相应的流设备接口函数,如下:
GPI_Init(..)
GPI_DeInit(..)
GPI_Read(..)
GPI_Write(..)
…
可以在GPI_Read函数中读取GPIO的状态,在GPI_Write函数中设置GPIO的状态,当然也可以通过GPI_IoControl函数来实现。
然后在该路径下面创建makefile文件,并在里面包含下面一行就可以了:
!INCLUDE $(_MAKEENVROOT)\makefile.def
接下来创建模块导出文件gpio.def,具体内容如下:
LIBRARY GPIO_LIB
EXPORTS
DllEntry
GPI_Init
GPI_Deinit
GPI_Open
GPI_Close
GPI_Read
GPI_Write
GPI_Seek
GPI_IOControl
GPI_PowerDown
GPI_PowerUp
最后创建用于编译的sources文件,具体内容如下:
!ifndef BSP_EP94XX_GPIO
SKIPBUILD=1
!endif
TARGETNAME=gpio
RELEASETYPE=PLATFORM
TARGETTYPE=DYNLINK
TARGETLIBS= \
$(_SYSGENSDKROOT)\lib\$(_CPUINDPATH)\coredll.lib
DLLENTRY=DllEntry
SOURCES= gpio.c
(4) 添加GPIO驱动的注册表配置:
打开”\PLATFORM\EP94XX\files\”目录下的platform.reg文件,添加下面的配置:
IF BSP_EP94XX_GPIO
; Add these entries to your registry to enable the gpio device
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\GPIO]
prefix"="GPI"
"Dll"="gpio.dll"
"Order"=dword:1
ENDIF
(5) 添加驱动模块到NK中
打开”\PLATFORM\EP94XX\files\”目录下的platform.bib文件,添加如下内容:
IF BSP_EP94XX_GPIO
gpio.dll $(_FLATRELEASEDIR)\gpio.dll NK SHK
ENDIF
#9
那你就找个可用的在修改吧
#10
楼上能不能给个可用的参考啊,谢谢啊!我昨天回去又适当修改了下,编译是通过了,但是还是看不到打印消息?
#11
问题解决了,和公司高手学习了下,按着FM的改了下,呵呵
#1
有sources文件没啊?
#2
DWORD_GPI_Write 中间是不是多了个下划线?
#3
第一,检查下你的写的流驱动的过程对不对,是否对注册表、*.bib文件 *.def 文件进行了修改,最好不要落下哪一步,那样的话会出现莫名的问题(当然有的可能无关你的问题)
第二,“error LNK2001: unresolved external symbol DllEntry“很有可能是你的导出的接口函数没有使用编辑器的扩展关键字_declspec(dllexport)字样
我写过的类似的代码:
#include "stdafx.h"
#include<windows.h>
//#include<C:\WINCE600\PUBLIC\COMMON\DDK\INC\ceddk.h>
#include<ceddk.h>
#pragma comment(lib,"CEDDK.lib")
#include<Ddkreg.h>
#include<dbgapi.h>
//#include<oalintr.h>
#include<Nkintr.h>
_declspec(dllexport) extern DWORD AUD_Init(LPCTSTR pContext,LPCVOID lpvBusContext);
_declspec(dllexport) extern BOOL AUD_Deinit(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Open(DWORD hDeviceContext,DWORD AccessCode,DWORD ShareMode);
_declspec(dllexport) extern BOOL AUD_Close(DWORD hOpenContext);
_declspec(dllexport) extern BOOL AUD_IOControl( DWORD hOpenContext,DWORD dwCode,//用于描述这次IOControl操作的语意
PBYTE pBufIn,//指向需要传送给驱动程序使用的数据
DWORD dwLenIn,//要传送给驱动程序使用的数据长度
PBYTE pBufOut,//指向驱动程序传给应用程序使用的数据
DWORD dwLenOut,//要传送给应用程序使用的数据长度
PDWORD pdwActualOut//返回实际处理数据的长度
);
_declspec(dllexport) extern void AUD_PowerUp(DWORD hDeviceContext);
_declspec(dllexport) extern void AUD_PowerDown(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Read(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Write(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Seek(DWORD hOpenContext,LONG Amount,WORD Type);
在函数使用前进行了一个声明,虽然比较笨,但是可行
哈哈 楼主 给分哦
第二,“error LNK2001: unresolved external symbol DllEntry“很有可能是你的导出的接口函数没有使用编辑器的扩展关键字_declspec(dllexport)字样
我写过的类似的代码:
#include "stdafx.h"
#include<windows.h>
//#include<C:\WINCE600\PUBLIC\COMMON\DDK\INC\ceddk.h>
#include<ceddk.h>
#pragma comment(lib,"CEDDK.lib")
#include<Ddkreg.h>
#include<dbgapi.h>
//#include<oalintr.h>
#include<Nkintr.h>
_declspec(dllexport) extern DWORD AUD_Init(LPCTSTR pContext,LPCVOID lpvBusContext);
_declspec(dllexport) extern BOOL AUD_Deinit(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Open(DWORD hDeviceContext,DWORD AccessCode,DWORD ShareMode);
_declspec(dllexport) extern BOOL AUD_Close(DWORD hOpenContext);
_declspec(dllexport) extern BOOL AUD_IOControl( DWORD hOpenContext,DWORD dwCode,//用于描述这次IOControl操作的语意
PBYTE pBufIn,//指向需要传送给驱动程序使用的数据
DWORD dwLenIn,//要传送给驱动程序使用的数据长度
PBYTE pBufOut,//指向驱动程序传给应用程序使用的数据
DWORD dwLenOut,//要传送给应用程序使用的数据长度
PDWORD pdwActualOut//返回实际处理数据的长度
);
_declspec(dllexport) extern void AUD_PowerUp(DWORD hDeviceContext);
_declspec(dllexport) extern void AUD_PowerDown(DWORD hDeviceContext);
_declspec(dllexport) extern DWORD AUD_Read(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Write(DWORD hOpenContext,PUCHAR pBuffer,ULONG Count);
_declspec(dllexport) extern DWORD AUD_Seek(DWORD hOpenContext,LONG Amount,WORD Type);
在函数使用前进行了一个声明,虽然比较笨,但是可行
哈哈 楼主 给分哦
#4
#5
非常感谢楼上的细心回答啊,呵呵!我早上在网上找了个比较细的步骤,按它的方法修改了一下,光是编译我加进去那个流驱动的目录不会出错,但是在把目标文件拷到板子的时候,串口打印信息没我驱动里那条信息。我就把目标文件全删了,整个重新编译了一遍,结果冒出一千多个错,这回光编译我那个就不行了,救命哦,55~~
#6
这种问题大多是重复定义,或者两个cpp用公共变量出现的问题。看看头文件和全局变量
#7
这个因为我也是在网上找的别人的帖子学的,应该没那个问题吧?
我现在编译似乎能通过了, 但是烧录到板子还是看不到我的打印信息啊?
我现在编译似乎能通过了, 但是烧录到板子还是看不到我的打印信息啊?
#8
除了流驱动程序的内容外,我几乎是照抄这个方法,那不会不适合我的工程吧?
1. DWORD XXX_Init(LPCTSTR pContext, DWORD dwBusContext):
该函数用于初始化一个流设备驱动,在设备被加载的时候调用,调用成功后会返回一个句柄。 pContext:在Active注册表键路径下的一个字符串 dwBusContext:不常用,这里可以设为0 2. BOOL XXX_Deinit(DWORD hDeviceContext): 卸载一个设备驱动。 hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
3. DWORD XXX_Open(DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode): 打开一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
AccessCode:访问权限代码,一般是只读或者只写或者读写
ShareMode:共享模式,是否支持共享或者独享
4. BOOL XXX_Close(DWORD hOpenContext):
关闭一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
5. DWORD XXX_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count):
从设备上面读取数据。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:读取数据的字节数
6. DWORD XXX_Write(DWORD hOpenContext, LPCVOID pBuffer, DWORD Count):
写数据到设备上面。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:写入数据的字节数
7. DWORD XXX_Seek(DWORD hOpenContext, long Amount, WORD Type):
移动设备中的数据指针。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
Amount:移动的字节数
Type:FILE_BEGIN表示从头移动
FILE_CURRENT表示从当前位置移动
FILE_END表示从末尾往前移动
8. void XXX_PowerUp(DWORD hOpenContext):
打开设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
9. void XXX_PowerDown(DWORD hOpenContext):
关闭设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
10. BOOL XXX_IOControl(DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut):
设备IO控制操作函数。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
dwCode:操作码
pBufIn:输入Buffer
dwLenIn:输入Buffer的size
pBufOut:输出Buffer
dwLenOut:输出Buffer的size
pdwActualOut:实际输出的字节数
11. BOOL XXX_PreClose(DWORD hOpenContext):
标记一个正要关闭的句柄为无效,并唤醒所有正在休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
12. BOOL XXX_PreDeinit(DWORD hDeviceContext):
标记一个设备实例为无效,并唤醒所有休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
上面这些函数就是流设备驱动的所有接口函数,理解起来应该不难。下面介绍一个实际的流设备驱动的例子,是基于WinCE6.0的(和WinCE5.0比一些配置文件稍有不同)。这里介绍的是一个操作GPIO的流设备驱动并介绍具体添加流设备驱动的步骤:
(1) 更改BSP工程文件,添加GPIO驱动的选项:
在BSP目录下面的”CATALOG”文件夹下面找到”BspName.pbcxml”并用记事本打开,然后添加GPIO驱动的选项,首先找到<BSP>…</BSP>并在里面添加下面一行:
<BspItemId>Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx</BspItemId>
然后在< CatalogFile >…</CatalogFile>中添加下面的驱动描述:
<Item Id="Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx">
<Title>GPIO</Title>
<Description>GPIO Driver</Description>
<Type>BspSpecific</Type>
<Variable>BSP_EP94XX_GPIO</Variable>
<Location>Device Drivers</Location>
<SourceCode>
<Title>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Title>
<Path>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Path>
</SourceCode>
</Item>
上面实际上添加了GPIO驱动,环境变量为BSP_EP94XX_GPIO,源代码位于路径”\Platform\EP94XX\SRC\DRIVERS\gpio”下面。
(2) 创建GPIO驱动文件夹并更改dir文件:
进入”\Platform\EP94XX\SRC\DRIVERS\”目录,创建一个名为”gpio”的文件夹,这个文件夹包含GPIO驱动。然后打开dirs文件,在末尾添加”gpio”。
(3) 开发GPIO驱动:
进入”\Platform\EP94XX\SRC\Drivers\gpio”并创建gpio.c文件,在文件中封装相应的流设备接口函数,如下:
GPI_Init(..)
GPI_DeInit(..)
GPI_Read(..)
GPI_Write(..)
…
可以在GPI_Read函数中读取GPIO的状态,在GPI_Write函数中设置GPIO的状态,当然也可以通过GPI_IoControl函数来实现。
然后在该路径下面创建makefile文件,并在里面包含下面一行就可以了:
!INCLUDE $(_MAKEENVROOT)\makefile.def
接下来创建模块导出文件gpio.def,具体内容如下:
LIBRARY GPIO_LIB
EXPORTS
DllEntry
GPI_Init
GPI_Deinit
GPI_Open
GPI_Close
GPI_Read
GPI_Write
GPI_Seek
GPI_IOControl
GPI_PowerDown
GPI_PowerUp
最后创建用于编译的sources文件,具体内容如下:
!ifndef BSP_EP94XX_GPIO
SKIPBUILD=1
!endif
TARGETNAME=gpio
RELEASETYPE=PLATFORM
TARGETTYPE=DYNLINK
TARGETLIBS= \
$(_SYSGENSDKROOT)\lib\$(_CPUINDPATH)\coredll.lib
DLLENTRY=DllEntry
SOURCES= gpio.c
(4) 添加GPIO驱动的注册表配置:
打开”\PLATFORM\EP94XX\files\”目录下的platform.reg文件,添加下面的配置:
IF BSP_EP94XX_GPIO
; Add these entries to your registry to enable the gpio device
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\GPIO]
prefix"="GPI"
"Dll"="gpio.dll"
"Order"=dword:1
ENDIF
(5) 添加驱动模块到NK中
打开”\PLATFORM\EP94XX\files\”目录下的platform.bib文件,添加如下内容:
IF BSP_EP94XX_GPIO
gpio.dll $(_FLATRELEASEDIR)\gpio.dll NK SHK
ENDIF
1. DWORD XXX_Init(LPCTSTR pContext, DWORD dwBusContext):
该函数用于初始化一个流设备驱动,在设备被加载的时候调用,调用成功后会返回一个句柄。 pContext:在Active注册表键路径下的一个字符串 dwBusContext:不常用,这里可以设为0 2. BOOL XXX_Deinit(DWORD hDeviceContext): 卸载一个设备驱动。 hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
3. DWORD XXX_Open(DWORD hDeviceContext, DWORD AccessCode, DWORD ShareMode): 打开一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
AccessCode:访问权限代码,一般是只读或者只写或者读写
ShareMode:共享模式,是否支持共享或者独享
4. BOOL XXX_Close(DWORD hOpenContext):
关闭一个设备。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
5. DWORD XXX_Read(DWORD hOpenContext, LPVOID pBuffer, DWORD Count):
从设备上面读取数据。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:读取数据的字节数
6. DWORD XXX_Write(DWORD hOpenContext, LPCVOID pBuffer, DWORD Count):
写数据到设备上面。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
pBuffer:存放数据的Buffer
Count:写入数据的字节数
7. DWORD XXX_Seek(DWORD hOpenContext, long Amount, WORD Type):
移动设备中的数据指针。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
Amount:移动的字节数
Type:FILE_BEGIN表示从头移动
FILE_CURRENT表示从当前位置移动
FILE_END表示从末尾往前移动
8. void XXX_PowerUp(DWORD hOpenContext):
打开设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
9. void XXX_PowerDown(DWORD hOpenContext):
关闭设备电源。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
10. BOOL XXX_IOControl(DWORD hOpenContext, DWORD dwCode, PBYTE pBufIn, DWORD dwLenIn, PBYTE pBufOut, DWORD dwLenOut, PDWORD pdwActualOut):
设备IO控制操作函数。
hDeviceContext:设备驱动的句柄,在XXX_Open调用时返回的
dwCode:操作码
pBufIn:输入Buffer
dwLenIn:输入Buffer的size
pBufOut:输出Buffer
dwLenOut:输出Buffer的size
pdwActualOut:实际输出的字节数
11. BOOL XXX_PreClose(DWORD hOpenContext):
标记一个正要关闭的句柄为无效,并唤醒所有正在休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
12. BOOL XXX_PreDeinit(DWORD hDeviceContext):
标记一个设备实例为无效,并唤醒所有休眠的线程
hDeviceContext:设备驱动的句柄,在XXX_Init调用时返回的
上面这些函数就是流设备驱动的所有接口函数,理解起来应该不难。下面介绍一个实际的流设备驱动的例子,是基于WinCE6.0的(和WinCE5.0比一些配置文件稍有不同)。这里介绍的是一个操作GPIO的流设备驱动并介绍具体添加流设备驱动的步骤:
(1) 更改BSP工程文件,添加GPIO驱动的选项:
在BSP目录下面的”CATALOG”文件夹下面找到”BspName.pbcxml”并用记事本打开,然后添加GPIO驱动的选项,首先找到<BSP>…</BSP>并在里面添加下面一行:
<BspItemId>Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx</BspItemId>
然后在< CatalogFile >…</CatalogFile>中添加下面的驱动描述:
<Item Id="Item:Cirrus Logic:bsp_ep94xx_gpio_ep9407_EP94xx">
<Title>GPIO</Title>
<Description>GPIO Driver</Description>
<Type>BspSpecific</Type>
<Variable>BSP_EP94XX_GPIO</Variable>
<Location>Device Drivers</Location>
<SourceCode>
<Title>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Title>
<Path>$(_WINCEROOT)\PLATFORM\EP94XX\SRC\DRIVERS\gpio</Path>
</SourceCode>
</Item>
上面实际上添加了GPIO驱动,环境变量为BSP_EP94XX_GPIO,源代码位于路径”\Platform\EP94XX\SRC\DRIVERS\gpio”下面。
(2) 创建GPIO驱动文件夹并更改dir文件:
进入”\Platform\EP94XX\SRC\DRIVERS\”目录,创建一个名为”gpio”的文件夹,这个文件夹包含GPIO驱动。然后打开dirs文件,在末尾添加”gpio”。
(3) 开发GPIO驱动:
进入”\Platform\EP94XX\SRC\Drivers\gpio”并创建gpio.c文件,在文件中封装相应的流设备接口函数,如下:
GPI_Init(..)
GPI_DeInit(..)
GPI_Read(..)
GPI_Write(..)
…
可以在GPI_Read函数中读取GPIO的状态,在GPI_Write函数中设置GPIO的状态,当然也可以通过GPI_IoControl函数来实现。
然后在该路径下面创建makefile文件,并在里面包含下面一行就可以了:
!INCLUDE $(_MAKEENVROOT)\makefile.def
接下来创建模块导出文件gpio.def,具体内容如下:
LIBRARY GPIO_LIB
EXPORTS
DllEntry
GPI_Init
GPI_Deinit
GPI_Open
GPI_Close
GPI_Read
GPI_Write
GPI_Seek
GPI_IOControl
GPI_PowerDown
GPI_PowerUp
最后创建用于编译的sources文件,具体内容如下:
!ifndef BSP_EP94XX_GPIO
SKIPBUILD=1
!endif
TARGETNAME=gpio
RELEASETYPE=PLATFORM
TARGETTYPE=DYNLINK
TARGETLIBS= \
$(_SYSGENSDKROOT)\lib\$(_CPUINDPATH)\coredll.lib
DLLENTRY=DllEntry
SOURCES= gpio.c
(4) 添加GPIO驱动的注册表配置:
打开”\PLATFORM\EP94XX\files\”目录下的platform.reg文件,添加下面的配置:
IF BSP_EP94XX_GPIO
; Add these entries to your registry to enable the gpio device
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\GPIO]
prefix"="GPI"
"Dll"="gpio.dll"
"Order"=dword:1
ENDIF
(5) 添加驱动模块到NK中
打开”\PLATFORM\EP94XX\files\”目录下的platform.bib文件,添加如下内容:
IF BSP_EP94XX_GPIO
gpio.dll $(_FLATRELEASEDIR)\gpio.dll NK SHK
ENDIF
#9
那你就找个可用的在修改吧
#10
楼上能不能给个可用的参考啊,谢谢啊!我昨天回去又适当修改了下,编译是通过了,但是还是看不到打印消息?
#11
问题解决了,和公司高手学习了下,按着FM的改了下,呵呵