Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   C++ DLL in Delphi unter Linux verwenden (https://www.delphipraxis.net/199398-c-dll-delphi-unter-linux-verwenden.html)

Harry Stahl 20. Jan 2019 17:50

Delphi-Version: 5

C++ DLL in Delphi unter Linux verwenden
 
Kann man eine DLL, die zur Verwendung von C++ Builder gedacht ist, auch unter Delphi verwenden, wenn ja wie?

Ich würde das mal grundsätzlich annehmen. Aber bevor ich da was versuche, frage ich zur Sicherheit mal nach.

[ Möchte vorausschicken, dass ich C-Code ein wenig lesen kann, aber ansonsten mit C-Programmierung Null-Erfahrung habe ]

Ich habe halt hier eine DLL (libQPLLinux1612-86_64.so) für Linux, die für den Aufruf von C++-Programmen gedacht ist (es handelt sich hier um die DEBENU bzw. Foxit QuickPDF-Library).

Hier habe ich neben der .so-Datei folgende Dateien:

FoxitQPLLinuxDPP1612.h datei (C++ Import Header) (ca. 2000 Zeilen)
FoxitQPLLinuxDPP1612.cpp datei (C++ import Class) (ca. 7000 Zeilen)

Ich würde wahrscheinlich nur 1-2 Dutzend Funktionen aus der DLL benötigen.

In der .h -(Header) Datei stehen Dinge wie
Code:
"int UnlockKey(const std::wstring& LicenseKey);"
in der cpp stehen dann die Funktionen, wie z.B.:

Code:
"int FoxitQPLLinuxCPP1612::UnlockKey(const std::wstring& LicenseKey)
{
   if (loadError) return 0;
   return FoxitQPLUnlockKey(instanceID, WideStringParm(LicenseKey));
}"
Für die Windows-Version gibt es auch eine Delphi-Import-Datei für eine Windows-QPDF-DLL. Interessanterweise steht dort: "// This file was generated by an automated process".

Man kann da mit Delphi also auch die DLL verwenden (statt der auch existierenden DCU-Version). Für die Linux-DLL-Version gibt es sowas leider nicht.

Aber vom Grundsatz müsste ich doch einfach den C++Code in Pascal umschreiben können (und mich da an der DLL-Implementierung für Windows orientieren) und dann sollte die .so doch auch mit Delphi nutzbar sein, oder? Geht ja zumindest mit FMX-Linux auch, wo ja auch eine .so-Datei zum Einsatz kommen.

Gibt es da evtl. irgendwelche Generatoren, die einem den C++Code in Delphi-Pascal umwandeln?

jaenicke 20. Jan 2019 18:18

AW: C++ DLL in Delphi verwenden
 
Das sollte klappen, ja.

Ja, es gibt automatische Konverter, aber an einer manuellen Nachbearbeitung geht meistens kein Weg vorbei.

Oft geht es manuell auch schneller. Dafür gibt es aber einen per Package in Delphi integrierbaren Helper:
http://rvelthuis.de/programs/convertpack.html

Harry Stahl 20. Jan 2019 23:08

AW: C++ DLL in Delphi unter Linux verwenden
 
OK, danke.

Ich habe mal mit der Umsetzung für Linux / Delphi angefangen.

Die DLL lädt man unter Linux per "dlopen", das funktioniert soweit auch, ich erhalte ein gültiges handle.

Die Entsprechung von GetProcAdress ist unter Linux "dlsym", aber da habe ich ein Problem.

Unter Delphi (Windows) siehts so aus (verkürzt):

Delphi-Quellcode:

procedure AssignProc(var Proc: Pointer; ProcName: PAnsiChar);
begin
  Proc := GetProcAddress(FDLLHandle, ProcName);
end;

...
AssignProc(@DebenuPDFLibraryUnlockKey, 'DPLUnlockKey');
Unter C++ findet aber wohl so eine Art Typecast statt, da sieht es so aus:
Code:
FoxitQPLUnlockKey = (FoxitQPLFuncType121)AttachFunction("DPLUnlockKey");
Die Zuweisung in der AttachFunction sieht so aus (leicht gekürzt):

Code:
void* FoxitQPLLinuxCPP1612::AttachFunction(const char* funcName, bool ignoreError)
{
   void* address = dlsym(soHandle, funcName);
   return address;
}
FoxItQPLFunctType ist in der .h-Datei so deklariert:
Code:
typedef int (*FoxitQPLFuncType121)(int, wchar_t*);
Was bedeutet das für die Übernahme nach Delphi? So hätte ich es mal angenommen, funktioniert aber nicht:

Delphi-Quellcode:
    Proc := dlsym (FDLLHandle, ProcName); // Hier erhalte ich einen Fehler "Segmentations fault (11).

In der Klasse habe ich das so deklariert, muss das evtl. anders sein (C-Paramenter-Reihenfolge, etc?):

Delphi-Quellcode:

type
  TDebenuPDFLibraryDLL1612 = class
  private
    FLibraryLoaded: Boolean;
    FDLLHandle: Cardinal;
    FInstanceID: Integer;
  ...
   
    DebenuPDFLibraryUnlockKey: function(InstanceID: Integer; LicenseKey: PWideChar): Integer; stdcall;

  ...
   end;
In der C++-Klasse ist es so definiert:

Code:
typedef int (*FoxitQPLFuncType121)(int, wchar_t*);

FoxitQPLFuncType121 FoxitQPLUnlockKey;
Jemand eine Idee, was ich hier ändern müsste?

Zacherl 21. Jan 2019 10:52

AW: C++ DLL in Delphi unter Linux verwenden
 
Zitat:

Zitat von Harry Stahl (Beitrag 1423684)
Delphi-Quellcode:
    Proc := dlsym (FDLLHandle, ProcName); // Hier erhalte ich einen Fehler "Segmentations fault (11).

Sicher, dass dein Handle an der Stelle korrekt ist? Und auch dass
Delphi-Quellcode:
ProcName
auf einen validen nullterminierten String zeigt? Anders kann ich mir den SegFault hier beim Aufruf eigentlich nicht erklären. Deine Type Definitions und der ganze Rest haben hier eigentlich noch gar keinen Einfluss auf irgendwas.

Harry Stahl 21. Jan 2019 18:32

AW: C++ DLL in Delphi unter Linux verwenden
 
OK, danke, der Tipp war gut. Da stimmte ein Bezug zum DLL-Namen nicht, so dass die Datei gar nicht geladen werden konnte.

Interessanterweise hatte ich das gedebugt, der Debugger zeigte mir aber einen Wert <> 0 für das Handle an (obwohl tatsächlich null).

Da scheint es ein kleines Debug-Problem unter Linux zu geben?

Wie auch immer, jetzt funktioniert es...

Was mir allerdings nicht gelingt, ist der richtige Aufruf der Unlockkey-Function, ich erhalte (trotz richtigem Key) immer Null zurück, müsste aber 1 sein.

Muss ich evtl.
DebenuPDFLibraryUnlockKey: function(InstanceID: Integer; LicenseKey: PWideChar): Integer; stdcall;

ändern in
DebenuPDFLibraryUnlockKey: function(InstanceID: Integer; LicenseKey: PWideChar): Integer; cdecl;

Oder noch was anderes (denn diese Änderung allein bringt es leider auch nicht)...

Fritzew 21. Jan 2019 18:57

AW: C++ DLL in Delphi unter Linux verwenden
 
Char* auf C-Seite ist im Normalfall ein PAnsiChar in Delphi.

Harry Stahl 21. Jan 2019 21:55

AW: C++ DLL in Delphi unter Linux verwenden
 
Nein, mit PAnsiChar komme ich leider auch nicht weiter

In C++ ist die Funktion wie folgt deklariert:

Code:
int FoxitQPLLinuxCPP1611::UnlockKey(const std::wstring& LicenseKey)
{
    if (loadError) return 0;
    return FoxitQPLUnlockKey(instanceID, WideStringParm(LicenseKey));
}
Also das sieht für mich eindeutig nach Widestring aus. In der Windows-Unit, welche den Zugriff auf die Windows-QPDF-Dll ermöglicht, geht es auch per Widechar...

Klaus01 22. Jan 2019 05:05

AW: C++ DLL in Delphi unter Linux verwenden
 
.. viellieicht hilft das: https://stackoverflow.com/questions/...g-vs-stdstring

Grüße Klaus

Fritzew 22. Jan 2019 07:39

AW: C++ DLL in Delphi unter Linux verwenden
 
Zitat:

Zitat von Harry Stahl (Beitrag 1423773)
Code:
int FoxitQPLLinuxCPP1611::UnlockKey(const std::wstring& LicenseKey)
{
    if (loadError) return 0;
    return FoxitQPLUnlockKey(instanceID, WideStringParm(LicenseKey));
}

Dann sollte Deine Deklaration allerdings korrekt sein,
wie ist die function WideStringParm deklariert?

Wo ich mir nicht sicher bin ist die calling convention, ich denke es müsste cdecl sein......

Klaus01 22. Jan 2019 07:46

AW: C++ DLL in Delphi unter Linux verwenden
 
.. kommt es nicht zu Problemen weil wchar_t unter Linux 4 Byte groß ist.
wstring basiert auf wchar_t.
Delphi hat meines Wissens keinen (String)character type der 4 Byte groß ist.

Grüße
Klaus


Alle Zeitangaben in WEZ +1. Es ist jetzt 18:08 Uhr.
Seite 1 von 2  1 2      

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz