AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein Programmieren allgemein Delphi Interface Problem - AV beim Beenden des Programms
Thema durchsuchen
Ansicht
Themen-Optionen

Interface Problem - AV beim Beenden des Programms

Ein Thema von Aviator · begonnen am 17. Okt 2016 · letzter Beitrag vom 18. Okt 2016
Antwort Antwort
Aviator

Registriert seit: 3. Jun 2010
1.611 Beiträge
 
Delphi 10.3 Rio
 
#1

Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 14:13
Hallo Delphianer,

nachdem ich einige Hilfe im Bezug auf DLL Programmierung von euch erhalten habe, habe ich ein weiteres Problem.

Aktuell habe ich folgenden Aufbau:
  • User DLL (DLL zum Verwalten und zum Auslesen von Benutzerdaten aus einer Datenbank)
  • Session DLL (DLL um eine SessionID zu erzeugen mit der ein Benutzer nach der Anmeldung arbeiten kann)
  • 2 unterschiedliche Interfaces

Ich denke, dass der Aufbau der Interfaces in der Sache egal ist. Deshalb poste ich die im Moment mal nicht. Sollten sie dennoch benötigt werden, dann kann ich sie natürlich einstellen.

Beim Start des Programmes lade ich beide DLLs. Eine DLL stellt nur eine Methode, die andere zwei im Export Abschnitt zur Verfügung, welche jeweils eine Interface Instanz zurückgeben. Das funktioniert soweit auch alles perfekt. Als Übergabeparameter erhält die exportierte Methode ein Interface, welches Datenbank Verbindungsparameter zur Verfügung stellt.

Delphi-Quellcode:
function GetInstance{(DataBaseConnectionInfo: IDataBaseConnectionInfo)}: IDMSSession; stdcall;
begin
  Result := TSessionManager.Create{(DataBaseConnectionInfo)};
end;


exports
  GetInstance;
Delphi-Quellcode:
function GetInstanceInt(DataBaseConnectionInfo: IDataBaseConnectionInfo): IDMSUserManager; stdcall;
begin
  Result := TDMSUserManager.Create(DataBaseConnectionInfo) as IDMSUserManager;
end;

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

exports
  GetInstanceInt,
  GetInstanceExt;
Die erste DLL (User DLL) kann ich problemlos laden und auch Funktionen darin aufrufen.

Die zweite DLL (Session DLL) lässt sich laden, ich kann auch Funktionen daraus aufrufen, aber beim Beenden des Programms schmiert die Anwendung mit einer AV ab.
Alle Aufrufkonventionen stimmen überein, es werden keine const Parameter verwendet wenn ich Interfaces zurückgebe, ...

Das an beide DLLs übergebene Interface habe ich testweise auch ein zweites Mal als zusätzliche Instanz erzeugt und auch mal ganz weggelassen (siehe die Codebeispiele oben). Trotzdem der gleiche Fehler. Nehme ich die Session DLL komplett raus, passiert gar nichts (wie erwartet). Lasse ich die Session DLL drin und nehme die User DLL raus, schmiert das Programm auch ab. Es hängt also wohl mit der Session DLL zusammen.

Die Fehlermeldung lautet:

Im Projekt Project1.exe ist eine Exception der Klasse $C0000005 mit der Meldung 'access violation at 0x0040d9a2: read of address 0x03fa9bc8' aufgetreten.

Das Programm beendet sich sauber, wenn ich die Variable der das Interface zugeordnet wurde vor dem Freigeben der Klasse in der die Variable verwaltet wird auf nil setze.

Warum ist das so? Wieso funktioniert es bei einer DLL und bei einer andere nicht?

Das Programm schmiert in der System.pas in der folgendes Procedure ab. Deshalb dachte ich, dass das etwas mit den Interfaces zu tun haben muss. Nur weiß ich nicht was.

Delphi-Quellcode:
function _IntfClear(var Dest: IInterface): Pointer;
{$IFDEF PUREPASCAL}
var
  P: Pointer;
begin
  Result := @Dest;
  if Dest <> nil then
  begin
    P := Pointer(Dest);
    Pointer(Dest) := nil;
    IInterface(P)._Release;
  end;
end;
{$ELSE !PUREPASCAL}
{$IFDEF CPUX86}
asm
        MOV EDX,[EAX]
        TEST EDX,EDX
        JE @@1
        MOV DWORD PTR [EAX],0
{$IFDEF ALIGN_STACK}
        SUB ESP, 4
{$ENDIF ALIGN_STACK}
        PUSH EAX
        PUSH EDX
        MOV EAX,[EDX] // <--------- Hier bleibt der Debugger stehen!!!!!!!!!!!!!!
        CALL DWORD PTR [EAX] + VMTOFFSET IInterface._Release
        POP EAX
{$IFDEF ALIGN_STACK}
        ADD ESP, 4
{$ENDIF ALIGN_STACK}
@@1:
end;
{$ENDIF CPUX86}
{$ENDIF !PUREPASCAL}
Ich hoffe, dass mir jemand bei dem Problem helfen kann. Ich weiß langsam wirklich nicht mehr weiter.
  Mit Zitat antworten Zitat
Fritzew

Registriert seit: 18. Nov 2015
Ort: Kehl
678 Beiträge
 
Delphi 11 Alexandria
 
#2

