Section courante

A propos

Section administrative du site

Utilisation du langage de programmation assembleur

Le Free Pascal prend en charge l'insertion d'instructions assembleur entre le code Pascal. Le mécanisme pour cela est le même que sous Turbo Pascal et Delphi. Il existe cependant des différences substantielles, comme cela sera expliqué dans les sections suivantes.

Utiliser l'assembleur dans les sources

Il existe essentiellement 2 façons d'intégrer du code assembleur dans la source pascal. La première est la plus simple, en utilisant un bloc ASM :

  1. Var
  2.  I:Integer;
  3. BEGIN
  4.  I:=3;
  5.  ASM
  6.   MOVL I,%EAX
  7.  END;
  8. END.

Tout ce qui se trouve entre le bloc ASM et END est inséré en tant qu'assembleur dans le code généré. Selon le mode du lecteur assembleur, le compilateur effectue la substitution de certains noms par leurs adresses.

La deuxième méthode consiste à implémenter une procédure ou une fonction complète en assembleur. Cela se fait en ajoutant un modificateur assembleur à l'entête de la fonction ou de la procédure :

  1. Function GetEipasEBX:Pointer;Assembler;ASM
  2.  MOVL (%ESP),%EBX
  3.  RET
  4. END;

Il est toujours possible de déclarer des variables dans une procédure assembleur :

  1. Procedure Move(Const Source;Var Dest;Count:SizeInt);Assembler;
  2. Var
  3.  SaveEsi,SaveEdi:LongInt;
  4. ASM
  5.  MOVL %EDI,SaveEdi
  6. END;

Le compilateur réservera de l'espace sur la pile pour ces variables, il insère des commandes pour cela. Notez que le nom d'assembleur d'une fonction assembleur sera toujours 'mangled' par le compilateur, c'est-à-dire que l'étiquette de cette fonction ne sera pas le nom de la fonction déclarée. Pour changer cela, un modificateur Alias peut être utilisé :

  1. Function GetEipasEBX:Pointer;Assembler;[Alias:'FPC_GETEIPINEBX'];ASM
  2.  MOVL (%ESP),%EBX
  3.  RET
  4. END;

Pour rendre la fonction disponible en code assembleur en dehors de l'unité courante, le modificateur Public peut être ajouté :

  1. Function GetEipasEBX:Pointer;Assembler;[Public,Alias:'FPC_GETEIPINEBX'];ASM
  2.  MOVL (%ESP),%EBX
  3.  RET
  4. END;

Assembleur en ligne Intel 80x86

Syntaxe Intel

Le Free Pascal prend en charge la syntaxe Intel pour la famille Intel de microprocesseurs Ix86 dans ses blocs ASM.

La syntaxe Intel dans votre bloc ASM est convertie en syntaxe AT&T par le compilateur, après quoi elle est insérée dans la source compilée. Les constructions d'assembleur prises en charge sont un sous-ensemble de la syntaxe d'assemblage normale. Dans ce qui suit, nous spécifions quelles constructions ne sont pas prises en charge dans Free Pascal, mais existant dans Turbo Pascal :

L'assembleur en ligne Intel prend en charge les macros suivantes :

Syntaxe AT&T

Dans les versions anciennes, le Free Pascal n'utilisait que GNU comme assembleur pour générer ses fichiers objets pour les microprocesseurs Intel x86. Ce n'est qu'après un certain temps qu'un assembleur interne a été créé, ayant écrit directement dans un fichier objet.

Étant donné que l'assembleur du GNU assembler utilise la syntaxe d'assemblage AT&T, le code que vous écrivez doit utiliser la même syntaxe. Les différences entre la syntaxe AT&T et Intel telle qu'utilisée dans Turbo Pascal sont résumées ci-dessous :

Vous trouverez plus d'informations sur la syntaxe AT&T dans le manuel as, bien que les différences suivantes avec l'assemblage AT&T normal doivent être prises en compte :

L'assembleur en ligne AT&T prend en charge les macros suivantes :

Macro Description
__RESULT Représente la valeur de retour du résultat de la fonction.
__SELF Représente le pointeur de méthode objet dans les méthodes.
__OLDEBP Représente l'ancien pointeur de base dans les routines récursives

Assembleur en ligne Motorola 680x0

Le lecteur assembleur en ligne pour la famille de microprocesseurs Motorola 680x0 utilise la syntaxe Motorola Assembler (q.v). Quelques différences existent :

L'assembleur en ligne prend en charge les macros suivantes :

Signalisation des registres modifiés

Lorsque le compilateur utilise des variables, il les entrepose parfois, ou le résultat de certains calculs, dans les registres du microprocesseur. Si vous insérez du code assembleur dans votre programme qui modifie les registres du processeur, cela peut interférer avec l'idée du compilateur sur les registres. Pour éviter ce problème, Free Pascal vous permet d'indiquer au compilateur quels registres ont changé dans un bloc ASM. Le compilateur va alors sauvegarder et recharger ces registres s'il les utilisait. Dire au compilateur quels registres ont changé se fait en spécifiant un ensemble de noms de registre derrière un bloc d'assemblage, comme suit :

  1. ASM
  2.  {... }
  3. END ['R1', ... ,'Rn'];

Ici, R1 à Rn sont les noms des registres que vous modifiez dans votre code assembleur. Par exemple :

  1. ASM
  2.  MOVL BP,%EAX
  3.  MOVL 4(%EAX),%EAX
  4.  MOVL %EAX,__RESULT
  5. END ['EAX'];

Cet exemple indique au compilateur que le registre EAX a été modifié.

Pour les routines assembleur, c'est-à-dire les routines entièrement écrites en assembleur, l'ABI du processeur et de la plate-forme doit être respectée, c'est-à-dire que la routine elle-même doit savoir quels registres sauvegarder et quoi ne pas, mais elle peut dire au compilateur en utilisant la même méthode ce que registres ont été modifiés ou non. Le compilateur enregistrera les registres spécifiés dans la pile à l'entrée et les restaurera à la sortie de la routine.

La seule chose que le compilateur fait normalement est de créer un cadre de pile minimal si nécessaire (par exemple, lorsque des variables sont déclarées). Tout le reste appartient au programmeur.



Dernière mise à jour : Mercredi, le 28 juin 2023