PB173 - Ovladače jádra - Linux VIII. Jiří Slabý ITI, Fakulta Informatiky 15. 11. 2011 J. Slabý (ITI) PB173/02 15. 11. 2011 1 /15 Přerušení LDD3 kap. 7 a 10 • Práce s časem • Pozdržení vykonávání kódu (spánek) • Časovače • Odložené vykonání kódu • Přerušení HW • Změny v kontextu přerušení • Zámky • Alokace J. Slabý (ITI) PB173/02 15.11.2011 2/ 15 Pozdržení vykonávání Čekání na událost trvající pevně danou dobu O Spánek • Zapojení plánovače • Rozlišení: 1-10 ms (podle hodnoty hz - jiffies) O Busy-waiting Smyčka • Rozlišení: ns API • linux/delay.h • ad 1. ssleep, msleep • ad 2. mdelay, udelay, ndelay V jádře lze čekat jen omezenou dobu J. Slabý (ITI) PB173/02 15.11.2011 3/ 15 Část I „Měkká" přerušení J. Slabý (ITI) PB173/02 15. 11. 2011 4/15 Časovače • Documentation/timers/* • Činnost v nastavených časech (i periodicky) • Kontext přerušení (nelze spát) • /proc/timer_iist (jen nová jádra) 2 druhy O Obyčejné • Fungují na každé architektuře • Založené na jiffies • Rozlišení: 1-10 ms O High-res • Jen pro vybrané architektury » Ve valné většině kódu není třeba • Rozlišení: ns J. Slabý (ITI) PB173/02 15.11.2011 5/ 15 Obyčejné časovače API • linux/timer.h, struct timer_list • DEFINE_TIMERS, setup_timerD • mod.timer (=del_timer+add_timer) • del_timer_sync static void my.fun (unsigned long data); static DEFINE.TIMER( my.timer , my.fun, 0, 30); static void my.fun (unsigned long data) { my.timer. data *= 2; mod-timer(&my_timer, jiffies + msecs.to.jiffies (data)) ; } mod-timer(&my_timer, jiffies + msecs.to.j if f ies (1 00)) ; del_timer_sync(&my_timer) ; Úkol: Každou vteřinu, kdy je modul v systému, vypsat hz a jiffies J. Slabý (ITI) PB173/02 15. 11. 2011 6/15 Odložené vykonání kódu Při potřebě vykonat kód, který nelze vykonat teď • Držím zámek, jsem v přerušení, ... • Nespecifikuje se pevně daný moment (na rozdíl od časovačů) 2 druhy O Workqueue O Tasklet Rozdíly « Rychlost zavolání • Kontext zavolání • Množství kódu (tj. rychlost vykonání) J. Slabý (ITI) PB173/02 15. 11. 2011 7/15 Workqueue • Speciální proces volající funkce ve frontě o Globální, společný pro všechny - většinou stačí • Vlastní - pro speciální případy • Lze v něm spát • Spustí se, až se plánovač rozhodne • Lze specifikovat minimální prodlevu • Documentation/workqueue.txt J. Slabý (ITI) PB173/02 15. 11. 2011 8/15 Workqueue API • linux/workqueue.h, struct work.struct a Definice funkce (práce): declare_works, init_workd 9 Globální proces: • Zařazení do fronty: schedule_work • Vyřazení z fronty (předčasně): cancei.work.sync • Vlastní proces: • Vytvoření procesu: create.workqueue, destroy_workqueue a Zařazení do fronty: queue.work • Vyřazení z fronty: cancei.work.sync (stejně) « Delayed (s min. prodlevou): *_deiayed_work Úkol: nahrát modul ieee80211 (requestjnodule). Nelze volat přímo v moduie_init a je nutný i vlastní proces J. Slabý (ITI) PB173/02 15. 11. 2011 9/15 Tasklet • Běží v kontextu přerušení (jako časovače) • Nelze spát a Musí být rychlý • Spustí se po příštím přerušení API • linux/interrupt.h, struct tasklet_struct • Definice funkce (taskletu): declare_tasklets, taskiet_initD • Zařazení do fronty: taskiet_scheduie a Vyřazení z fronty: taskiet_kiii Úkol: vytvořit tasklet, spustit a vypsat posloupnost volání pomocí dump_stack J. Slabý (ITI) PB173/02 15. 11. 2011 10/15 Část II „Tvrdá" přerušení J. Slabý (ITI) PB173/02 15. 11. 2011 11/15 Přerušení HW • HW informuje o změně stavu • Časovač tiknul, přišel paket, přečten blok z disku, ... • CPU přeruší chod programu/jádra a zavolá jádro • Cyklus? Priority přerušení! • OS musí obsloužit přerušení • Zjistí zdroj a zavolá odpovídající ovladač (jeho obsluhu přerušení) • Přerušeni je identifikováno číslem (IRQ) • Vynuluje přerušení na řadiči přerušení Nutná podpora HW (CPU, řadiče, sběrnice, zařízení) J. Slabý (ITI) PB173/02 15. 11. 2011 12/15 Přerušení v Linuxu • 1inux/interrupt.h • request_irq, free_irq • Flags: hlavně irqf_shared • Návratová hodnota z obsluhy: irq_none, irq_handled • /proc/interrupts static irqreturn.t my.handler( int irq , void *data) { /* data == my.data */ return my.device.raised.interrupt ? IRQ-HANDLED : IRQ.NONE; } static int my.probe( struct pci.dev *pdev, ...) { /* enable device etc. */ my.data = kmalloc (...) ; ret = request.irq (pdev->irq , my.handler, IRQF.SHARED, "my", my.data); } Úkol: vytvořte obsluhu pro Combo (zatím vracejte irq_handled) J. Slabý (ITI) PB173/02 15. 11. 2011 13/15 Kontext přerušení Změny • Zámky a Třeba používat *_irqf lags varianty • Jinak by mohlo dojít k deadlocku • Alokace • gfp_atomic • A tedy nevelké alokace • Kód • Rychlý, krátký • (Víc kódu později v taskletu, ještě víc ve workqueue) J. Slabý (ITI) PB173/02 15. 11. 2011 14/15 Úkol Práce s přerušením (součást domácího) O Vytvořit pojmenování (makra) pro registry (v tabulce) O Spustit 100 ms periodický časovač (pro každé Combo zařízení) • Vyvolat z Comba přerušení 0x1000 (registr 0x60) O Navázat se v probe na přerušení (request_irq) O Povolit v Combu přerušení 0x1000 (registr 0x44) O Implementovat obsluhu • Přečíst stav (registr 0x40), vypsat stav - limitované • Nenula: odsouhlasit přerušení (registr 0x64), vrátit irq_handled » Jinak: vrátit irq_none Offset Len R/W Contents Meaning 0x0040 4B R bitmap Raised interrupts 0x0044 4B R/W bitmap Enabled interrupts 0x0060 4B W bitmap Raise interrupts 0x0064 4B W bitmap Acknowledge interrupts Tabulka: Specifikace baru 0 (pokračování z minula) J. Slabý (ITI) PB173/02 15. 11. 2011 15/15