AW: Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 14:42
Zitat:
Das Programm beendet sich sauber, wenn ich die Variable der das Interface zugeordnet wurde vor dem Freigeben der Klasse in der die Variable verwaltet wird auf nil setze.
Das ist das richtige Vorgehen. Nur dann kannst Du sicher sein wann das Interface freigegeben wird.
Wahrscheinlich ist die Dll schon gar nicht mehr geladen wenn es bei Dir knallt.
Fritz Westermann
  Mit Zitat antworten Zitat
Aviator

Registriert seit: 3. Jun 2010
1.611 Beiträge
 
Delphi 10.3 Rio
 
#3

AW: Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 14:58
Zitat:
Das Programm beendet sich sauber, wenn ich die Variable der das Interface zugeordnet wurde vor dem Freigeben der Klasse in der die Variable verwaltet wird auf nil setze.
Das ist das richtige Vorgehen. Nur dann kannst Du sicher sein wann das Interface freigegeben wird.
Wahrscheinlich ist die Dll schon gar nicht mehr geladen wenn es bei Dir knallt.
Ach echt? Ich dachte immer, dass das egal sei. Interface-Instanzen geben sich doch selbst frei wenn die Referenzzählung auf 0 runter geht.

Und ja, die DLL wird vorher von mir entladen. Nur wieso funktioniert es bei der einen DLL und bei der andere nicht? Und gerade bei der, bei der es nicht funktioniert, rufe ich nicht mal eine Funktion auf. Ich lade sie nur und lasse mir eine Instanz geben. Das war es auch schon. Mit der anderen DLL arbeite ich bereits schon durchgehend und da passiert nichts.
  Mit Zitat antworten Zitat
Fritzew

Registriert seit: 18. Nov 2015
Ort: Kehl
678 Beiträge
 
Delphi 11 Alexandria
 
#4

AW: Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 15:21
Zitat:
Und ja, die DLL wird vorher von mir entladen. Nur wieso funktioniert es bei der einen DLL und bei der andere nicht? Und gerade bei der, bei der es nicht funktioniert, rufe ich nicht mal eine Funktion auf. Ich lade sie nur und lasse mir eine Instanz geben. Das war es auch schon. Mit der anderen DLL arbeite ich bereits schon durchgehend und da passiert nichts.
Einfache Regel: Wenn Du die Dll entlädst darf keine Referenz auf ein Interface der Dll mehr da sein.
Ansonsten wird beim Versuch das freizugeben code aufgerufen den es gar nicht mehr gibt.
Und ja greift auch die Referentzählung aber was soll den dann gemacht werden. Kann ja nicht gehen wenn die dll nicht mehr da ist.

Und du rufst Code auf in dem Du eine Instanz holst.

Die Klasse existiert ja in Deiner DLL.
Fritz Westermann
  Mit Zitat antworten Zitat
Benutzerbild von DeddyH
DeddyH

Registriert seit: 17. Sep 2006
Ort: Barchfeld
27.644 Beiträge
 
Delphi 12 Athens
 
#5

AW: Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 15:26
http://www.delphipraxis.net/189047-[erledigt]-interface-aus-dll-per-latebinding-freelibrary-toedlich.html
Vielleicht hilft es.
Detlef
"Ich habe Angst vor dem Tag, an dem die Technologie unsere menschlichen Interaktionen übertrumpft. Die Welt wird eine Generation von Idioten bekommen." (Albert Einstein)
Dieser Tag ist längst gekommen
  Mit Zitat antworten Zitat
Aviator

Registriert seit: 3. Jun 2010
1.611 Beiträge
 
Delphi 10.3 Rio
 
#6

AW: Interface Problem - AV beim Beenden des Programms

  Alt 17. Okt 2016, 15:44
Zitat:
Und ja, die DLL wird vorher von mir entladen. Nur wieso funktioniert es bei der einen DLL und bei der andere nicht? Und gerade bei der, bei der es nicht funktioniert, rufe ich nicht mal eine Funktion auf. Ich lade sie nur und lasse mir eine Instanz geben. Das war es auch schon. Mit der anderen DLL arbeite ich bereits schon durchgehend und da passiert nichts.
Einfache Regel: Wenn Du die Dll entlädst darf keine Referenz auf ein Interface der Dll mehr da sein.
Ansonsten wird beim Versuch das freizugeben code aufgerufen den es gar nicht mehr gibt.
Und ja greift auch die Referentzählung aber was soll den dann gemacht werden. Kann ja nicht gehen wenn die dll nicht mehr da ist.

Und du rufst Code auf in dem Du eine Instanz holst.

Die Klasse existiert ja in Deiner DLL.
Okay! Ist verständlich. Danke für die Info.

Ist ja eigentlich das Gleiche was Fritzew auch schon geschrieben hat. Dennoch danke dafür.
Überlege, ob ich das dann auch ganz raus lasse. Nur räume ich eben immer gerne selbst alles auf was ich benutzt habe. In dem Fall kann ich auch eigentlich sicher sein, dass ich die DLL(s) nicht mehr benötige.
  Mit Zitat antworten Zitat
Benutzerbild von jaenicke
jaenicke

Registriert seit: 10. Jun 2003
Ort: Berlin
9.704 Beiträge
 
Delphi 11 Alexandria
 
#7

AW: Interface Problem - AV beim Beenden des Programms

  Alt 18. Okt 2016, 06:58
Dafür gibt es in unserer DLL Schnittstelle ein Shutdown. Das wird vor dem Entladen aufgerufen, daraufhin koppelt sich die DLL aus allen Eventhandlern usw. aus.
Sebastian Jänicke
AppCentral
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 18:15 Uhr.
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz