Exos Assembleur
Rappels de cours⚓︎
Le langage assembleur⚓︎
Un processeur est constitué de circuits logiques, etc. Il ne prend en entrée que des 0 et des 1, et
ne ressort que des 0 et des 1. À l’aide, par exemple, de circuits comme un multiplexeur, on peut
voir qu’il peut utiliser des parties de circuits différentes suivant le code binaire reçu. Ainsi, on
peut imaginer qu’un signal 00 1100 0101 lui fasse réaliser l’addition entre 1100 et 0101, et qu’un
signal 01 1100 0101 lui fasse réaliser la soustraction entre 1100 et 0101.
Un ordinateur fonctionne donc avec des instructions passées en binaire (des 0 et des 1), toutes
codées sur le même nombre de bits (par ex : 32 bits pour un processeur 32 bits).
Par exemple :
| En binaire | En hexadécimal | En assembleur |
|---|---|---|
00100100000010000000000000001010 |
2408000a |
li $t0, 10 |
00100100000010010000000000001100 |
2409000c |
li $t1, 12 |
00000001000010010101000000100000 |
01095020 |
add $t2, $t0, $t1 |
Pour plus de lisibilité, on peut utiliser un langage d’assemblage, ou langage assembleur, qui représente les instructions par des codes faciles à lire et à retenir pour un humain. Lors de l’assemblage, chaque code d’instruction est traduit en nombre. Chaque instruction correspond à une des tâches basiques que sait faire le CPU : charger une valeur dans une des cases du registre, additionner deux valeurs et charger le résultat dans une des cases du registre, etc.
C’est quoi le registre ? Comment fonctionne un processeur ?⚓︎
Prenons l’exemple d’un processeur MIPS R2000. Il s’agit d’un processeur 32 bits (= 4 octets, c’est la taille des registres) constitué de :
- 32 registres de 32 bits (stocke les valeurs temporaires, les pointeurs d’adresses, etc.)
- un compteur de programmes PC (= Program Counter) sur 32 bits (stocke l’adresse de l’instruction en cours d’exécution)
- un registre d’instruction RI sur 32 bits (stocke l’instruction en cours d’exécution, chaque instruction est donc codée sur 32 bits.)
- une mémoire vive adressable de 232 octets (permet de stocker les instructions du programme, les données, etc.)
La mémoire est comme un grand tableau dont chaque case mesure 32 bits (4 octets), la mémoire mesure \(2^{32}\) octets, donc elle contient \(2^{30}\) cases de 4 octets. Pour les repérer, on utilise des adresses, également sur 32 bits, il y a donc \(2^{32}\) adresses différentes. Chaque case mémoire (qui prend 4 octets) a donc une adresse qui est un multiple de 4. Ces adresses peuvent être représentées :
- en binaire : \(00000000 00000000 00000000 00000000_2\) est la toute première adresse (
0) ; et \(11111111 11111111 11111111 11111100_2\) est l’adresse de la dernière case. - en hexadécimal : \(00 00 00 00\_{16}\) est la toute première adresse (
0) ; et \(FF FF FF FC\_{16}\) est l’adresse de la dernière case.
Voir schéma en bas de cette page (annexe A)
Concrètement, que se passe-t-il lorsqu’on exécute un programme ?⚓︎
Le langage Python que nous utilisons en classe est un langage de « haut niveau », évolué. Lorsqu’il est interprété, il est traduit en langage « machine », de manière à pouvoir être exécuté sur un ordinateur.
À partir du langage machine, lorsqu’on fait exécuter le programme :
- le programme est stocké en mémoire :
- les lignes de code sont chargées dans les cases de la mémoire allouées au code,
- les constantes (données =
.data) du programme sont chargées dans les cases de la mémoire allouées aux données. - Le PC s’initialise à la première ligne de code, l’exécute, il peut s’agir :
- d’affectations de variables,
- de calculs par le CPU,
- de sauts (branchements) dans le programme en modifiant la valeur du PC.
S’il n’y a pas eu de branchement, le PC passe à la ligne suivante (son adresse prend \(+4\)).
Exemple 1⚓︎
| Langage assembleur | Rôle | En Python |
|---|---|---|
.text |
Annonce que les lignes suivantes sont des instructions à sauvegarder ligne par ligne dans la mémoire, et à exécuter. | On suppose que la variable x est stockée dans le registre $t0 et y dans le registre $t1. |
li $t0, 10 |
Charge la valeur immédiate 10 dans le registre $t0. |
x = 10 |
li $t1, 12 |
Charge la valeur immédiate 12 dans le registre $t1 |
y = ~12 |
add $t2, $t0, $t1 |
Copie dans le registre $t2 le résultat de la somme du contenu du registre $t0 avec le contenu du registre $t1. |
z = x + y |
li $v0, 1 |
Charge la valeur immédiate 1 dans le registre $v0 pour annoncer que le prochain appel système syscall sera un print_int. |
print(z) |
move $a0, $t2 |
Copie le contenu de $t2 dans le registre $a0 pour annoncer que le prochain appel système syscall utilisera le contenu de la variable $a0, c’est-à-dire le contenu de $t2. |
print(z) |
syscall |
Appel système : exécute le print_int contenu dans $a0 |
print(z) |
li $v0,10 |
Charge la valeur immédiate 10 dans le registre $v0 pour annoncer que le prochain appel système syscall sera exit. |
Fin du programme. |
syscall |
Appel système : exit. |
Pour ce qui suit, on se servira des tables en bas de cette page (annexe B): MIPS Reference Card.
Exercice 1⚓︎
Traduire ce programme en Python le programme en assembleur suivant.
.text
li $v0, 5
syscall
move $t0, $v0
li $v0, 5
syscall
move $t1, $v0
add $t2, $t0, $t1
li $v0, 1
move $a0, $t2
syscall
li $v0,10
syscall
Structures conditionnelle et boucles⚓︎
Toutes ces structures ont pour conséquence de ne pas toujours exécuter toutes les lignes
d’instructions les unes après les autres, dans l’ordre dans lequel elles sont écrites. On réalise
donc des branchements (jump) en affectant au PC l’adresse de la prochaine instruction à réaliser.
En assembleur, les labels servent à référencer certaines parties du programme en donnant un nom à une ligne du programme. Ceci permet d’y faire référence pour faire des sauts dans le programme, pour réaliser des branchements, pour des structures conditionnelles ou des boucles. Nous ignorerons les questions de « délais de branchement ».
Les branchements conditionnels⚓︎
Le résultat d’un test peut envoyer à une autre adresse d’instruction. Par exemple, pour la ligne
d’instruction : beq $t0, $t1, Label ; si la valeur contenue dans le registre $t0 est égale à celle
stockée dans le registre $t1 alors la prochaine instruction à exécuter est celle placée après
l’étiquette Label. beq vient de branch on equal (on fait le branchement si c’est égal). Il y a aussi
bne (branch on not equal), bgt (branch greater than) etc.
Exemple 2⚓︎
li $t0, 2
li $t1, 3
blt $t0, $t1, endif // branch less than
move $t1, $t0
endif:li $v0, 1 // print_int
move $a0, $t1
syscall
Code Python⚓︎
x = 2
y = 3
if x >= y : # si x < y, on va directement à endif, sinon, on exécute tout
y = x
print(y) # on affiche le max des deux
Exercice 2⚓︎
Traduire le programme suivant en Python.
.text
li $v0, 5 // read_int
syscall
move $t0, $v0
bge $t0, $zero, endif // greater than or equal 0
sub $t0, $zero, $t0
endif:li $v0, 1 // print_int
move $a0, $t0
syscall
Les branchements inconditionnels⚓︎
Une instruction peut envoyer directement (sans condition) à une autre adresse d’instruction. Par
exemple, la ligne d’instruction : j Label renvoie à la ligne Label.
Exemple 3⚓︎
.data
msg : .asciiz "gagné !" // on sauve une donnée string
.text
li $t0, 5
li $t1, 0
while:beq $t0, $t1, end // si $t0=$t1, aller à end
li $v0, 5 // read_int
syscall
move $t1, $v0
j while // renvoie à la ligne while
end: li $v0, 4 // print_string
la $a0, msg // adresse de la chaîne à imprimer
syscall // imprime le message
Code Python⚓︎
x = 5
y = 0
while x != y : # si x = y, on arrête la boucle
y = int(input())
print("gagné !")
Exercice 3⚓︎
Traduire ce programme en Python. Que fait-il ?
.data
msg: .asciiz "reste =" // on sauve une donnée string
.text
li $v0, 5 // read_int
syscall
move $t0, $v0
li $v0, 5 // read_int
syscall
move $t1, $v0
while:blt $t0, $t1, end //si $t0>$t1, aller à end
sub $t0, $t0, $t1
j while //retour au Label while
end: li $v0, 4 // print_string
la $a0, msg // adresse de la chaîne à imprimer
syscall // on l’imprime
li $v0, 1 // print_int
move $a0, $t0 // valeur de l’entier à imprimer
syscall // on l’imprime
Exercice 4⚓︎
Traduire ce programme en Python. Que fait-il ?
.data
msg: .asciiz "résultat =" // on sauve une donnée string
.text
li $t0, 15
li $t1, 1
li $t2, 0
while:bgt $t1, $t0, end //si $t1>$t0, aller à end
add $t2, $t2, $t1
addi $t1, $t1, 1
j while //retour au Label while
end: li $v0, 4 // print_string
la $a0, msg // adresse de la chaîne à imprimer
syscall // on l’imprime
li $v0, 1 // print_int
move $a0, $t2 // valeur de l’entier à imprimer
syscall // on l’imprime
Exercice 5⚓︎
Traduire ce programme en Python. Que fait-il ?
.data
msg: .asciiz "résultat =" // on sauve une donnée string
.text
li $t0, 10
li $t1, 1
li $t2, 2
while:bgt $t2, $t0, end //si $t1>$t0, aller à end
mul $t1, $t1, $t2
addi $t2, $t2, 1
j while //retour au Label while
end: li $v0, 4 // print_string
la $a0, msg // adresse de la chaîne à imprimer
syscall // on l’imprime
li $v0, 1 // print_int
move $a0, $t1 // valeur de l’entier à imprimer
syscall // on l’imprime
Annexes⚓︎
Annexe A⚓︎

Annexe B⚓︎

