Einzelnen Beitrag anzeigen

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, 13: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