![]() |
Problem mit dynamischen dll´s
Hi,
ich komme gerade bei einem kleinen Problem nicht wirklich weiter. Ich habe ein Konstrukt aufgebaut das so aussieht: Programm <-> Globale Dll <-> Plugin Dll Die Global Dll ist dabei statisch im Programm eingebunden und die Plugin Dll wird wiederum von der Dlobal Dll dynamisch geladen. So sieht der Code aus: Programm:
Code:
Code der GlobalPlugin.dll
unit Unit1;
. . . function GlobalPlugin_show_optionwindow(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_show_about(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_show_additionalwindow(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_init(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_free(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_LoadPlugin(PluginName: string): boolean; stdcall; external 'GlobalPlugin.dll'; implementation {$R *.dfm} procedure TForm1.FormCreate(Sender: TObject); begin GlobalPlugin_init; end; procedure TForm1.btn1Click(Sender: TObject); begin GlobalPlugin_LoadPlugin(cbb1.Text); end; procedure TForm1.FormDestroy(Sender: TObject); begin GlobalPlugin_free; end; procedure TForm1.btn2Click(Sender: TObject); begin GlobalPlugin_Show_about; end; end.
Code:
und der Code des Plugin:
// Dll Import
function GlobalPlugin_Send_Cue(slave_id, kanal: Integer): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_Send_Value(value: Integer): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_show_optionwindow(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_show_about(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_show_additionalwindow(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_init(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_free(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_LoadPlugin(PluginName: string): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_Connect(com_port: word): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_Disconnect(): boolean; stdcall; external 'GlobalPlugin.dll'; function GlobalPlugin_Com_Status(): boolean; stdcall; external 'GlobalPlugin.dll'; library GlobalPlugin; uses ShareMem, Windows, SysUtils, Classes, dialogs, StrUtils; type T_Plugin_init = function(): Boolean; stdcall; T_Plugin_free = function(): Boolean; stdcall; T_Plugin_show_optionwindow = function(): Boolean; stdcall; T_Plugin_show_about = function(): Boolean; stdcall; T_Plugin_show_additionalwindow = function(): Boolean; stdcall; {$R *.res} var Handle: THandle; function Plugin_init(): boolean; var Plugin_init: T_Plugin_init; begin if Handle <> 0 then begin @Plugin_init := GetProcAddress(Handle, 'Plugin_init'); if @Plugin_init <> nil then begin result := Plugin_init; end; end; end; function Plugin_free(): boolean; var Plugin_free: T_Plugin_free; begin if Handle <> 0 then begin @Plugin_free := GetProcAddress(Handle, 'Plugin_free'); if @Plugin_free <> nil then begin result := Plugin_free; end; end; end; // Exports function GlobalPlugin_free(): boolean; stdcall; begin try if Handle <> 0 then begin Plugin_free; FreeLibrary(Handle); end; ComPort.Open := false; ComPort.Free; result := true; except result := false; end; end; function GlobalPlugin_LoadPlugin(PluginName: string): boolean; stdcall; begin if Handle <> 0 then begin Plugin_free; FreeLibrary(Handle); end; try Handle := LoadLibrary(PChar(ExtractFilePath(ParamStr(0)) + '/Plugins/' + PluginName)); Plugin_init; result := true; except result := false; end; end; function GlobalPlugin_show_optionwindow(): boolean; var Plugin_show_optionwindow: T_Plugin_show_optionwindow; begin if Handle <> 0 then begin @Plugin_show_optionwindow := GetProcAddress(Handle, 'Plugin_show_optionwindow'); if @Plugin_show_optionwindow <> nil then begin result := Plugin_show_optionwindow; end; end; end; function GlobalPlugin_show_about(): boolean; var Plugin_show_about: T_Plugin_show_about; begin if Handle <> 0 then begin @Plugin_show_about := GetProcAddress(Handle, 'Plugin_show_about'); if @Plugin_show_about <> nil then begin result := Plugin_show_about; end; end; end; function GlobalPlugin_show_additionalwindow(): boolean; var Plugin_show_additionalwindow: T_Plugin_show_additionalwindow; begin if Handle <> 0 then begin @Plugin_show_additionalwindow := GetProcAddress(Handle, 'Plugin_show_additionalwindow'); if @Plugin_show_additionalwindow <> nil then begin result := Plugin_show_additionalwindow; end; end; end; // Globale Hilfsfunktionen exports GlobalPlugin_LoadPlugin, GlobalPlugin_show_optionwindow, GlobalPlugin_show_about, GlobalPlugin_show_additionalwindow, GlobalPlugin_free; begin end.
Code:
Problem ist folgendes. Wenn ich das Plugin wechsel, sprich in der Comobox ein andere Plugin auswähle und den Button mit "GlobalPlugin_LoadPlugin(cbb1.Text)" klicke, friert das Programm ein.
unit MainU;
interface uses ShareMem, SysUtils, Classes, Windows, Forms, About_window, Optional_Window, Dialogs; function Plugin_init(): boolean; stdcall; function Plugin_free(): boolean; stdcall; function Plugin_show_optionwindow(): Boolean; stdcall; function Plugin_show_about(): Boolean; stdcall; function Plugin_show_additionalwindow(): Boolean; stdcall; implementation var Optionalwindow: TForm2; Aboutwindow: TForm1; // Dll Export Functions function Plugin_init(): Boolean; stdcall; begin try Optionalwindow := TForm2.Create(Application); Aboutwindow := TForm1.Create(Application); result := true; except result := false; end; end; function Plugin_free(): Boolean; stdcall; begin try Optionalwindow.Free; Aboutwindow.Free; result := true; except result := false; end; end; function Plugin_show_optionwindow(): Boolean; stdcall; begin //// Nicht vorhanden end; function Plugin_show_about(): Boolean; stdcall; begin Aboutwindow.ShowModal; end; function Plugin_show_additionalwindow(): Boolean; stdcall; begin Optionalwindow.Show; end; end. Beim beenden hängt es sich auch grundsätzlich auf. Wo kann der Fehler liegen ? Gruß Yannic |
W
Also die Anwendung stimmt.
If würd sagen der Fehler liegt hier: Handle := LoadLibrary(PChar(ExtractFilePath(ParamStr(0)) + '/Plugins/' + PluginName)); Du hast die falschen Slashs,die brauchst du: Handle := LoadLibrary(PChar(ExtractFilePath(ParamStr(0)) + '\Plugins\' + PluginName)); Zweitens eine Frage...das Plugin soll ja auch eine Dll sein, oder? Du hast aber statt library MainU; dort unit MainU; und dort auch die Funktionen exportieren, sonst können die Funktionen nicht dynamisch geladen werden. Und drittens würd ich sagen, das du bei der Globalen Dll die Imports rausnihmst....weil aus der eigenen Dll Funktionen exportieren? Warum das den? und das geht glaub ich auch garnet xD Um es mal zu erläutern. Und damit ich das richtig verstanden habe Deine Anwendung --> Läd Plugin DLL (Funktion von GlobalDLL); GlobaleDLL läd die Plugin DLL --> ruft Init aus der Plugin DLL auf |----- Resultat zur Anwendung --|--Weiterleitung-----------------|----- Resultat zur Globalen DLL --| So würde das dan Aussehen. Falls du weitere Dinge "Weiterleiten" willst, dann benutzt du besser Callbacks :-D Damit kannst du besser Daten an deine Anwendung schicken :-D |
AW: Problem mit dynamischen dll´s
Strings in DLL-Funktionen? Wenn du keinen gemeinesamen Speichermanager verwendest wirds krachen.
Schau doch mal was Delphi so als kommentar in der Hauptunit eines neuen DLL-Projekts schreibt. |
AW: Problem mit dynamischen dll´s
Liste der Anhänge anzeigen (Anzahl: 1)
@NickelM,
daran liegt es leider nicht. Beim ersten geladenen Plugin fubktioniert soweit alles, erst danach tauchen die Probleme auf. Das ganze ist eine Unit die in einer Dll eingebunden ist. "weil aus der eigenen Dll Funktionen exportieren? Warum das den?", das funktioniert normalerweise. Der Grund dafür ist folgender. Es gibt Funktionen usw. die alle Plugins benötigen. Wenn man die jetzt in jedes Plugin manuell einbinden würde, dann hat man eine extrem aufgeblasene dll so sind die Funktionen die für alle Dll´s relevant sind, mit in der Global.dll. Um weiterleiten geht es eigentlich auch weniger. Zurück zum Programm kommt nur ein Feedback in Form einer boolean variable. @Bernhard Geyer Ich verwende ShareMem. Ich habe das Testprojekt mal angehängt. Gruß Yannic Nachtrag: Ich habe ebend nochmal versucht ein wenig zu debuggen. Dabei konnte ich nun stark eingrenzen, wo die Probleme auftreten: if Handle <> 0 then begin Plugin_free; FreeLibrary(Handle); end; Diese Stelle in der Global.dll erzeugt die Probleme. Um genau zu sein "Invalid Pointer Opertaion" Gruß Yannic |
AW: Problem mit dynamischen dll´s
Hallo Blamaster,
die Probleme kommen daher, weil Du nicht ShareMem als ERSTE Unit in der uses-Liste schreibst. Muß in allen Deinen Projekten (exe und dlls) so sein, wenn Du Funktionen ex/importierst die Pascalstrings übergeben (direkt oder auch nur als Bestandteil eines Objekts). Zitat:
blauweiss |
AW: Problem mit dynamischen dll´s
an erster stelle in der usesliste vom Project der Exe und dll's und nicht an erster stelle
von der Unit der Exe und dll's hatte das auch mal verwechselt ^^ |
AW: Problem mit dynamischen dll´s
Hallo Capa,
genau.... da bin ich (natürlich) auch mal vor einiger Zeit drauf reingefallen, deswegen ist es mir bei Blamasters Projekt aufgefallen 8-) Gruss, blauweiss |
Aw: W
Moin Nickel,
Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 13:49 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-2025 by Thomas Breitkreuz