MSX Village forum

La Place des Développeurs Nombre à virgule fixe

aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 25/02/2021 à 16h36
ericb59 :
Peux tu expliquer la petite histoire s'il te plaît ?
Comment ça marche et comment utiliser un Q12.4 ?
Pas sur ce topic, mais peut être sur le topic Fusion-c ? ou sdcc ? :D


C'est pas spécifique au C, donc je fais un topic à part (je ferais un résumé sur le wiki après si ça semble utile).

Donc, un nombre à virgule fixe, c'est simplement un nombre réel (par ex. 3,42) qu'on va coder sur un nombre entier en lui appliquant un facteur (un multiplicateur).

Pour comprendre le principe, il suffit d'imaginer un programme qui utiliserait le mètre comme unité finale mais qui aurait besoin à un moment de pouvoir déplacer un objet de moins d’une unité, disons 0,2 m.
Vu qu'il serait bien trop lourd pour nos vénérables MSX de travailler avec de vrais nombres à virgule flottante (comme les float du C), l'astuce pourrait simplement être de stocker les nombres et de faire ses calculs sous la forme Val = mètres * 100 (en cm donc), puis de convertir en mètre quand c'est nécessaire (mètres = Val / 100). Du coup, pour ajouter 0,2 m, on ajouterait la valeur entière 0,2 * 100 (c-à-d 20). À l'inverse, le nombre entier 4213 représenterait en fait 42,13 m.
Un tel format serait un nombre à virgule fixe de 2 décimales (100).
Simple. :)

