我正在尝试在MATLAB中使用第三方外部DLL(来自usbmicro),但它一直在崩溃MATLAB。这来自指示C程序中函数调用语法的文档:
int USBm_About( char *about );
我尝试了这个MATLAB脚本(是的,它非常kludgy,我是一个MATLAB菜鸟):
>> loadlibrary('USBm.dll','USBmAPI.h')
>> libfunctions('USBm')
>> s='sssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss';
>> st=strcat(s,s,s,s);
>> vp = libpointer('voidPtr',[int8(st) 0]);
>> result=calllib('USBm','USBm_About',vp)
和这一个:
>> loadlibrary('USBm.dll','USBmAPI.h')
>> libfunctions('USBm')
>> s='sssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss';
>> st=strcat(s,s,s,s);
>> vp=libpointer('cstring',st);
>> result=calllib('USBm','USBm_About',vp)
在这两种情况下,calllib()
调用都会导致MATLAB因分段错误而崩溃。
MATLAB的版本是7.10;操作系统是Windows Vista。
这是libfunctionsview USBm的截图:
这是头文件:
#ifndef FILE_USBmAPI_h
#define FILE_USBmAPI_h
// Prototypes for this DLL.
// These are the API functions available to the .dll user.
// Discovery routine
extern "C" __declspec(dllexport) int USBm_FindDevices(void);
// Return info about devices
extern "C" __declspec(dllexport) int USBm_NumberOfDevices(void);
extern "C" __declspec(dllexport) int USBm_DeviceValid(unsigned char);
extern "C" __declspec(dllexport) int USBm_DeviceVID(unsigned char);
extern "C" __declspec(dllexport) int USBm_DevicePID(unsigned char);
extern "C" __declspec(dllexport) int USBm_DeviceDID(unsigned char);
extern "C" __declspec(dllexport) int USBm_DeviceFirmwareVer(unsigned char);
extern "C" __declspec(dllexport) int USBm_DeviceMfr(unsigned char, char *);
extern "C" __declspec(dllexport) int USBm_DeviceProd(unsigned char, char *);
extern "C" __declspec(dllexport) int USBm_DeviceSer(unsigned char, char *);
// General USBmicro U4xx device access
extern "C" __declspec(dllexport) int USBm_ReadDevice(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_SetReadTimeout(unsigned int);
extern "C" __declspec(dllexport) int USBm_WriteDevice(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_CloseDevice(unsigned char);
// DLL string info access
extern "C" __declspec(dllexport) int USBm_RecentError(char *);
extern "C" __declspec(dllexport) int USBm_ClearRecentError(void);
extern "C" __declspec(dllexport) int USBm_DebugString(char *);
extern "C" __declspec(dllexport) int USBm_Copyright(char *);
extern "C" __declspec(dllexport) int USBm_About(char *);
extern "C" __declspec(dllexport) int USBm_Version(char *);
// General U4x1 device functions
// -----------------------------
// Port initialization
extern "C" __declspec(dllexport) int USBm_InitPorts(unsigned char);
extern "C" __declspec(dllexport) int USBm_InitPortsU401(unsigned char);
extern "C" __declspec(dllexport) int USBm_InitPortsU421(unsigned char);
extern "C" __declspec(dllexport) int USBm_InitPortsU451(unsigned char);
// Port/bit reading and writing
extern "C" __declspec(dllexport) int USBm_WriteA(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_WriteB(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_WriteABit(unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_WriteBBit(unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_ReadA(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_ReadB(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_SetBit(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_ResetBit(unsigned char, unsigned char);
// Port direction
extern "C" __declspec(dllexport) int USBm_DirectionA(unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionAOut(unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionAIn(unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionAInPullup(unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionB(unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionBOut(unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionBIn(unsigned char);
extern "C" __declspec(dllexport) int USBm_DirectionBInPullup(unsigned char);
// Strobbing a byte of data
extern "C" __declspec(dllexport) int USBm_StrobeWrite(unsigned char, unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_StrobeRead(unsigned char, unsigned char *, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_StrobeWrite2(unsigned char, unsigned char, unsigned char, unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_StrobeRead2(unsigned char, unsigned char *, unsigned char, unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_StrobeWrites(unsigned char, unsigned char *, unsigned char *);
extern "C" __declspec(dllexport) int USBm_StrobeReads(unsigned char, unsigned char *, unsigned char *);
// Reading pin-change latches
extern "C" __declspec(dllexport) int USBm_ReadLatches(unsigned char, unsigned char *);
// LCD routines
extern "C" __declspec(dllexport) int USBm_InitLCD(unsigned char, unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_LCDCmd(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_LCDData(unsigned char, unsigned char);
// SPI routines
extern "C" __declspec(dllexport) int USBm_InitSPI(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_SPIMaster(unsigned char, unsigned char *, unsigned char *);
extern "C" __declspec(dllexport) int USBm_SPISlaveWrite(unsigned char, unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_SPISlaveRead(unsigned char, unsigned char *, unsigned char *);
// 2-wire routines
extern "C" __declspec(dllexport) int USBm_Wire2Control(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_Wire2Data(unsigned char, unsigned char *);
// Stepper routine
extern "C" __declspec(dllexport) int USBm_Stepper(unsigned char, unsigned char, unsigned char, unsigned char, unsigned char, unsigned char, unsigned char, unsigned char);
// 1-wire routines
extern "C" __declspec(dllexport) int USBm_Reset1Wire(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_Write1Wire(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_Read1Wire(unsigned char, unsigned char *);
extern "C" __declspec(dllexport) int USBm_Write1WireBit(unsigned char, unsigned char);
extern "C" __declspec(dllexport) int USBm_Read1WireBit(unsigned char, unsigned char *);
#endif // multiple inclusion prevention
// End of file
//---------------------------------------------------------------------------
我尝试更改此行:
extern "C" __declspec(dllexport) int USBm_About(char *);
到此:
extern "C" __declspec(dllimport) int USBm_About(char *);
在头文件中,然后重新启动MATLAB。我再次运行我的代码,MATLAB仍然崩溃。
答案 0 :(得分:2)
我从USBmicro网站下载了DLL,并通过调用use it尝试loadlibrary()
。不幸的是,正如你所说的那样,这使我的MATLAB会话崩溃了。
在一些research之后,我发现这个DLL暴露其函数的方式与MATLAB期望的调用约定(cdecl vs. stdcall)不兼容。
以下是我修复头文件的方法:
#ifndef FILE_USBmAPI_h
#define FILE_USBmAPI_h
#ifdef __cplusplus
extern "C" {
#endif
// ...
int __stdcall USBm_About(char *);
int __stdcall USBm_Version(char *);
// ...
#ifdef __cplusplus
}
#endif
#endif
现在您可以调用任何导出的函数。例如:
%# load library and see exported functions signatures
if ~libisloaded('USBm')
loadlibrary('USBm.dll','USBmAPI.h')
libfunctions('USBm','-full')
end
%# call the function: `int USBm_About(char *)`
str = repmat(' ',1,100); %# allocate buffer
pStr = libpointer('stringPtr',str); %# pointer to string
[num str2] = calllib('USBm','USBm_About',pStr)
clear pStr
%# unload library
unloadlibrary USBm
请注意MATLAB为我们提供的函数的签名:
[int32, cstring] USBm_About(cstring)
有趣的是,它为此函数提供了额外的输出参数。原因是MATLAB并不真正支持通过引用传递,尽管您可以创建与C指针兼容的MATLAB参数。
因此,如果C函数返回通过引用传递的输入参数中的数据,MATLAB将创建其他输出参数以返回这些值(以及以Ptr
或PtrPtr
结尾的任何输入参数)。
现在虽然输入参数pStr
类似于指向char类型的指针,但它不是真正的,因为它不包含MATLAB字符数组str
的地址。当函数执行时,它会返回正确的结果,但不会修改str
中的值(在这种情况下pStr
也不会这样做。)
我得到的输出是:
>> num
num =
0
>> str2
str2 =
USBm.dll by USBmicro L.L.C. (www.usbmicro.com). Supports: U401, U421
答案 1 :(得分:0)
这可能是您的包含文件需要的标准问题
extern "C" __declspec(dllexport)
当你构建.dll但Matlab(或任何使用你的.dll的项目)需要:
extern "C" __declspec(dllimport)
当你包括它时。
这就是人们通常拥有以下宏的原因:
#ifdef USBMAPIEXPORTS
#define USBMAPIDECLSPEC __declspec(dllexport)
#else
#define USBMAPIDECLSPEC __declspec(dllimport)
#endif
然后在你的标题中写下你的函数:
extern "C" USBMAPIDECLSPEC int USBm_FindDevices(void);
并且在构建.dll时需要定义USBMAPIEXPORTS
(但在Matlab中使用它时则不需要。)
请注意,这不是Matlab的问题。 This is the standard way to use a .dll. 你试过这个吗?