Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Delphi Übergabe von Interface an DLL führt zu AccessViolation (https://www.delphipraxis.net/190478-uebergabe-von-interface-dll-fuehrt-zu-accessviolation.html)

Aviator 7. Okt 2016 17:06


Übergabe von Interface an DLL führt zu AccessViolation
 
Liste der Anhänge anzeigen (Anzahl: 1)
Hallo Delphianer,

ich mal wieder mit meinem DLL Problem. Ich habe jetzt angefangen meine DLL zu programmieren und stoße schon ziemlich direkt am Anfang auf einen Fehler.

Meine DLL exportiert zwei Funktionen. Beide geben eine Interface Instanz zurück. Das sollte ja noch kein Problem sein denke ich. :cyclops:

Mein aktueller SourceCode in der DPR der DLL die davon betroffen ist sieht so aus:

Delphi-Quellcode:
library DMSUserManager;

uses
  System.SysUtils,
  System.Classes,
  DMSUM.Manager in 'Source\Classes\DMSUM.Manager.pas',
  Common.Interfaces.Internal in '..\Common\Interfaces\Common.Interfaces.Internal.pas',
  Common.Interfaces in '..\Common\Interfaces\Common.Interfaces.pas',
  Common.Types in '..\Common\Types\Common.Types.pas';



{$R *.res}

function GetInstanceInt(DataBaseConnectionInfo: IDataBaseConnectionInfo): IDMSUserManager;
begin
  Result := TDMSUserManager.Create(DataBaseConnectionInfo) as IDMSUserManager;
end;

function GetInstanceExt(DataBaseConnectionInfo: IDataBaseConnectionInfo): IDMSUserExchanger;
begin
  Result := TDMSUserManager.Create(DataBaseConnectionInfo) as IDMSUserExchanger;
end;

exports
  GetInstanceInt,
  GetInstanceExt;

begin
end.
IDataBaseConnectionInfo ist wie folgt deklariert:

Delphi-Quellcode:
IDataBaseConnectionInfo = interface(IInterface)
['{5C829F64-F5D0-48B9-A7C7-B75A74C3AD7F}']
  procedure SetConnectionInfo(const AServer: PWideChar; const ADatabase: PWideChar; const AUserName: PWideChar;
    const APassword: PWideChar); stdcall;
  function GetConnectionString: PWideChar; stdcall;
end;
IDMSUserManager und IDMSUserExchanger sind so deklariert:

Delphi-Quellcode:
  IDMSUserExchanger = interface(IInterface)
  ['{96FE1197-EEC4-44D6-A305-E8C8FAE89484}']
    procedure FreeAllocatedMemory(var UserExchange: TUserExchange); stdcall; overload;
    procedure FreeAllocatedMemory(var UserExchange: array of TUserExchange); stdcall; overload;
    function GetActiveUserCount(var UserCount: Integer): Boolean; stdcall;
    function GetLastError: PWideChar; stdcall;
    function GetActiveUsers(const Index: Integer; var UserExchange: array of TUserExchange): Boolean; stdcall;
  end;

  IDMSUserManager = interface(IDMSUserExchanger)
  ['{0B695F23-B4E0-4FEA-9EFA-586B487F95B5}']
    function AddNewUser(var UserExchange: TUserExchange): Boolean; stdcall;
    function DeleteUser(const UserID: Integer): Boolean; stdcall;
    function GetAllUsers(var UserExchange: array of TUserExchange): Boolean; stdcall;
    function SetActivationState(const UserID: Integer; const Deactivated: Boolean): Boolean; stdcall;
  end;
Nun habe ich als Hostapplikation meine Anwendung hinterlegt und die von der DLL exportierte Methode GetInstanceInt() ausgeführt. Zuvor natürlich eine Instanz des
Delphi-Quellcode:
IDataBaseConnectionInfo
erzeugt und als Parameter übergeben. Ich vermute auch, dass hier der Fehler liegt. Nur weiß ich nicht wo ich ansetzen soll.

Hier noch mein SourceCode wie ich die DLL anspreche bzw. die GetInstance() Funktion aufrufe:

Delphi-Quellcode:
type
  TfrmDocuments = class(TForm)
  private
    ConData: IDataBaseConnectionInfo;
    Mgr: IDMSUserManager;
    ...
  public
    ...
  end;

implementation

procedure TfrmDocuments.btn1Click(Sender: TObject);
var
  cnt: Integer;
  ue: array of TUserExchange;
begin
  ConData := TConnectionStore.Create;
  ConData.SetConnectionInfo('_Server_', 'Dokumentenverwaltung', 'foo', 'bar');

  Mgr := GetInstanceInt(ConData);
  Mgr.GetActiveUserCount(cnt);

  SetLength(ue, cnt);
  if not Mgr.GetAllUsers(ue) then
    ShowMessage(Mgr.GetLastError);

end;
Ich habe bewusst die Variablen
Delphi-Quellcode:
ConData
und
Delphi-Quellcode:
Mgr
als privates Feld deklariert, hatte sie vorher aber auch schon direkt in der Procedure deklariert. Beides ergibt das selbe Ergebnis.

Die Fehlermeldung dich ich bekomme wenn die Zeile
Delphi-Quellcode:
Mgr := GetInstanceInt(ConData);
ausgeführt wird lautet wie folgt:

"Im Projekt DocuMentS.exe ist eine Exception der Klasse $C0000005 mit der Meldung 'access violation at 0x015bdd77: read of address 0x00000004' aufgetreten."

Ich habe auch schon diverse BreakPoints gesetzt und mir auch (obwohl ich mich damit null auskenne) das Disassembly Fenster angeschaut. Anbei mal ein Screenshot. Vielleicht kann ja hemand von euch etwas damit anfangen.

Es ist ein erster Test ob es so funktioniert wie ich mir das vorstelle. Also bitte nicht an den Variablen Namen stören. Aber wie man sieht funktioniert es leider nicht. Die Frage ist nur warum nicht. Ich hoffe jemand von euch kann mir helfen.

bepe 7. Okt 2016 17:40

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Bin gerade auf dem Sprung aber auf die Schnelle kommen mir zwei Dinge in den Sinn:

- deine beiden Funktionen benutzen die falsche Aufrufkonvention
- Delphi kennt überhaupt keine Funktionen

Fritzew 7. Okt 2016 18:24

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Liste der Anhänge anzeigen (Anzahl: 1)
Hallo,

als erstes solltest Du die
Exporte auch mit stdcall aufrufen. also

Delphi-Quellcode:
function GetInstanceInt(DataBaseConnectionInfo: IDataBaseConnectionInfo): IDMSUserManager; stdcall; external 'DMSUserManager.dll';
 function GetInstanceExt(DataBaseConnectionInfo: IDataBaseConnectionInfo): IDMSUserExchanger; stdcall; external 'DMSUserManager.dll';
natürlich auch in der Dll ändern.

von was sind Deine konkreten Klassen abgeleitet?


Gruss Fritz

himitsu 7. Okt 2016 18:32

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Zitat:

Zitat von Fritzew (Beitrag 1350157)
als erstes solltest Du die
Exporte auch mit stdcall aufrufen.

Nee, wieso denn?

Es ist nur wichtig, dass auf beiden Seiten die Schnittstellen gleich aussehn
und wenn hier EXE und DLL mit Delphi geschrieben sind, dann kann die Aufrufkonvention auch problemlos "register" bleiben.

Was man aber nicht machen darf, ist einen PWideChar wie einen String aussehn zu lassen.

DelphiStrings sind zwar intern zu PChar kompatibel (
Delphi-Quellcode:
PChar(String)
ist also OK), aber andersrum natürlich nicht (niemals
Delphi-Quellcode:
String(PChar)
).

Und wenn in der DLL ein String in einen PChar gecastet wurde, dann darf der dennoch nicht zurück nach String gecastet werden.
* Die speicherverwaltung von DLL und EXE sind standardmäßig getrennt (jeder hat seine Eigene)
* und bei Strings gibt es den Sonderfall des Leerstrings, wo der
Delphi-Quellcode:
PChar(String)
-Cast auf eine Konstante mit dem Inhalt #0#0 umleitet.

Fritzew 7. Okt 2016 18:36

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Zitat:

Es ist nur wichtig, dass auf beiden Seiten die Schnittstellen gleich aussehn
und wenn hier EXE und DLL mit Delphi geschrieben sind, dann kann die Aufrufkonvention auch problemlos "register" bleiben.
Der Meinung bin ich nicht. Eine Dll sollte soweit als möglich Compilerunabhängig sein.
Und wenn man das mit so etwas einfachem wie stdcall erreichen kann ist das schon mal gut.

himitsu 7. Okt 2016 18:44

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Auch andere Sprachen kennen viele der Konventionen, somit ist es nicht per se inkompatibel, zu anderen Sprachen.

Fritzew 7. Okt 2016 19:01

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Nenn mir einen Win32 compiler (nicht Delphi / Fpc) der unter Win32 die Register Aufruf Konvention von Delphi kennt.
Ich lerne gerne dazu... :-)

