Rédacteur d'unité de chargement dynamique de dll

Soyez le premier à donner votre avis sur cette source.

Vue 7 127 fois - Téléchargée 642 fois

Description

Ce petit programme permet de faciliter la rédaction d'une unité permettant le chargement dynamique d'une dll en Delphi.

Il faut placer les déclarations telles qu'elles sont dans les fichiers de la dll, fournir quelques informations, et le code est généré automatiquement.

L'unité générée charge automatiquement la dll au démarrage de l'appli, et la décharge à sa fermeture.

Elle propose les routines telles qu'elles aurait été proposées si on avait simplement utilisé un chargement statique (external).

Le code généré reproduit presque le comportement exact de du chargement via external.

Presque, car une petite différence (inconnue) permet de se débarrasser d'un problème bien connu en Delphi.

En effet certains composants (Le GroupBox, par exemple) placés sur une fiche contenue dans une dll accèder depuis un exe sous thème XP conduit systématiquement à une violation d'accès à la fermeture de l'exe.

Le chargement dynamique de la librairie est pour le moment le seul remède que je connaisse à ce problème.

Si vous en avez un autre, n'hésitez pas à me le signaler !

Source / Exemple :


unit pasAccesDll;

{$DEFINE EXTERNAL}

//==============================================================================
{$IFDEF EXTERNAL}
interface

procedure AfficherForm(); stdcall; external 'Project2.dll';

implementation

end.
//==============================================================================
{$ELSE}

interface

uses
  SysUtils, Dialogs, Windows;

type TAfficherForm = procedure(); stdcall;

var
  hProject2LibraryHandle: Integer;

  AfficherForm: TAfficherForm;

implementation

initialization
  hProject2LibraryHandle:= SafeLoadLibrary('Project2.dll');
  if hProject2LibraryHandle = 0 then
    ShowMessage('Echec du chargement de la librairie Project2.dll')
  else
  begin
    AfficherForm:= GetProcAddress(hProject2LibraryHandle, 'AfficherForm');
    if @AfficherForm = nil then ShowMessage('Echec du chargement de la routine AfficherForm');
  end;
finalization
  FreeLibrary(hProject2LibraryHandle);
//==============================================================================
{$ENDIF}
end.

Conclusion :


Le zip contient l'appli de traduction dans le dossier 'StaticToDynamic'.

Le dossier 'fiche dans dll' contient quant à lui un exemple d'utilisation : une dll affichant une fiche avec le thème XP.
Par une directive de compilation de l'unité qui déclare la routine de la dll, on peut choisir entre les deux types de chargement :

1 Le code simple et concis avec external, mais qui peut planter si les conditions sont réunies (Comme c'est le cas dans l'exemple).
2 Le code généré, long et semblant inutile, mais qui ne plante pas !

Codes Sources

A voir également

Ajouter un commentaire Commentaires
Messages postés
3874
Date d'inscription
mardi 8 mars 2005
Statut
Modérateur
Dernière intervention
7 novembre 2014
14
J'adore mes posts ! On dirait que je suis skizo.

Je me demande ce que devient florenth...
Messages postés
3874
Date d'inscription
mardi 8 mars 2005
Statut
Modérateur
Dernière intervention
7 novembre 2014
14
Je me suis la tête encore 3 heures sur ce problème.

Apparement y en aurait plutôt 2 d'ailleurs.

1 Si on fait un Form.Free dans la finalization de la dll, on aboutit à une exception dans la dll
uxtheme.dll. Pour empècher ça, on peut faire un Action:= caFree; ou utiliser la méthode de Shinning.

2 Une exception dans la méthode Free de la classe TObject (Parfois après un petit temps d'execution anormal).

test eax, eax | -> Quitte si Self pointe sur nil.
jz sur le ret |
mov dl, $01
mov exc, [eax] -> C'est là que l'exception à lieu, à la récuération de l'adresse
de la classe.
call dword ptr [ecx- $04] -> Appel de la méthode Destroy de la classe.
ret

Le tout avec réserve, je suis pas pro.

L'exception a donc lieu car l'adresse de l'objet qu'on décide de libéré est invalide.

Comme il y a a priori n'importe quoi dans eax, cela explique peut être pourquoi des fois il y a erreur et des fois pas.

Apparement, on peut se débarrasser de cette erreur en mettant en commentaire l'appel à la métode UnloadThemeData.

Cet appel se trouve dans la méthode Destroy de la classe (Dont j'ai pas noté le nom, mais elle est toute seule) contenue dans l'unité Delphi7\Source\Lib\Themes.pas.

Une fois mis en commentaire, faut s'arranger pour que la compilation de la fiche de la dll prenne en considération cette modife. (Le dcu doit se trouver dans Delphi7\Lib).

Evidement, c'est pas une solution, mais ça permet peut être de mieux cerné le problème...

florenth -> Je pense à toi, mais ça risque effectivement de prendre une semaine. :,(
Messages postés
3874
Date d'inscription
mardi 8 mars 2005
Statut
Modérateur
Dernière intervention
7 novembre 2014
14
Désolé florenth mais je suis toutjours pas parvenut à faire passer mes fichiers depuis mon PC vers le net.

Je pourrais peut être le faire la semaine prochaîne...

Pour info, j'utilise une Delphi 7 Peros et j'ai un XP SP2 thème classique.

J'ai un peu regardé ce qu'il y a en source fournit avec Delphi. Apparement, les unités uxtheme et themes gère une partie du thème XP.

uxtheme charge dynamiquement la dll uxtheme.dll, et themes propose avant tout une classe, mais je sais pas à qui (Du côté de la VCL à mon avis).

Mon appli qui plante avec external parvient jusqu'à un point d'arrêt situé sur le end. de la finalization de l'unité themes... L'exception à lieu après ça. J'ai aussi essayé d'empécher le déchargement de la classe et de la dll uxtheme, mais cela n'a donné aucun résultat.
C'est marrant, chacun a sa méthode qu imarche chez lui !
Pour que je puisse faire une analyse plus complète, RT15, tu pourrais m'envoyer la version compilee de ton projet (avec la dll et define EXTERNAL), comme ça je comparerai les executables.

mon mail : mon_pseudo (à remplacer bien sur) -AT- msn -DOT- com

Merci d'avance et pourvu qu'on trouve le pourquoi du comment !! ^^
Messages postés
3826
Date d'inscription
vendredi 23 juillet 2004
Statut
Modérateur
Dernière intervention
10 mai 2021
44
Alors là tu as en a raté encore plus que tu ne pensais

Le Lien que tu donnes est celui de la question de RT15 :-)

aller bonne nuit ^^
Afficher les 8 commentaires

Vous n'êtes pas encore membre ?

inscrivez-vous, c'est gratuit et ça prend moins d'une minute !

Les membres obtiennent plus de réponses que les utilisateurs anonymes.

Le fait d'être membre vous permet d'avoir un suivi détaillé de vos demandes et codes sources.

Le fait d'être membre vous permet d'avoir des options supplémentaires.