Le TP est à rendre, avant le mardi 11 novembre minuit...
Le rendu de TP se fera uniquement à travers l'interface web TPLab. Vous devrez fournir un unique fichier .c
contenant le code des fonctions demandées.
Vous pouvez utiliser le fichier Makefile
fournit pour faciliter la compilation.
Certaines questions appellent à une réponse que vous pouvez mettre en commentaire dans votre fichier principal.
Attention : les points suivants ne rapportent rien, mais ne pas les respecter pourra retrancher jusqu'à 10 points sur la note finale :
tp1-nom.c
",
-Wall -Werror -Wextra -pedantic -std=c99
,
Le seul fait que votre programme fonctionne ne suffit pas pour avoir une bonne note. Les points suivants seront pris en compte :
Les QR codes (Quick Response Codes) sont un type de code barres à 2 dimensions développés par Toyota en 1994. L'objectif original était de pouvoir suivre les pièces de véhicules (à travers notamment de numéros de séries) lors du processus de construction. Ils sont maintenant largement utilisés à des fins commerciales pour permettre aux utilisateurs de smartphone d'accéder facilement à des adresses web automatiquement. Les spécifications des QR codes font l'objet d'une norme ISO : ISO/IEC 18004:2006 et ils sont dans le domaine public.
Par exemple, le QR code suivant contient le texte "INFO528 - TP1 : QR codes
" :
Il existe d'autres types de codes barres bidimensionnels :
Tous ces codes incorporent des codes correcteurs d'erreurs permettant de retrouver l'information même lorsque qu'une partie du code est abimée / manquante. Par exemple, cette version du QR code précédent contient encore le texte "INFO528 - TP1 : QR codes
" malgré la déformation et les parasites !
Dans ce TP, nous allons commencer par décoder les petits QR codes, sans nous occuper de la correction des erreurs. Les petits QR codes (version 1) peuvent uniquement contenir au plus 17 caractères ASCII, ou 25 caractères alphanumériques (lettres majuscules / chiffres / ponctuation) ou 41 chiffres. Voici quelques exemples :
ASCII | alphanumérique | chiffres |
"Vive les maths ! " |
"INFO528 : QRCODES ETC. " |
"31415926535 " |
Le TP sera fait en utilisant le langage C. L'archive du TP contient quelques définitions de types et fonctions que vous devrez utiliser. Le fichier "qrcodes.h
" contient les entêtes des fonctions avec des commentaires. Reportez vous y en cas de problèmes.
Les points importants sont :
netpbm
" définit un type "bit
" pouvant contenir "0
" ou "1
".
bit
s bidimensionnel, c'est à dire par un double pointeur. Le type "QRCode
" est un synonyme pour ce type. Attention, un QR code est un tableau de lignes, ce qui implique que si "qr
" est de type "QRCode
", alors "qr[i][j]
" est le bit contenu sur la ligne "i
" et la colonne "j
".
1
" est représenté par une case noire, et le bit "0
" par une case blanche.
alloc_qrcode
" permet de réserver la mémoire pour un QR code. Elle prend la taille du QR code en argument. Dans notre cas, vous devrez utiliser la constante "SIZE_V1
" comme taille (cette constante vaut 21 pour les QR codes de version 1).
N'oubliez pas de libérer la mémoire avec "free_qrcode
" lorsque vous n'avez plus besoin d'un QR code.
PBM
, vous pouvez la convertir en un QR code avec la fonction "pbm_to_qrcode_version1
". Reportez-vous aux commentaires dans le fichier "qrcodes.h
" pour plus de détails
qrencode
". Par exemple, l'exemple précédent à été créé avec
$ qrencode -o exemple-ASCII.png 'Vive les maths !'
qrencode
" pour plus de détails.
Attention, la commande "qrencode
" crée des images au format PNG. Il faut donc les convertir au format PBM avec une commande supplémentaire :
$ convert exemple-ASCII.png exemple-ASCII.pbm
qrcode_version1_to_pbm
" permet de transformer un QR code (en C) en une image au format PBM.
Les QR codes comportent tous des parties fixes qui ne contiennent aucune information. Pour les QR codes de version 1, les partie fixes, avec leurs couleurs, sont :
La fonction "check_fixed_bits_version1
" compte le nombre d'erreurs dans les bits fixes. Elle prends en argument un "QRCode
".
Complétez le code de la fonction
int check_fixed_bits_version1(QRCode qr);
Pour tester, vous devez :
do_test
" du fichier "tests.c
",
$ ./qrcodes -T ...
Par défaut, la fonction "do_tests
" appelle la fonction "test_check_fixed
" sur les trois fichiers d'exemples. Vous devriez obtenir le résultat suivant :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
nombre d'erreurs | 0 | 3 | 0 |
Par la suite, vous pourrez écrire de nouvelles fonctions sur le modèle de "test_check_fixed
" pour tester vos fonctions.
Le format (taux de correction d'erreurs et numéro du masque à appliquer) est contenu sur 5 bits de données + 10 bits de redondance, soit un total de 15 bits numérotés de 0
à 14
. Comme ces 15 bits sont très importants, il sont stockés à deux endroits dans le QR code :
Écrivez la fonction
void get_format_bits(QRCode qr, bit format[2][15]);
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
format -1 |
110110001000001 |
111110110011010 |
001001110111110 |
format -2 |
110110001000001 |
111110111101110 |
001001110111110 |
Pour éviter que les 15 bits de formats soit égales à 000000000000000
ou 111111111111111
, on applique également un masque aux 15 bits du format avant de le stocker. Le masque appliqué est donné par 101010000010010
: le i-ème bit du format est modifié exactement lorsque le i-ème bit du masque est un "1
".
Le plus simple pour appliquer un masque est d'utiliser l'opération XOR (notée "^
" en C) entre les bits du format et les bits du masque. Ceci fonctionne car un XOR entre un bit b et "0
" donne simplement b, alors qu'un XOR entre b et "1
" renvoie le complément de b.
supprimer la fonction get_raw_format_bits
des fichiers
Ajoutez quelques lignes dans la fonction
void get_format_bits(QRCode qr, bit format[2][15]);
pour appliquer le masque avant de renvoyer le résultat.
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
format -1 |
011100001010011 |
010100110001000 |
100011110101100 |
format -2 |
011100001010011 |
010100111111100 |
100011110101100 |
Vous pouvez utiliser la fonction "print_bits_array
" (c.f. le fichier "qrcodes.h
" pour afficher un tableau de bits.
Pour le moment, nous ne pouvons pas corriger les erreurs dans les 15 bits du format, mais nous pouvons repérer (au moins) une erreur en comparant les deux copies du format. Nous allons vérifier qu'il n'y a pas eu d'erreur et récupérer les bits 0 et 1 pour le taux de correction, ainsi que les bits 2, 3 et 4 pour le numéro du masque.
Pour transformer un tableau de 3 bits en valeur de type "int
", il suffit de faire
n = T[2]*4 + T[1]*2 + T[0];
(C'est presque pareil pour un tableau de 2 bits !)
Écrivez la fonction correspondante :
int get_format(QRCode qr, int *correction_level, int *mask);
Elle
correction_level
" et le numéro du masque dans "mask
" (passage par adresse),
-1
" s'il y avait une erreur dans le format (deux copies non-identiques) et renvoie "0
" sinon.
Attention à ne pas confondre les bits de poids faible et les bits de poids fort : si les bits 0 à 4 du format sont 10110
, le taux de correction est 10
, c'est à dire 2
et le numéro du masque est 110
, c'est à dire 6
.
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
formats identiques | oui | non | oui |
taux / masque | 1 / 6 | 1 / 2 | 2 / 1 |
Comme le format, les données sont masquées pour éviter des grandes plages uniformes (et d'autres motifs qui pourraient empêcher de lire facilement le QR code). Alors que le format était toujours masqué avec le même masque, les données peuvent utiliser 8 masques différents :
bits masqués | ||||
masque | 000 |
001 |
010 |
011 |
formule | (i+j)%2 == 0 |
i%2 == 0 |
j%3 == 0 |
(i+j)%3 == 0 |
bits masqués | ||||
masque | 100 |
101 |
110 |
111 |
formule | (i/2+j/3)%2 == 0 |
(i*j)%2+(i*j)%3 == 0 |
((i*j)%3+i*j)%2 == 0 |
((i*j)%3+i+j)%2== 0 |
Programmez la fonction
void data_mask(QRCode qr, int m);
qui applique le masque approprié sur le QR code "qr
".
Remarque : comme on ne se servira plus du format ou des parties fixes du QR code, on peut masquer le QR code entier sans se préoccuper des coordonnées.
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
fichier masqué |
La fonction existante "qrcode_version1_to_pbm
" permet de transformer un QR code en image. Elle prend 3 arguments :
"test1-masque.pbm"
),
10
),
QRcode
à transformer.
La zone de données d'un QR code est découpée en octets que l'on lit de la manière suivante :
Notez bien les points suivants :
Le type "byte
" est défini comme un synonyme de unsigned char
. Il s'agit donc simplement d'un nombre entre 0
et 255
, codé sur 8 bits. On peut facilement convertir un tableau de 8 bits en un octet avec les opérations bit à bit en utilisant la méthode détaillée plus haut.
Programmer les deux fonctions (très similaires) qui permettent de récupérer un octet de données.
byte get_upward_byte(QRCode qr, int i, int j); byte get_downward_byte(QRCode qr, int i, int j);
Les arguments "i
" et "j
" correspondent aux coordonnées (ligne/colonne) du bit en haut à gauche du rectangle correspondant à l'octet. Par exemple, pour lire le premier octet, on appellera la fonction "get_upward_byte
avec i=17
et j=19
.
Attention, n'oubliez pas les deux cas particuliers où l'octet est découpé en deux morceaux. Ils correspondent aux cas où i==4
et j==11
(en montant) et i==4
et j==9
en descendant.
Vous pouvez maintenant utiliser la fonction
void get_data_bytes(QRCode qr, byte *data);
(déjà définie) pour lire tous les octets d'un QR code. (Cette fonction appelle les fonctions "get_upward_byte
" et "get_downward_byte
".)
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
données | 41 05 66 97 66 52 06 c6 57 32 06 d6 17 46 87 32 02 10 ec a6 2a c7 d8 91 0a e6 |
20 b3 41 57 63 8c c6 7e 09 5a 8d 12 bd 10 52 69 18 00 ec 29 f4 34 f9 c3 4e e1 |
10 2d 3a 27 d0 94 60 ec 11 61 7a 85 11 1f fd 16 b0 29 47 41 72 7b 42 bc 84 6e |
La macro "BIT(u,i)
" permet de récupérer le i
-ème bit (en partant de la droite) d'un nombre positif u
. Par exemple, BIT(10,0) --> 0
et BIT(10,1) --> 1
car 10
est 00001010
en base 2. Cette macro est définie dans qrcodes.h
comme un synonyme de u>>i & 1
.
Nous allons avoir besoin de récupérer des bits arbitraires dans un tableau d'octets. Il faut donc aller chercher le bit approprié dans la case appropriée :
Écrivez la fonction
bit get_bit(const byte *data, int nb) ;
qui donne le nb
-ème bit de données.
Attention, à l'intérieur d'un octet, les bits sont numérotés à partir de la gauche ! Pour obtenir le tout premier bit de données, il faut donc utiliser BIT(data[0],7)
.
Il est maintenant facile de récupérer quelques bits au milieu des données et de les combiner en un nombre entier comme vu plus haut (avec les opérations bit à bit).
Écrivez la fonction
unsigned int get_bits(const byte *data, int offset, int nb);
qui renvoie l'entier correspondant aux nb
bits commençant à offset
dans les données.
Par exemple, pour récupérer le mode de codage (4 premiers bits), il faudra appeler
code = get_bits(data, 0, 4);
et dans le cas d'octets, pour récupérer le nombre d'octets (taille du message, codée sur 8 bits), il faudra faire
taille = get_bits(data, 4, 8);
Les premiers bits de données donnent :
Les fonctions
int get_mode(const byte *data, int offset); int get_length(const byte *data, int offset, int m);
sont déjà écrites : elles permettent de récupérer le mode de codage ainsi que le nombre de caractères du message. Le paramètre "offset
" peut être ignoré pour le moment : il doit valoir 0
pour toutes les applications de ce TP.
Pour tester :
fichier | exemple-ASCII.pbm |
exemple-alphanumerique.pbm |
exemple-numerique.pbm |
mode / taille | 4 / 16 | 2 / 22 | 1 / 11 |
Nous avons maintenant tous les ingrédients nécessaires pour récupérer le message complet : il suffit d'appeler la fonction get_bits(data, ..., 8)
pour récupérer chaque octet au bon endroit dans les données.
Attention : pour l'ASCII, les vraies données commencent au bit 12, car les bits de 0 à 11 servent à donner le mode et le nombre de caractères. Le premier octet de données est donc obtenu avec get_bits(data, 12, 8)
.
Écrivez la fonction
int decode_bytes(const byte *data, int offset, int nb, char *result) ;
qui va chercher chaque charactère (octet) dans les données et les stocke dans la chaîne result
. L'entier renvoyé correspond au nombre de charactères lus.
Attention, la chaine de caractères "result
" doit avoir une taille d'au moins nb+1
car le dernier octet d'une chaine est forcément le caractère nul ("'\0'
").
N'oubliez donc pas de mettre ce caractère en fin de chaine !
Pour tester :
fichier | exemple-ASCII.pbm |
message | "Vive les maths ! " |
(Bonus)
Écrivez les fonction permettant de décoder le mode numérique et le mode alphanumérique
int decode_numeric(const byte *data, int offset, int nb, char *result) ; int decode_alphanumeric(const byte *data, int offset, int nb, char *result) ;
Note : le codage des modes numérique et alpha-numérique est expliqué en commentaire dans le corps des fonctions "decode_numeric
" et "decode_alphanumeric
".
Pour tester :
fichier | exemple-alphanumerique.pbm |
exemple-numerique.pbm |
message | "INFO528 : QRCODES ETC. " |
"31415926535 " |
(Bonus)
Après un message de taille t, il peut y avoir une suite codée avec un autre mode. Il suffit de prendre les 4 bits suivant pour avoir le mode et continuer comme précédemment. Lorsque le mode devient "0000
", c'est que le message est vraiment fini.
Modifier la fonction
int decode_bytes(const byte *data, int offset, int nb, char *result);
pour qu'elle décode aussi les messages multi-modes.
Pour tester :
fichier | exemple-2modes.pbm |
message | "info 528 -- TP1 " |
La fonction principale a deux modes de fonctionnement :
-d
",
-i
".
Vous pouvez utiliser l'argument "-h
" pour avoir un petit message d'aide sur les autres arguments.
Le mode de décodage appelle la fonction "decode_file
" et le mode informatif appelle la fonction "info_file
"...
Complétez les deux fonctions
void info_file(const char *filename, int margin_size, int module_size); void decode_file(const char *filename, int margin_size, int module_size);
et testez le programme sur des petits QR codes que vous générerez avec la commande "qrencode
".