0

我知道这个问题之前在这里这里被问过,不幸的是,提供的答案无法解决我的问题。我在 C# 应用程序中使用 Stasm ( http://www.milbo.users.sonic.net/stasm/index.html ) 库。下面是我调用“AsmSearchDll”函数的代码。

        [DllImport(@"stasm\stasm_dll.dll", CallingConvention = CallingConvention.Cdecl)]
    internal static extern void AsmSearchDll
    (
        [Out] out Int32 pnlandmarks,
        [Out] out Int32[] landmarks,
        [In, MarshalAs(UnmanagedType.LPStr)] String image_name,
        [In, MarshalAs(UnmanagedType.LPStr)] String image_data,
        [In] Int32 width,
        [In] Int32 height,
        [In] Int32 is_color,
        [In, MarshalAs(UnmanagedType.LPStr)] String conf_file0,
        [In, MarshalAs(UnmanagedType.LPStr)] String conf_file1
    );

    public void SearchFacialFeatures()
    {
        string image_name = "image-5.jpg"; // imagePath;
        var image = new Image<Bgr, byte>(image_name).Convert<Gray, byte>();

        int pnlandmarks = 0;
        var landmarks = new int[500];

        var imageData = Marshal.PtrToStringAnsi(image.MIplImage.imageData);
        int imgWidth = image.Width;
        int imgHeight = image.Height;
        int is_color = 1;
        string confile_file0 = Path.GetFullPath(@"data\mu-68-1d.conf");
        string config_file1 = Path.GetFullPath(@"data\mu-76-2d.conf");
        string sDataDir = @"\stasm\data";


        AsmSearchDll(out pnlandmarks, out landmarks, image_name, imageData, imgWidth, imgHeight, 1, null, null);

        MessageBox.Show(image_name);
    }

问题是应用程序在到达这条线时停止

AsmSearchDll(out pnlandmarks, out landmarks, image_name, imageData, imgWidth, imgHeight, 0, null, null);

最初,每当调用 AsmSearchDll 函数时,应用程序就会退出,经过多次处理代码后,该函数停止了。现在应用程序出现了,但从未处理 AsmSearchDll 函数。我可以说出来,因为我用 VS 浏览了代码。永远无法到达函数下方的消息框。

我有一种强烈的感觉,该函数正在引发内部错误。对我来说不幸的是,这是我与 Interop/DllImport 的第一次交易。

我的问题是,我做错了什么,我该如何解决这个问题?我已经坚持了一天。

编辑:为非托管函数添加了代码

非托管函数的签名

void AsmSearchDll (
int *pnlandmarks,          // out: number of landmarks, 0 if can't get landmarks
int landmarks[],           // out: the landmarks, caller must allocate
const char image_name[],   // in: used in internal error messages, if necessary
const char image_data[],   // in: image data, 3 bytes per pixel if is_color
const int width,           // in: the width of the image
const int height,          // in: the height of the image
const int is_color,        // in: 1 if RGB image, 0 if grayscale
const char conf_file0[],   // in: 1st config filename, NULL for default
const char conf_file1[])   // in: 2nd config filename, NULL for default, "" if none

调用函数的 C++ 示例

    const char *image_name = "../data/test-image.jpg";

IplImage *img = cvLoadImage(image_name, CV_LOAD_IMAGE_COLOR);
if(img == NULL) {
    printf("Error: Cannot open %s\n", image_name);
    return -1;
}
// sanity checks (AsmSearchDll assumes imageData is vector of b,r,g bytes)

if(img->nChannels != 3 || img->depth != IPL_DEPTH_8U ||
        img->origin != 0 || img->widthStep != 3 * img->width) {
    printf("Error: %s is an unrecognized image type\n", image_name);
    return -1;
}

// locate the facial landmarks with stasm

int nlandmarks;
int landmarks[500]; // space for x,y coords of up to 250 landmarks
AsmSearchDll(&nlandmarks, landmarks,
             image_name, img->imageData, img->width, img->height,
             1 /* is_color */, NULL /* conf_file0 */, NULL /* conf_file1 */);

谢谢你的帮助。

4

2 回答 2

2

使用这个定义:

[DllImport(@"stasm\stasm_dll.dll", CallingConvention = CallingConvention.Cdecl, CharSet=CharSet.Ansi)]
internal static extern void AsmSearchDll
(
    [Out] out Int32 pnlandmarks,
    [Out] Int32[] landmarks,       // <-- the `out` keyword is removed
    [In, MarshalAs(UnmanagedType.LPStr)] String image_name,
    [In] IntPtr image_data,        // <-- should not be passed as string
    [In] Int32 width,
    [In] Int32 height,
    [In] Int32 is_color,
    [In, MarshalAs(UnmanagedType.LPStr)] String conf_file0,
    [In, MarshalAs(UnmanagedType.LPStr)] String conf_file1
);

我猜主要问题是冗余out关键字 on landmarks,这导致参数作为指向数组的指针传递,这意味着指向 int 的指针的指针。您需要的是指向 int的指针。

同样将字节数组作为字符串传递可能会破坏字节(image_data参数)的内容。现在不需要使用Marshal.PtrToStringAnsi函数了。直接传递image.MIplImage.imageData给函数。

让我知道它是否有效。我无法测试它。

于 2013-04-24T11:42:05.510 回答
0

您的问题是您的程序正在尝试在 ../data/%filename% 中查找某些文件,这会导致查看包含 Debug 和 Release 文件夹的 bin 文件夹,并且您的工作目录实际上是 Debug 或 Release,具体取决于您的配置. 这就是程序返回代码 0x01 和错误的原因。一个简单的解决方案是将包含所有必需文件的数据文件夹复制到 Debug 或 Release 文件夹的父文件夹,这应该是一个快速的解决方案,或者您可以考虑修复这是通过修改 dll 查找这些文件夹的文件夹路径来实现的。

发生此问题是因为路径是硬编码的。

于 2015-07-03T04:04:50.690 回答