Вызов C DLL из C # - PullRequest
       22

Вызов C DLL из C #

7 голосов
/ 14 апреля 2011

Я пытаюсь вызвать C DLL из C #, но у меня нет никакой радости. Документация по DLL предоставляет пример функции для VB, которая выглядит так:

Declare Function TransGeogPt Lib "c:\DLLS\GDAit.dll" (ByVal sGridFile As String, ByVal lDirection As
Long, ByVal dLat As Double, ByVal dLong As Double, pdLatNew As Double, pdLongNew As Double,
pdLatAcc As Double, pdLongAcc As Double) As Long

Declare Function TransProjPt Lib "c:\DLLS\GDAit.dll" (ByVal sGridFile As String, ByVal lDirection As
Long, ByVal dLat As Double, ByVal dLong As Double, ByVal lZone As Long, pdLatNew As Double,
pdLongNew As Double, pdLatAcc As Double, pdLongAcc As Double) As Long

Поэтому я сделал следующее:

public class GDAIt
{
    public static string gridFileName = @"C:\Nat84.gsb";

    [DllImport(@"c:\GDAit.dll")]
    public static extern long TransGeogPt(string sGridFile, long lDirection, double dLat, double dLong, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc);

    [DllImport(@"c:\GDAit.dll")]
    public static extern long TransProjPt(string sGridFile, long lDirection, double dLat, double dLong, long lZone, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc);

    public static long CallTransGeogPt(string sGridFile, long lDirection, double dLat, double dLong, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc)
    {
        return TransGeogPt(sGridFile, lDirection, dLat, dLong, ref pdLatNew, ref pdLongNew, ref pdLatAcc, ref pdLongAcc);
    }

    public static long CallTransProjPt(string sGridFile, long lDirection, double dLat, double dLong, long lZone, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc)
    {
        return TransProjPt(sGridFile, lDirection, dLat, dLong, lZone, ref pdLatNew, ref pdLongNew, ref pdLatAcc, ref pdLongAcc);
    }


    public static void Process()
    {
        double latitude = 0.0;
        double longitude = 0.0; 
        double latAcc = 0.0; 
        double longAcc = 0.0;

        long result = 0;
        result = CallTransProjPt(gridFileName,
                                        1,
                                        394980,
                                        7619799,
                                        51,
                                        ref latitude,
                                        ref longitude,
                                        ref latAcc,
                                        ref longAcc);
        Console.WriteLine(string.Format("Result was {0}, Lat: {1}, Long: {2}", result, latitude, longitude));

        int error = Marshal.GetLastWin32Error();

        Console.WriteLine(string.Format("Last error recieved was {0}", error));

    }

}

Мне все еще не очень повезло, и я пробовал другие параметры в настройке DLLImport, такие как; SetLastError = true, CharSet = CharSet.Auto, CallingConvention = CallingConvention.Cdecl)

Вывод, полученный из кода:

Result was 4690529317195612196, Lat: 0, Long: 0
Last error recieved was 6

Если я правильно смотрю информацию об ошибках Win32, я думаю, это относится к; ERROR_INVALID_HANDLE Неверный дескриптор.
6 (0x6)

Полагаю, есть проблема с передачей имени файла в виде строки или способа, которым я передаю удваивается по ref? Тем не менее, я действительно не знаю, и я в растерянности относительно того, как исследовать проблему дальше.

Любые идеи очень ценятся.

Спасибо.

Ответы [ 4 ]

6 голосов
/ 18 апреля 2011

Я нашел причину моих неудачных попыток, используя инструмент под названием; Microsoft (R) P / Invoke Interop Assistant в соответствии с предложением ответа в этой теме .

Я использовал этот инструмент для ввода некоторых прототипов функций C и получения его для создания необходимого прототипа C # от моего имени. Прототип C выглядел следующим образом:

long __stdcall TransProjPt(LPSTR psGridFile, long lDirection, double dEasting, double
dNorthing, long lZone, double* pdEastNew, double* pdNorthNew, double* pdEastAcc,
double* pdNorthAcc) 

При вводе этого в инструмент Interop assistant он показал, что вместо использования long (как я делал в моем первоначальном вопросе) их следует объявить как int. Это привело к следующему выводу, который означал, что мой код выше теперь работает, как я и надеялся. Yay.

    /// Return Type: int
    ///psGridFile: LPSTR->CHAR*
    ///lDirection: int
    ///dEasting: double
    ///dNorthing: double
    ///lZone: int
    ///pdEastNew: double*
    ///pdNorthNew: double*
    ///pdEastAcc: double*
    ///pdNorthAcc: double*
    [System.Runtime.InteropServices.DllImportAttribute("<Unknown>", EntryPoint="TransProjPt", CallingConvention=System.Runtime.InteropServices.CallingConvention.StdCall)]
public static extern  int TransProjPt([System.Runtime.InteropServices.MarshalAsAttribute(System.Runtime.InteropServices.UnmanagedType.LPStr)] System.Text.StringBuilder psGridFile, int lDirection, double dEasting, double dNorthing, int lZone, ref double pdEastNew, ref double pdNorthNew, ref double pdEastAcc, ref double pdNorthAcc) ;

Спасибо за помощь всем.

3 голосов
/ 14 апреля 2011

Вы можете определить свои подписи c #, используя атрибуты маршаллинга для ваших строковых параметров.

[DllImport(@"c:\GDAit.dll")]
public static extern long TransGeogPt([MarshalAs(UnmanagedType.LPStr)] string sGridFile, long lDirection, double dLat, double dLong, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc);

[DllImport(@"c:\GDAit.dll")]
public static extern long TransProjPt([MarshalAs(UnmanagedType.LPStr)] string sGridFile, long lDirection, double dLat, double dLong, long lZone, ref double pdLatNew, ref double pdLongNew, ref double pdLatAcc, ref double pdLongAcc);

Я также прокомментирую ответ Марка Совула и скажу: попытайтесь звонить с помощью StdCall вместо Cdecl.

Кроме того, в качестве меры предосторожности, я бы, вероятно, перепроверил, чтобы убедиться, что компилятор настроен на компиляцию кода x86, в случае, если он компилируется для 64-битной версии.

2 голосов
/ 14 апреля 2011

Попробуйте изменить string sGridFile на StringBuilder sGridFile

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

1 голос
/ 14 апреля 2011

Похоже, вы не единственный, кто столкнулся с этой проблемой, вы пробовали StdCall? Это сработало для этого парня: http://www.mail-archive.com/delphi@ns3.123.co.nz/msg01227.html

...