在我决定在这里发布之前,我一直在阅读这个问题。我有一个C dll,我需要访问C#程序内部。 DLL的代码非常简单。因为它只不过是生物识别驱动程序的一个钩子(它是C,因为它必须包含来自驱动程序的lib和头文件才能使代码工作)。这是dll的代码:
#include <stdio.h>
#include <windows.h>
#include <DPCN_TM.h>
__declspec(dllexport) unsigned char * ConvertPlatinumTemplateToDpTemplate(const unsigned char* inputData);
HRESULT Convert(
const unsigned char* inputData,
const size_t size,
DPCN_DATA_TYPE inputDataType,
DPCN_DATA_TYPE outputDataType,
DPCN_PURPOSE purpose,
unsigned char** ppbOutputData,
const void * outputParameters,
size_t * pcbData)
{
HRESULT hr = 0;
size_t cbData = 0;
if (FAILED(hr = DPCNConvertFingerprintData(inputData, size, inputDataType, purpose, NULL, outputDataType, outputParameters, NULL, &cbData))) {
return hr;
}
if (!(*ppbOutputData = (unsigned char *)malloc(cbData))) {
return DPCN_ERR_NO_MEMORY;
}
hr = DPCNConvertFingerprintData(inputData, size, inputDataType, purpose, NULL, outputDataType, outputParameters, *ppbOutputData, &cbData);
*pcbData = cbData;
return hr;
}
unsigned char * ConvertPlatinumTemplateToDpTemplate(const unsigned char* inputData) {
HRESULT hr = 0;
const size_t inputSize = sizeof(inputData);
DPCN_DATA_TYPE inputDataType = DPCN_DT_DP_TEMPLATE;
DPCN_DATA_TYPE outputDataType = DPCN_DT_DP_PLATINUM_TEMPLATE;
unsigned char *pbOutputData = NULL;
size_t cbData = 0;
hr = Convert(inputData, inputSize, inputDataType, outputDataType, DPCN_PURPOSE_IDENTIFICATION, &pbOutputData, NULL, &cbData);
return pbOutputData;
}
正如您所看到的,DLL的内容非常简单。从代码中你可以看到我需要在C#程序中访问这个函数。
unsigned char * ConvertPlatinumTemplateToDpTemplate(const unsigned char* inputData);
现在在我的C#代码中我已经完成了这个:
[DllImport(@"path_to_dll\DPFPTemplateConvert.dll")]
public extern byte[] ConvertPlatinumTemplateToDpTemplate(byte[] inputData);
当我调用该函数时,我最终收到此错误:
A first chance exception of type 'System.Runtime.InteropServices.MarshalDirectiveException' occurred in DLLImportTest.exe
Additional information: Cannot marshal 'return value': Invalid managed/unmanaged type combination.
我做错了什么?
答案 0 :(得分:1)
无符号字符*无法转换为.NET字节数组,原因很简单:该数组的长度应该是多少?
即便如此,如果指针指向由该函数分配的内存,那么将指针从函数中传出是一个坏主意。谁会释放这段记忆?
您应该让.NET端为结果分配byte [],并将其传递给函数。
如果.NET方面事先不知道分配的数组需要多大,请按照此处的说明使用回调:http://blog.getpaint.net/2012/04/30/marshaling-native-arrays-back-as-managed-arrays-without-copying/