Comme les divisions/multiplications par 100 sont coûteuses, on va plutôt utiliser des facteurs puissance de 2 (1, 2, 4, 8, 16, etc.) comme base pour la partie fractionnelle. Du coup, les divisions/multiplications peuvent être remplacés par des décalages de bits qui sont énormément moins coûteux.
Sur un entier 8 ou 16-bits (les 32-bits sont trop coûteux pour le Z80), on va donc décider du nombre de bits qui serviront à coder la partie entière et de ceux (le reste) qui serviront à coder la partie fractionnelle.
C’est le format Q ! ^^ (https://en.wikipedia.org/wiki/Q_(number_format))

Par ex., un nombre 8-bits codé au format Q4.4, utilise 4-bits pour l’entier et 4-bits pour la fraction. Pour convertir une valeur en Q4.4, il suffit de décaler le nombre 4 fois vers la gauche (x << 4), Pour convertir un Q4.4 en valeur, il suffit de décaler le nombre 4 fois vers la droite (x >> 4). Si on l’utilise pour stocker un nombre non signé, on pourra coder dans un Q4.4 des nombres allant de 0 à 15,9375 (255/16) avec une précision de 0,0625 (1/16).
On peut utiliser avec le Q4.4 n’importe quelles opérations mathématiques de base tant que tous les termes sont au même format.

Il existe autant de Q qu’ils y a de combinaisons de bits (ne pas sortir cette phrase de son contexte, hein :oups).
Voici quelques exemples que j’utilise :
- Q2.6 : Entier 8-bits qui permet de coder des chiffres entre -1,0 et 1,0 avec une précision de 0,015625. Très utile pour stocker des vecteurs de direction par ex.
- Q8.8 : Entier 16-bits avec partie entière et fractionnelle sur 8 bits. Très performant à utiliser avec un nombre non-signé. Sinon, il faut prendre qq précautions.
- Q6.10 : Entier 16-bits avec partie entière de 6 bits et fractionnelle sur 10 bits. Permet de manipuler les données en Ko (avec un facteur 1024). Utile pour les petits nombre ayant besoin d’une très grande précision (~0,0001).

Pour mon projet Final Smash, j’ai choisi le Q12.4 car mon unité de base étant déjà en cm, je n’avais pas besoin de plus de 1/16e de cm de précision (et les petits shifts sont un peu moins coûteux).

On peut évidemment passer d'un Q à l'autre avec de simple décalage de bits (:oups).
Par ex. :
- Q4.4 << 2 ⇒ Q2.6
- Q8.8 >> 4 ⇒ Q12.4
(en cas de valeur signée négative c'est un peu plus compliqué car il faut préserver le signe)

J’utilise les virgules fixes pour mon outil de pré-génération de tables mathématiques : CMSXmath (https://github.com/aoineko-fr/CMSXmath). L’alliance de ces tables pré-calculées et de nombres à virgule fixe permet de faire des calculs mathématiques complexes en utilisant très peu de temps du Z80 (en contrepartie d’un coût en mémoire).

Voilà, j’espère avoir été exhaustif mais si vous avez des questions, hésitez pas.


On est toujours ignorant avant de savoir.
Github    
aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 25/02/2021 à 18h45
Voici l'ensemble des format Qm.n possibles pour les entiers 8 et 16-bits avec leur valeur de précision et leurs bornes :


Dispo aussi au format PDF Qm.n.pdf

Et pour être totalement complet, à noter que pour les entiers signés, certains excluent le bit de signe dans la notation.
Du coup, Q7.8 ⇒ 1-bit de signe, 7-bits entier, 8-bits fractionnel


On est toujours ignorant avant de savoir.
Github    
Metalion Membre non connecté

Conseiller Municipal

Rang

Avatar

Inscrit le : 23/12/2009 à 15h32

Messages: 1482

Le 25/02/2021 à 18h59
Il faut juste se souvenir que tout ça n'est qu'une représentation de l'esprit, un exercice intellectuel ... Cela ne devient en fait réellement concret que quand on arrive au stade final et que l'on veut garder uniquement un composant (généralement la valeur entière).

Travailler en Q4.4, par exemple, ne consistera en fait qu'à faire des opérations normales sur un entier 8-bits.
Seul le programmeur saura qu'il manipule en fait un nombre codé différemment.

Et lorsqu'à la fin, il voudra récupérer la valeur entière, il lui suffira de faire >>4 sur la valeur.
Avec en bonus, la possibilité d'utiliser le dernier bit sorti vers la droite pour faire l'arrondi.

Imaginons que le résultat d'une série d'opérations donne le nombre 58, soit en format binaire 00111010b. Ce nombre est en fait la représentation de 3,625 en Q4.4 (puisque 58/16=3,625). Lorsque l'on va faire >>4 sur cette valeur pour récupérer la valeur entière, il restera bien 3 dans le nombre (0011b), et le dernier bit sorti à droite sera 1. Ce qui peut être alors utilisé pour décider de faire l'arrondi vers l'entier supérieur (ou pas).

Exemple de l'arrondi vers l'entier supérieur en Q4.4 d'une valeur contenue dans 'a' :

Code :

; valeur de a >> 4
srl     a
srl     a
srl     a
srl     a

; ajout du carry (qui contient le dernier bit sorti à droite)
adc     0


Edité par Metalion Le 25/02/2021 à 21h07


MSX1: Daewoo DPC-200 / Yamaha CX5M
MSX2: Sony HB-F9P
MSXVR
Vidéo: V9990 (GFX-9)
Audio: MSX-Music (FM-PAC) / MSX-Audio (Audiowave) / OPL4 (Monster Sound FM Blaster) / OPNB (Neotron)
   
aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 25/02/2021 à 21h08
@Metalion, tu peux me confirmer que c'est bien les versions optimales des shifts droite/gauche :

Code ASM :
; a >> 1 (10cc)
srl    a
 
; a >> 2 (18cc)
rrca
rrca
and a, #0x3F
 
; a >> 3 (23cc)
rrca
rrca
rrca
and a, #0x1F
 
; a >> 4 (28cc)
rlca
rlca
rlca
rlca
and a, #0x0F
 
; a >> 5 (23cc)
rlca
rlca
rlca
and a, #0x07
 
; a >> 6 (18cc)
rlca
rlca
and a, #0x03
 
; a >> 7 (13cc)
rlca
and a, #0x01
 
; a << 1 (5cc)
add a, a
 
; a << 2 (10cc)
add a, a
add a, a
 
; a << 3 (15cc)
add a, a
add a, a
add a, a
 
; a << 4 (20cc)
add a, a
add a, a
add a, a
add a, a
 
; a << 5 (23cc)
rrca
rrca
rrca
and a, #0xE0
 
; a << 6 (18cc)
rrca
rrca
and a, #0xC0
 
; a << 7 (13cc)
rrca
and a, #0x80
 


Si c'est bien le cas, c'est marrant de voir que le choix du format de virgule fixe peut avoir un impact sur les performances.
Temps de conversion in/out :
- Q7.1 : 15cc
- Q6.2 : 28cc
- Q5.3 : 38cc
- Q4.4 : 48cc
- Q3.5 : 46cc
- Q2:6 : 36cc
- Q1:7 : 26cc


On est toujours ignorant avant de savoir.
Github    
ericb59 Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : compte ++ Groupe : Shoutbox

Inscrit le : 17/04/2012 à 10h25

Messages: 5481

Le 26/02/2021 à 07h35
Merci. Très bien expliqué. :top

Un ou deux exemples concret me manque pour comprendre comment ça s’utilise. J’ai besoin de visualiser ... ;)


banniere-ericb59e
Site web    
Metalion Membre non connecté

Conseiller Municipal

Rang

Avatar

Inscrit le : 23/12/2009 à 15h32

Messages: 1482

Le 26/02/2021 à 09h02
aoineko :
@Metalion, tu peux me confirmer que c'est bien les versions optimales des shifts droite/gauche :

J'ai pas regardé dans le détail, mais voici un tableau que j'avais fait il y a bien longtemps:



ericb59 :
Un ou deux exemples concret me manque pour comprendre comment ça s’utilise

Imaginons que tu doives calculer l'expression 'a x cos(b)', avec 'a' un entier 8 bits.

Tu sais que cos(b) est un nombre réel compris entre -1 et 1. On va utiliser une simple table qui va nous donner la valeur de cos(b) pour b. Partons du principe que les valeurs exactes -1 et 1 seront gérées séparément (par exemple à l'aide d'un test avant le calcul), et que l'on ne gère que les angles de 0° à 90°. Donc, on doit stocker des valeurs comprises entre 0 et 0,999999. Pour stocker ce genre de valeurs, le format Q0.8 non signé est parfait.

Prenons par exemple cos(45°) = racine carrée (2)/2 = 0.70710678118
Pour coder ce nombre en Q0.8 non signé, il suffit de le multiplier par 256 : 0.70710678118 x 256 = 181.019335984
On ne peut évidemment pas prendre les décimales, donc on ne stocke que la valeur entière : 181 (entier 8 bit).

Plaçons nous au moment du calcul, lorsque tu veux faire, par exemple, 33 x cos(45°) = 23.3345237792 :
- tu prends la valeur de la table pour 45° (181)
- tu la multiplies par 33
- tu obtiens donc un entier 16 bits dont la valeur est 5973

Cet entier 16 bits est ton résultat, codé en Q8.8.
Normal, puisqu'en fait, on a multiplié les 2 membres de l''expression par 256 : a x (256 x cos(b)) = 256 x résultat
5973 / 256 = 23.33203125, on a donc une erreur de 0,002, ce qui est acceptable.

Et pour obtenir la valeur entière, il suffit de garder l'octet haut du résultat.
5973d = 1723h, la valeur entière est donc : 17h = 23d. Edité par Metalion Le 26/02/2021 à 09h32


MSX1: Daewoo DPC-200 / Yamaha CX5M
MSX2: Sony HB-F9P
MSXVR
Vidéo: V9990 (GFX-9)
Audio: MSX-Music (FM-PAC) / MSX-Audio (Audiowave) / OPL4 (Monster Sound FM Blaster) / OPNB (Neotron)
   
aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 26/02/2021 à 10h16
Metalion :
aoineko :
@Metalion, tu peux me confirmer que c'est bien les versions optimales des shifts droite/gauche :

J'ai pas regardé dans le détail, mais voici un tableau que j'avais fait il y a bien longtemps:


On a bien les mêmes timing. :top


On est toujours ignorant avant de savoir.
Github    
aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 26/02/2021 à 11h47
Et pour les utilisateurs de C, je conseille de passer par des macros de conversion.
Par ex., pour Final Smash, j'utilise :
Code C :
// Unit conversion (Pixel <> Q10.6)
#define UNIT_TO_PX(a)                (u8)((a) / 64)
#define PX_TO_UNIT(a)                (i16)((a) * 64)
 


J'ai fait exprès de laisser les multiplications/divisions par 64 car du coup le compilateur va pouvoir adapter les opérations en fonction du type de donnée qu'on lui donne.
Si j'ai une variable int16 x qui contient un nombre au format Q10.6, je peux faire ce genre de chose :
Code C :
 
#define PI PX_TO_UNIT(3.14159265f) // Le compilateur va faire la conversion en float, puis convertir en int (précision optimale)
uint8 nbPixel = 10;
x += PX_TO_UNIT(nbPixel); // Le compilateur va faire la conversion via des décalages de bits (vitesse optimale)
if(x > 2*PI) // A noter une petite perte de précision car le x2 va être fait après la conversion en int
    x = PI;
 


L'autre gros avantage, c'est que tu peux changer la précision de ton unité sans devoir retoucher tout ton code.
Par ex., sur Final Smash, je suis passé de Q12.4 en Q10.6 en changeant juste les 2 macros. :)


On est toujours ignorant avant de savoir.
Github    
aoineko Membre non connecté

Conseiller Municipal

Rang

Avatar

Groupe : Shoutbox

Inscrit le : 02/01/2011 à 21h17

Messages: 2693

Le 26/02/2021 à 22h38
J'ai essayé de compiler toutes les infos qu'on a donné ici sur le wiki : http://msxvillage.fr/wiki/wiki.php?title=nombre-a-virgule-fixe

@Metalion, je me suis permis copier/coller tes infos ; j'espère que ça te dérange pas. :oups


On est toujours ignorant avant de savoir.
Github    
Metalion Membre non connecté

Conseiller Municipal

Rang

Avatar

Inscrit le : 23/12/2009 à 15h32

Messages: 1482

Le 27/02/2021 à 12h57
aoineko :
@Metalion, je me suis permis copier/coller tes infos ; j'espère que ça te dérange pas. :oups

Non, pas de soucis.


MSX1: Daewoo DPC-200 / Yamaha CX5M
MSX2: Sony HB-F9P
MSXVR
Vidéo: V9990 (GFX-9)
Audio: MSX-Music (FM-PAC) / MSX-Audio (Audiowave) / OPL4 (Monster Sound FM Blaster) / OPNB (Neotron)
   
Répondre
Vous n'êtes pas autorisé à écrire dans cette catégorie