Не удалось передать System.Drawing.Bitmap в C++ DLL

Я пишу DLL на С++ для поиска субкартинки в растровом изображении. Когда часть C++ выполняется, HBITMAP недействителен.

Вот мой код:

C#:

[System.Runtime.InteropServices.DllImport("FindSubPicture.dll", EntryPoint = "FindSubPictures", CallingConvention = CallingConvention.Cdecl)]
    private static extern System.IntPtr FindSubPicturesFct(System.IntPtr mImg, System.IntPtr sImg, int* nMatches);
    public static System.Collections.Generic.List<TPoint> FindSubPictures(System.Drawing.Bitmap mImg, System.Drawing.Bitmap sImg)
    {
        TPoint* PStack = null;
        int nMatches = 0;
        System.Collections.Generic.List<TPoint> MyList = new System.Collections.Generic.List<TPoint>();
        MyList.Clear();

        PStack = (TPoint*)FindSubPicturesFct(mImg.GetHbitmap(), sImg.GetHbitmap(), &nMatches);
        if(PStack == null) { return MyList;}

        for (int i = 0; i < nMatches; i++) { MyList.Add(new TPoint(PStack[i].x[0], PStack[i].x[1])); }
        try
        {
            System.Runtime.InteropServices.Marshal.FreeHGlobal((System.IntPtr)PStack);
        }catch(System.Exception ex) { System.Console.WriteLine(ex.Message); }

        return MyList;
    }

C++:

#include "FindSubPictures.h"
#include <stdio.h>
#include <stdlib.h>
#include <Windows.h>
#include <omp.h>

struct TPoint
{
    int x[2];

    TPoint(int fx, int fy) {
        x[0] = fx;
        x[1] = fy;
    }

    void Reset() { x[0] = 0; x[1] = 0; }
    void Set(int fx, int fy) { x[0] = fx; x[1] = fy; }
};

extern "C" __declspec(dllexport) TPoint* FindSubPictures(HBITMAP mImg, HBITMAP sImg, int* nMatches) {
    int mImgWidth = -1, mImgHeight = -1, sImgWidth = -1, sImgHeight = -1;
    TPoint* MyList = nullptr;

    if (mImg == nullptr || sImg == nullptr || nMatches == nullptr) { return nullptr; }


    return MyList;
}

Что ж, функция библиотеки C++ до сих пор ничего не делает. Это потому, что HBITMAP недействителен. В C# допустимо значение System.Drawing.Bitmap.

В точке останова

Когда я ввожу "mIng.", автозаполнение недоступно.

Я что-то упустил?


person Jan021981    schedule 11.09.2017    source источник
comment
Где количество совпадений в списке параметров?   -  person jdweng    schedule 11.09.2017
comment
Что ты имеешь в виду?   -  person Jan021981    schedule 11.09.2017
comment
Виноват. В вызывающей функции в С# отсутствовали совпадения, которые не нужны. : public static System.Collections.Generic.List‹TPoint› FindSubPictures(System.Drawing.Bitmap mImg, System.Drawing.Bitmap sImg)   -  person jdweng    schedule 11.09.2017


Ответы (2)


Получение растрового изображения не означает, что вы можете получить к нему доступ как к полю данных. Сначала вам придется заблокировать растровые данные в памяти. См. также ответы в этой теме здесь.

Bitmap.LockBits

Код будет выглядеть следующим образом:

Rectangle rect = new Rectangle(0, 0, mImg.Width, mImg.Height);
System.Drawing.Imaging.BitmapData bmpData =
    mImg.LockBits(rect, System.Drawing.Imaging.ImageLockMode.Read,
    mImg.PixelFormat);

// Get the address of the first line.
IntPtr ptr2mImg = bmpData.Scan0;
person yacc    schedule 11.09.2017
comment
Хм, теперь я не уверен, почему GetHbitmap() не сработало. Но, может быть, это натолкнет вас на мысль. Комментарий, если хотите. - person yacc; 11.09.2017

Хорошо, у меня есть решение.

HBITMAP - это всего лишь дескриптор "void *"

Сначала вам нужно получить полный объект с помощью функции GetObject

Код:

extern "C" __declspec(dllexport) TPoint* FindSubPictures(HBITMAP mImg, HBITMAP sImg, int* nMatches) {
    int mImgWidth = -1, mImgHeight = -1, sImgWidth = -1, sImgHeight = -1;
    TPoint* MyList = nullptr;

    if (mImg== nullptr || sImg== nullptr || nMatches == nullptr) { return nullptr; }

    BITMAP mBMP, sBMP;
    GetObject(mImg, sizeof(BITMAP), &mBMP);
    GetObject(sImg, sizeof(BITMAP), &sBMP);
    /*Now, BITMAP mBMP and sBMP are valid*/

    return MyList;
}
person Jan021981    schedule 11.09.2017