| Version 15 (modified by , 2 years ago) (diff) |
|---|
DOCS [Start][Config][User][Kernel] — COURS [9] [9bis] [10] [10bis] [11] — TD [9][10][11] — TP [9][10][11] — ZIP [gcc...][9][10][11]
Gestionnaire d'interruptions
Rappel de cours
Il est fortement recommandé de lire les transparents, toutefois, nous avons mis ci-après quelques rappels utiles pour répondre aux questions du TD.
Contrôleurs de périphériques
- Dans cette séance, nous allons manipuler 3 contrôleurs de périphériques: le TTY que vous connaissez déjà et deux autres, l'ICU et le TIMER.
- Ces trois contrôleurs s'utilisent grâce à des registres mappés (c.-à-d. placés) dans l'espace d'adressage du MIPS.
- Les registres du TTY sont placés à partir de l'adresse
0xd0200000, - ceux de l'ICU à partir de l'adresse
0xd2200000 - et enfin ceux du TIMER à partir de l'adresse
0xd3200000.
- Les registres du TTY sont placés à partir de l'adresse
- Le rôle de ces registres est rappelé en partie dans ce texte et pour plus de détails, vous devez revoir le cours.
- Le choix des adresses de ces contrôleurs est fait par le créateur du matériel, elles ne peuvent pas être changées par le logiciel.
- Ces adresses sont données dans le fichier ldscript du kernel (
kernel.ld) parce qu'elles ne sont utilisables que si le MIPS est en mode kernel (adresses >0x80000000).
IRQ et ISR
- Les IRQ (Interrupt ReQuest)s sont des signaux électriques à 2 états (ON/OFF, on dit aussi Actif/Inactif ou encore Levé/Baissé).
- Les IRQ sont levés par les contrôleurs de périphériques pour prévenir d'un événement (fin de commande, arrivée d'une donnée, etc.).
- Quand une IRQ se lève, la conséquence est que le programme en cours d'exécution sur le processeur recevant l'IRQ est interrompu et qu'il est dérouté vers le noyau pour que ce dernier exécute l'ISR prévue pour l'IRQ.
- Les IRQ provoquent donc l'exécution d'ISR (Interrupt Service Routine) par le noyau.
- Les IRQ sont des signaux d'état qui doivent rester levés/activés tant qu'ils n'ont pas été acquittés par une ISR.
- Les ISR sont des fonctions qui reçoivent en argument un identifiant du contrôleur de périphérique qui a levé l'IRQ.
- Notez que ce n'est pas le processeur qui est interrompu, c'est bien le programme, car le processeur est seulement dérouté vers le noyau, mais il continue à travailler.
- Une ISR doit faire deux choses,
- accéder aux registres du contrôleur de périphérique concerné pour faire ce que le périphérique demande
- acquitter l'IRQ, c'est-à-dire demander au contrôleur de périphérique de baisser/désactiver son IRQ (puisque celle-ci a été traitée).
- La demande d'acquittement est spécifique à chaque contrôleur de périphérique.
- Pour le TTY, il faut lire le registre
TTY_READ. - Pour le TIMER, il faut écrire dans le registre
TIMER_RSTIRQ.
- Pour le TTY, il faut lire le registre
Routage et masquage des IRQ
- Sur le schéma de la plateforme des TP, on peut voir que seuls les composants TTY et TIMER peuvent lever des IRQ.
- Les IRQ de ces contrôleurs de périphériques sont envoyés au composant ICU qui va les combiner pour produire un unique signal IRQ pour le processeur.
- Une IRQ peut être masquée, c'est-à-dire que le processeur ne va pas interrompre le programme en cours.
- Le masquage peut être demandé à plusieurs endroits : dans le composant ICU et dans le processeur lui-même (parfois même dans le contrôleur de périphérique).
- Le masquage est demandé par le noyau, le plus souvent de manière temporaire, quand il doit exécuter un code critique qui ne doit surtout pas être interrompu.
- Sur le schéma ci-dessus, on voit que l'IRQ du TTY0 est reliée à l'entrée n°
10de l'ICU, c'est un choix matériel qui n'est pas modifiable par logiciel. Son état est donc enregistré dans le bit n°10 du registreICU_STATE. - Il y a un
ANDavec le bit10du registreICU_MASK. Si le bit10du registreICU_MASKest à0, alors la sortie duANDest0et l'IRQ est masquée (donc invisible pour le processeur). - Le registre
ICU_HIGHESTcontient toujours le numéro de l'IRQ active la plus prioritaire, comme il n'y en a qu'une dans cet exemple,ICU_HIGHESTcontient10(l'IRQ prioritaire, pour cette ICU, est l'IRQ active dont le numéro est le plus petit). - La sortie IRQ de l'ICU est reliée à l'entrée
0des 6 IRQs possibles du MIPS et sa valeur s'inscrit dans le registreHWI0du registrec0_cause. - Il y a un
ANDavec le bitHWI0du registrec0_status. Si le bitHWI0du registrec0_statusest à 0, alors la sortie duANDest0et l'IRQ est aussi masquée. - Enfin, il y a un dernier
ANDavec le bit0dec0_status(correspondant au bitIEpour Interrupt Enable) qui permet de masquer globalement les IRQ et avec leNOTdu bit1dec0_status(correspondant au bitEXLEXception Level). - Quand le signal IRQ vue par le MIPS s'active (passe à 1), c'est que l'IRQ levée par le contrôleur de périphérique doit être prise en charge. Le programme en cours d'exécution est interrompu et dérouté vers
kentryà l'adresse0x80000180et en même tempsC0_EPC ← PC+4,c0_cause.XCODE ← 0,c0_status.EXL ← 1(notez que le nom officiel dec0_statusestC0_SR, mais dans ce document, on utilisec0_statuspour plus de clarté).
Conséquence dans le noyau
- Dans le schéma ci-après, à gauche c'est le matériel et à droite c'est un extrait de la RAM contenant les structures de données utilisées par le noyau pour la gestion des IRQ.
- À gauche, on voit que les IRQ venant des contrôleurs de périphériques sont connectés aux entrées d'IRQ de l'ICU.
- Il y a 32 entrées possibles. Sur notre plateforme, par exemple l'IRQ du TTY2 est connectée à l'entrée
12de l'ICU. Ce numéro d'entrée est le numéro qui identifie le contrôleur de périphérique. - Notez que le registre
ICU_MASKest en lecture seule, c'est-à-dire qu'il ne peut pas être écrit directement. Pour modifier le contenu du registreICU_MASK, il faut utiliser deux autres registres de l'ICU:ICU_SETetICU_CLEAR.ICU_SETpermet de mettre à1les bits deICU_MASK, etICU_CLEARpermet de les mettre à0. Pour mettre à1le bitidu registreICU_MASK, il faut écrire1dans le bitidu registreICU_SET. Pour mettre à0le bitjdu registreICU_MASK, il faut aussi écrire1dans le bitjdu registreICU_CLEAR.
- Il y a 32 entrées possibles. Sur notre plateforme, par exemple l'IRQ du TTY2 est connectée à l'entrée
- À droite, il y a les deux tableaux que le noyau utilise pour connaitre l'ISR à exécuter pour chaque numéro d'IRQ.
- Ce couple de tableaux se nomme vecteur d'interruption et comme il y a 32 entrées d'IRQ dans l'ICU, ces tableaux ont 32 cases chacun.
- Ici, le vecteur d'interruption est composé des tableaux
IRQ_VECTOR_ISR[]etIRQ_VECTOR_DEV[]. - Le vecteur d'interruption est indexé par les numéros d'IRQ. Il contient deux informations:
- dans la case n°
idu tableauIRQ_VECTOR_ISR[], on trouve le pointeur sur la fonction ISR à appeler si l'IRQ n°iest levée - dans la case n°
idu tableauIRQ_VECTOR_DEV[], on trouve le numéro de l'instance du périphérique.
- dans la case n°
- Cette dernière information est nécessaire dans le cas des contrôleurs de périphérique multi-instances comme le TTY afin de savoir quel jeu de registres la fonction ISR doit utiliser.
- En effet, il y a une fonction ISR unique à exécuter quel que soit le numéro du TTY. Dans cette plateforme, comme il y a 4 TTY, l'adresse de la fonction ISR est placée dans les cases
10,11,12, et13du tableauIRQ_VECTOR_ISR[]. Dans les cases10,11,12, et13du tableauIRQ_VECTOR_DEV[], on a les valeurs0,1,2et3qui correspondent bien au numéro d'instance des TTYs.
Rappel sur les 3 registres du coprocesseur 0 impliqués
- Nous rappelons ci-après les 3 registres du coprocesseur
0utilisés au moment de l'entrée dans le noyau, quelle que soit la cause : syscall (vu la semaine dernière), interruption (TD de cette semaine) et exception (dans le cas de problèmes lors de l'exécution du programme comme la division par 0). - On rappelle aussi que les seules instructions qui peuvent manipuler ces registres sont
mtc0etmfc0pour, respectivement, les écrire et les lire.
- Les bits
HWI0des registresc0_status(aussi nomméc0_sr) etc0_causecontiennent respectivement le mask et le l'état de l'entrée n°0d'interruption du MIPS. - Les bits
UM,IEetEXLsont liés au mode d'exécution du MIPS:UMest le bit de mode du MIPS (1=User Mode,0=Kernel Mode),IEest le bit de masque général des interruptions (1=autorisées,0=masquées)- et enfin
EXLest le bit que le MIPS met à1à l'entrée dans le noyau pour informer d'un niveau exceptionnel et dans ce cas les bitsUMetIEne sont plus significatifs, siEXLest à1alors le MIPS est en mode kernel, et les interruptions sont masquées.
Questions
La majorité des réponses aux questions ci-après sont dans le rappel du cours donné au début de cette page, c'est voulu.
- À quelles adresses dans l'espace d'adressage sont placés les registres des 3 contrôleurs de périphériques de la plateforme et comment le kernel les connaît ?
- Que signifie l'acronyme I.R.Q. ?
- Une IRQ est un signal électrique, combien peut-il avoir d'états ?
- Qu'est-ce qui provoque une IRQ ?
- Les IRQ relient des composants sources et des composants destinataires, quels sont ces composants ? Donnez un exemple.
- Que signifie masquer une IRQ ?
- Quels composants peuvent masquer une IRQ ?
- Est-ce qu'une application utilisateur peut demander le masquage d'une IRQ ?
- Que signifie l'acronyme I.S.R. ?
- Dans la plateforme des TPs, sur quelles entrées de l'ICU sont branchées les IRQ venant des TTYs et du TIMER ?
- Quelle valeur faut il avoir dans le registre
ICU_MASKsi on veut recevoir seulement les IRQ venant des 4 TTYs, dans le cas de la plateforme utilisée en TP ? Donnez le nombre en binaire et en hexadécimal. - L'écriture dans
ICU_MASKn'est pas possible, comment modifier ce registre pour mettre à 1 le bit10? - Sur une plateforme (autre que celle des TP) sur laquelle on aurait un TTY0 sur l'entrée 5, un TIMER sur l'entrée 2, et un autre TTY1 sur l'entrée 14. Que doit-on faire pour que seuls le TTY1 et le TIMER soient démasqués et que TTY0 soit masqué ?
- Si on définit dans le code C:
Ecrivez la fonction mettant à 1 le bit n°
struct icu_s { int state; // état des IRQ à l'entrée de l'ICU int mask; // masque des IRQ int set; // registre de mise à 1 des bits du registre mask int clear; // registre de mise à 0 des bits du registre mask int highest; // numéro de l'IRQ active la plus prioritaire int unused[3]; // 3 registres non utilisés }; extern volatile struct icu_s __icu_regs_map; // déclaration de cette struct pour gcc
irqdu registreICU_MASK:void icu_set_mask (int irq) - Si les 3 IRQ de la question précédente se lèvent au même cycle, quelles seront les valeurs des registres
ICU_STATE,ICU_MASKetICU_HIGHEST? - Dans quel mode est le processeur quand il traite une IRQ ?
- Que fait le processeur lorsqu'il reçoit une IRQ masquée ?
- Que signifie acquitter une IRQ ?
- Qui demande l'acquittement à qui ?
- Comment demande-t-on l'acquittement ? Donnez au moins un exemple.
- Est-ce qu'une IRQ peut se désactiver sans intervention du processeur ?
- Est-ce qu'une IRQ peut ne pas être attendue par le noyau ?
- Quelle est la valeur du champ
XCODEdu registrec0_causeà l'entrée dans le noyau en cas d'interruption ? - Quelle est la valeur écrite dans le registre
c0_EPCà l'entrée dans le noyau en cas d'interruption ? - Que se passe-t-il dans le registre
c0_statusà l'entrée dans le noyau en cas d'interruption et quelle est la conséquence ? - La routine
kentry(entrée du kernel à l'adresse0x80000180) appelle le gestionnaire d'interruption quand le MIPS reçoit une IRQ non masquée, que fait ce gestionnaire d'interruption ? - À l'entrée dans le noyau,
kentryanalyse le champXCODEdu registre dec0_causeet si c'est0alors il saute au code donné ci-après (ce n'est pas exactement le code que vous pouvez voir dans les fichiers sources pour que ce soit plus facile à comprendre).Pourquoi, ne pas sauver les registres persistants ?irq_handler: addiu $29, $29, -23*4 // 23 registers to save (18 tmp regs+HI+LO+$31+EPC+SR) mfc0 $27, $14 // $27 <- EPC (addr of syscall instruction) mfc0 $26, $12 // $26 <- SR (status register) sw $31, 22*4($29) // $31 because, it is lost by jal isrcall sw $27, 21*4($29) // save EPC (return address of IRQ) sw $26, 20*4($29) // save SR (status register) mtc0 $0, $12 // SR <- kernel-mode without INT (UM=0 ERL=0 EXL=0 IE=0) sw $1, 1*4($29) // save all temporary registers including HI and LO sw $2, 2*4($29) [etc. pour les autres sauvegardes des registres temporaires] jal isrcall // call the irq handler fonction écrite en C lw $1, 1*4($29) // restore all temporary registers including HI and LO lw $2, 2*4($29) [etc. pour les autres restaurations des registres temporaires] lw $26, 20*4($29) // get old SR lw $27, 21*4($29) // get return address of syscall lw $31, 22*4($29) // restore $31 mtc0 $26, $12 // restore SR mtc0 $27, $14 // restore EPC addiu $29, $29, 23*4 // restore the stack pointer eret // jr C0_EPC AND C0_SR.EXL <= 0
- La fonction
isrcall()a pour mission d'appeler la bonne ISR. Dans le code qui suit (extrait du fichierkernel/harch.c), on voit d'abord la déclaration de la structure qui décrit les registres présents dans l'ICU. En fait c'est un tableau de structures parce qu'il y a autant d'instances d'ICU que de processeurs (donné par NCPUS), ici, il y a un seul processeur MIPS, donc NCPUS=1.La déclarationstruct icu_s { int state; // state of all IRQ signals int mask; // IRQ mask to chose what we need for this ICU int set; // IRQ set --> enable specific IRQs for this ICU int clear; // IRQ clear --> disable specific IRQs for this ICU int highest; // highest pritority IRQ number for this ICU int unused[3]; // these 3 registers are not used }; extern volatile struct icu_s __icu_regs_map[NCPUS]; static int icu_get_highest (int icu) { return __icu_regs_map[icu].highest; } static void icu_set_mask (int icu, int irq) { __icu_regs_map[icu].set = 1 << irq; } void isrcall (void) { int irq = icu_get_highest (cpuid()); irq_vector_isr[irq] (irq_vector_dev[irq]); }
extern volatile struct icu_s __icu_regs_map[NCPUS];informe le compilateur que le symbole__icu_regs_mapest défini ailleurs et que c'est un tableau de structures de typestruct icu_s. Ainsi, le compilateurgccsait comment utiliser la variable__icu_regs_map.
Dans quel fichier est défini__icu_regs_map? Que font les fonctionsicu_get_highest(),icu_set_mask()etisrcall()? Comment s'appelle le couple de tableauxirq_vector_isr[irq]etirq_vector_dev[irq]et combien de cases ont-ils ? - Si
ICU_HIGHESTcontient 10 (dans le cas de notre plateforme) que doit faire la fonctionisrcall() - Que fait la fonction
icu_set_mask (int icu, int irq)? - Les registres du TIMER sont définis dans le code du noyau de la façon suivante :
Écrivez le code de la fonction
struct timer_s { int value; // timer's counter : +1 each cycle, can be written int mode; // timer's mode : bit 0 = ON/OFF ; bit 1 = IRQ enable int period; // timer's period between two IRQ int resetirq; // address to acknowledge the timer's IRQ }; extern volatile struct timer_s __timer_regs_map[NCPUS];
static void timer_init (int timer, int tick)qui initialise la période du timer n°timeravec l'entier nomméticket active les IRQ si la période donnée est non nulle. - La configuration des périphériques et des interruptions est faite dans la fonction
arch_init()appelée parkinit().
Écrivez les instructions C permettant d'ajouter le TIMER dans le noyau avec un tick de 1000000 (1 million de cycles). Il faut (1) initialiser le timer ; (2) démasquer l'IRQ venant du timer dans l'ICU, elle connectée sur son entrée n°0 ; (3) initialiser le vecteur d'interruption avec la fonctiontimer_isrpour ce timer0.
La suite en TP...
