Codage des nombres
Résumé : cf Résumé_Codage_des_Nombres
1 - Codage des Entiers Non Signés ( UNSIGNED )
1.1 - Bases
Définition : Une base B est un ensemble ordonné de N symboles.
Ex: Base 10 :
Cela permet d’exprimer des Quantités .
Nous utilisons habituellement cette base tout simplement parce que nous avons 10 doigts.
Le composant fondamental d’un processeur est le transistor, dont l’état peut être ouvert ou bloqué .
Les calculs dans un processeur reposent donc sur une base 2 ( base binaire ).
Expression d’une quantité en base 2 : j’applique la définition :
Ex:
Concernant la base 16, je dois disposer de 16 symboles.
Relation Base 2 <–> Base 16
Soit la valeur en base 2 :
Par définition :
soit
soit
soit
Pour passer de la base 2 à la base 16, il me suffit donc de faire des paquets de 4 bits, et de retrouver la correspondance entre les deux bases dans le tableau ci-dessus.
La base 16 peut être considérée comme une base binaire à écriture compactée.
Entendons nous bien : la quantité reste la même, qu’elle soit exprimée en base 2, 10 ou 16.
La plupart du temps dans nos programmes nous écrirons des nombres en base 10.
Le processeur verra cela comme de la base 2 de par son architecture, ( et les calculs seront effectués dans cette base )
Les bases 2 et 16 pourront être utilisées lors de l’écriture des programmes pour des questions de masquage ou d’écriture dans des registres de configuration des périphériques.
1.2 - Comment retrouver le codage d’une valeur ?
Reprenons l’exemple précédent :
Je peux écrire successivement :
soit:
Le codage binaire de 13 est donc
Autrement dit, je peux trouver le codage d’un nombre par divisions successives :
Bien entendu cela est vrai également pour la base 16 :
Le codage hexadécimal de 1234 est donc
Par conséquent, en binaire :
1.3 - Nombre de Valeurs Codables :
Le nombre de valeurs dans un processeur est toujours fini.
Le nombre de bits ( ou de digits ) détermine ce nombre de valeurs codables.
Ex: 4 bits : possibilités de codage, soit
Pour des grandeurs non signées ( unsigned ), je peux donc coder des valeurs allant de 0 à 15 ( ).
1.4 - Addition
En base 10 : si je dépasse 9, il y a une retenue sur le digit supérieur :
En binaire c’est pareil quand on dépasse 1 :
Sur 4 bits :
Et si ça déborde ?
En incluant la retenue qui déborde ( Carry ), le résultat est correct, mais ce dernier serait alors sur 5 bits.
Or il faut considérer un résultat sur 4 bits ( taille des opérandes ).
Ce résultat est donc faux, le bit d’état Carry permet alors d’indiquer un débordement pour une somme d’entiers non signés.
L’additionneur binaire :
Considèrons un additionneur élémentaire :
L’association parallèle de 4 additionneurs élémentaires permet de fabriquer un additionneur ( d’entiers non signés ) 4 bits :
2 - Codage des Entiers Signés ( SIGNED )
Jusqu’à présent toutes les valeurs étaient exclusivement positives.
Une première solution pour coder des quantités positives et négatives serait de modifier le bit de poids fort :
…
-2 –> 0b1010
-1 –> 0b1001
0 –> 0b0000
1 –> 0b0001
2 –> 0b0001
…
à ce moment-là, si j’additionne bit à bit 1 + (-1), j’obtiens 0b1010, soit -2.
Une addition avec ce codage supposerait quelques manipulations supplémentaires pour obtenir un résultat correct.
Il existe un codage permettant d’obtenir un résultat correct : le codage complément à 2 ( ou plus précisément complément à ).
Comme nous travaillons sur des intervalles finis , nous utilisons la propriété du modulo ( tout ce qui dépasse disparait ).
Exemple en base 10 : Je travaille dans l’intervalle [0 99], toutes les opérations se font modulo 100 :
40 - 30 = 10
J’obtiens le même résultat en faisant :
[ 40 + (100-30) ]modulo 100 = [ 40 + 70 ]modulo 100 = [ 110 ]modulo 100 = 10
Dans cette perspective on peut considérer que 70 représente une quantité négative -30
Tout ce qui est entre 0 et 49 sont des quantités positives, les valeurs comprises entre 50 et 99 représentent des quantités négatives.
En binaire c’est pareil, considérons un exemple sur 4 bits ( modulo ):
la quantité signée -1 aura le même codage que la quantité non signée , soit 15.
Réalisons alors l’addition 1 + (-1) :
Il y a une retenue dans C, mais le résultat est correct ( donc tout va bien )
Nous aurons donc sur 4 bits :
2.1- Définition d’un nombre codé en complément à 2
sur 4 bits :
or
donc
D’où la définition d’un nombre codé en complément à 2 ( sur 4 bits) :
2.2 - Entendons nous bien :
Même si tous les nombres négatifs ont un poids fort à 1 ( que l’on nommera d’ailleurs bit de signe) , Il ne suffit pas de mettre le poids fort à 1 pour trouver le codage de l’opposé d’un nombre positif.
2.3 - Comment retrouver le codage d’un nombre négatif à partir de sa valeur positive
- J’applique la définition du complément : par exemple +3 : 0b0011
Le codage de -3 en signé est le même que le codage de en non signé, soit 13 = 0b1101 - Nous pouvons remarquer que, quelque soit X :
Donc
Je peux donc prendre le codage de +3 : 0b0011, complémenter chaque bit et ajouter 1 pour trouver le codage de -3.
2.4 - Débordements
Si j’additionne 2 nombres de signes opposés ( ex : +7 + (-2), +2 + (-8), etc.. ) , je retomberai toujours dans l’intervalle de définition [-8 +7], donc pas de débordement.
L’addition de 2 nombres de même signe donnant un résultat de signe opposé correspond à un débordement signé ( oVerflow )
Remarquons au passage que les deux dernières retenues sont opposées dans ce cas.
2.5 - Extension de Format
Supposons que je passe d’un format signé 4 bits à un format signé 8 bits.
Nombres Positifs
il suffit d’ajouter des zéros sur les poids forts pour compléter :
+3 = 0b0011 –> 0b00000011
Nombres Négatifs
On peut remarquer que
Ainsi si j’applique la définition du codage complément à 2 :
Il suffit donc d’ajouter des 1 sur les poids forts pour étendre le format des nombres négatifs.
2.6 - L’additionneur/Soustracteur
Une soustraction peut être vue comme l’addition de l’opposé d’un nombre.
Nous avons vu précédemment que pour trouver le codage de l’opposé d’un nombre, il suffit de :
- complémenter ce nombre
- Ajouter 1
La complémentation conditionnelle peut être réalisée avec une porte ou exclusif ; La retenue de poids faible peut servir à ajouter 1.
3 - Codage des Réels : Virgule Fixe
Nous avons supposé précédemment que les pondérations étaitent entières :
Je peux très bien faire correspondre des pondérations fractionnaires à mes digits :
La Quantité Représentée est donc Fractionnaire
Le Codage ( les 0 et les 1 ) est le même pour la quantité +7 et +0.875.
Le format I.Q utilisé pour ce dernier est 1.3 ( 1 bit pour la partie entière, 3 bits pour la partie fractionnaire ).
Ce format est fixé a priori pour un calcul ( d’où le nom virgule fixe )
Dans les langages informatiques il n’existe pas de type standard ‘virgule fixe’.
Le processeur effectue ses calculs bit à bit ( cf additionneur ), le codage d’un entier étant le même que pour une quantité à virgule, c’est le type entier ( int ) qui est utilisé pour réaliser un calcul en virgule fixe.
Exemple :
Je dois réaliser le calcul suivant sur 8 bits : ( je dois normalement trouver 8.025 )
Je dois donc coder : 3.75, 2.3, -1.2 et 0.5.
Il me faut suffisamment de bits pour coder la partie entière ‘3’ de la valeur max ‘3.75’, soit 3 bits en signé.
On part donc sur le format I.Q 3.5
Le codage de 3.7 au format virgule fixe 3.5 est le même que le codage entier de , soit 120 = 0x78.
Il en va de même pour les autres valeurs :
– arrondi –> 74
– arrondi –> -38
REMARQUE : Comme le codage se fait sur un nombre de bits fini, je suis obligé d’arrondir ; ici l’erreur de codage sera toujours inférieure à
La multiplication de 2 valeurs au format 3.5 me donne un résultat au format 6.10 ( doublement du format des opérandes et décalage de la virgule comme en base 10 ).
J’ai donc un résultat fois trop grand.
Les arrondis font que mon résultat n’est pas exactement le résultat attendu.
4 - Codage des Réels Virgule Flottante
Dès lors que je déclare une variable en tant que float ou double dans un programme, il s’agit d’un codage virgule flottante.
4.1 - Norme IEEE P754 ( version 32 bits )
Le codage d’un nombre en virgule flottante se présente ainsi :
- s : bit de signe ( 0 positif, 1 négatif )
- E : Exposant
- M : Mantisse, 1 < M < 2, au format virgule fixe unsigned 1.23
Exemple : Codage de -60.47
s = 1
Comme le bit de poids fort de la mantisse vaut toujours 1, ce bit de poids fort disparait.
4.2 - Intérêts/inconvénients du codage virgule flottante :
Intérêts :
- Dynamique : possibilité de coder des valeurs très grandes ou très petites ( grâce à l’exposant )
- Précision : l’erreur ne porte que sur la mantisse, elle est donc bornée
Inconvénient : Si le processeur ne possède pas d’unité de calcul pour traiter les flottants, le calcul prend plus de temps ( il faut traiter la matrice, l’exposant, faire des décalages, normaliser, etc.. ).