Aviator 7. Okt 2016 19:23

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Hallo Fritze,

danke dir. :thumb:

Manchmal muss man nur mit der Nase drauf gestoßen werden. Die Aufrufkonvention
Delphi-Quellcode:
stdcall
hatte ich zwar in meinem Programm hinterlegt, aber nicht in der DLL. Jetzt flutscht es.

Nochmals vielen Dank! :cheers:

Ich hoffe alle weiteren Fehler bekomme ich von selbst gelöst.
Zitat:

Zitat von himitsu (Beitrag 1350159)
DelphiStrings sind zwar intern zu PChar kompatibel (
Delphi-Quellcode:
PChar(String)
ist also OK), aber andersrum natürlich nicht (niemals
Delphi-Quellcode:
String(PChar)
).

Wie denn dann? Hast du ein Beispiel?

Fritzew 7. Okt 2016 19:36

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Zitat:

Zitat von Aviator (Beitrag 1350168)
Zitat:

Zitat von himitsu (Beitrag 1350159)
DelphiStrings sind zwar intern zu PChar kompatibel (
Delphi-Quellcode:
PChar(String)
ist also OK), aber andersrum natürlich nicht (niemals
Delphi-Quellcode:
String(PChar)
).

Wie denn dann? Hast du ein Beispiel?

Das ist schlicht falsch. Der Delphi Compiler ersetzt deinen Cast String(Pchar) durch

Delphi-Quellcode:
procedure _UStrFromPWChar(var Dest: UnicodeString; Source: PWideChar);
Beweis: mit debug dcu compilieren und in den cast mit f7 reinsteppen

Gruss Fritz

Zacherl 7. Okt 2016 19:37

AW: Übergabe von Interface an DLL führt zu AccessViolation
 
Zitat:

Zitat von Aviator (Beitrag 1350168)
Zitat:

Zitat von himitsu (Beitrag 1350159)
DelphiStrings sind zwar intern zu PChar kompatibel (
Delphi-Quellcode:
PChar(String)
ist also OK), aber andersrum natürlich nicht (niemals
Delphi-Quellcode:
String(PChar)
).

Wie denn dann? Hast du ein Beispiel?

Doch doch, das kannst du ruhig machen. Delphi Hilfe sagt zur StrPas Funktion beispielsweise
Zitat:

This function is provided for backwards compatibility only. To convert a null terminated string to an AnsiString or native Delphi language string, use a typecast or an assignment.
Und im Assembly sieht man auch, dass Delphi da Compiler Magic betreibt:
Code:
Unit1.pas.30: S := 'hallo';
005C9BA3 B8D89B5C00       mov eax,$005c9bd8
005C9BA8 8945F8           mov [ebp-$08],eax
Unit1.pas.31: T := String(S);
005C9BAB 8D45F4           lea eax,[ebp-$0c]
005C9BAE 8B55F8           mov edx,[ebp-$08]
005C9BB1 E8EE0AE4FF      call @UStrFromPWChar


Alle Zeitangaben in WEZ +1. Es ist jetzt 22:17 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