8
contributi
Nessun oggetto della modifica |
|||
Riga 7: | Riga 7: | ||
Molto spesso, abbiamo dei tasks proprietari che devo essere eseguiti in tempi precisi, o molto spesso. Se il task è eseguito da un processo, possiamo metterlo nel file ''crontab''. Se il processo invece è eseguito da un modulo del kernel, abbiamo due possibilità. La prima è di mettere un processo nel file '' crontab'', che attiverà il modulo tramite una system call quando necessario, per esempio aprendo un file. In ogni caso questo metodo è molto inefficente -- si esegue un nuovo precesso di crontab, si legge un nuovo eseguibile nella memoria, e tutto solo per attivare il modulo del kernel che è comunque caricato nella memoria. | Molto spesso, abbiamo dei tasks proprietari che devo essere eseguiti in tempi precisi, o molto spesso. Se il task è eseguito da un processo, possiamo metterlo nel file ''crontab''. Se il processo invece è eseguito da un modulo del kernel, abbiamo due possibilità. La prima è di mettere un processo nel file '' crontab'', che attiverà il modulo tramite una system call quando necessario, per esempio aprendo un file. In ogni caso questo metodo è molto inefficente -- si esegue un nuovo precesso di crontab, si legge un nuovo eseguibile nella memoria, e tutto solo per attivare il modulo del kernel che è comunque caricato nella memoria. | ||
Invece di fare in questo modo, si può creare una funzione che sarà chiamato una volta per ogni interrupt. La strada da seguire è creare un task, tenuto in una struttura workqueue_struct, che terrà un puntatore sulla funzione. Successivamente, si userà queue_delayed_work per mettere il task in una lista chiamata my_workqueue, che è la lista dei tasks che saranno eseguiti nel prossimo interrupt. Poichè vogliamo che la funzione | Invece di fare in questo modo, si può creare una funzione che sarà chiamato una volta per ogni interrupt. La strada da seguire è creare un task, tenuto in una struttura workqueue_struct, che terrà un puntatore sulla funzione. Successivamente, si userà queue_delayed_work per mettere il task in una lista chiamata my_workqueue, che è la lista dei tasks che saranno eseguiti nel prossimo interrupt. Poichè vogliamo che la funzione continui ad essere eseguita, abbiamo bisogno di metterela ancora nella lista my_workqueue ogni volta che è chiamata, per il prossimo interrupt. | ||
C'è ancora un punto che dobbiamo ricordare. Quando un modulo è rimosso tramite '''rmmod''', prima di tutto è controllato il conteggio delle sue referenze. Se è zero, è eseguito ''module_cleanup''.Poi, il modulo è rimosso dalla memoria con tutte le sue funzioni. E' importante arrestare correttamente un modulo,o potrebbero accadere imprevisti spiacevoli. Si guardi il codice sotto dove si nota come può essere fatto in un modo sicuro. | C'è ancora un punto che dobbiamo ricordare. Quando un modulo è rimosso tramite '''rmmod''', prima di tutto è controllato il conteggio delle sue referenze. Se è zero, è eseguito ''module_cleanup''.Poi, il modulo è rimosso dalla memoria con tutte le sue funzioni. E' importante arrestare correttamente un modulo,o potrebbero accadere imprevisti spiacevoli. Si guardi il codice sotto dove si nota come può essere fatto in un modo sicuro. | ||
Riga 13: | Riga 13: | ||
<pre> | <pre> | ||
/* | /* | ||
* sched.c - | * sched.c - programmare una funzione che sia chiamata ogni interrupt. | ||
* | * | ||
* Copyright (C) 2001 by Peter Jay Salzman | * Copyright (C) 2001 by Peter Jay Salzman | ||
Riga 19: | Riga 19: | ||
/* | /* | ||
* | * I file headers necessari | ||
*/ | */ | ||
/* | /* | ||
* Standard | * Standard kernel moduli | ||
*/ | */ | ||
#include <linux/kernel.h> /* | #include <linux/kernel.h> /* Stiamo facendo un lavoro sul kernel */ | ||
#include <linux/module.h> /* | #include <linux/module.h> /* In specifico, un modulo*/ | ||
#include <linux/proc_fs.h> /* | #include <linux/proc_fs.h> /* Necessario perchè usiamo il processo fs */ | ||
#include <linux/workqueue.h> /* | #include <linux/workqueue.h> /* Qui programmiamo i tasks */ | ||
#include <linux/sched.h> /* | #include <linux/sched.h> /* Abbiamo bisogno di sospendere e attivare successivamente */ | ||
#include <linux/init.h> /* Per__init ed __exit */ | |||
#include <linux/init.h> /* | #include <linux/interrupt.h> /* Per irqreturn_t */ | ||
#include <linux/interrupt.h> /* | |||
struct proc_dir_entry *Our_Proc_File; | struct proc_dir_entry *Our_Proc_File; | ||
Riga 39: | Riga 38: | ||
/* | /* | ||
* | * Il numero di volte che il timer interrupt sono stati chiamati | ||
*/ | */ | ||
static int TimerIntrpt = 0; | static int TimerIntrpt = 0; | ||
Riga 48: | Riga 47: | ||
/* | /* | ||
* | * Il lavoro di questa struttura queue per questo task, da workqueue.h | ||
*/ | */ | ||
static struct workqueue_struct *my_workqueue; | static struct workqueue_struct *my_workqueue; | ||
Riga 56: | Riga 55: | ||
/* | /* | ||
* | * Questa funzione sarà chiamata ogni interrupt. Notare il puntatore nullo* | ||
* | * - le funzioni task possono essere usate per più di un obbiettivo, tutte le volte | ||
* | * che questa avrà un diverso parametro. | ||
*/ | */ | ||
static void intrpt_routine(void *irrelevant) | static void intrpt_routine(void *irrelevant) | ||
Riga 75: | Riga 74: | ||
/* | /* | ||
* | * Metti un dato dentro il file del processo fs. | ||
*/ | */ | ||
ssize_t | ssize_t | ||
Riga 82: | Riga 81: | ||
off_t offset, int buffer_length, int *eof, void *data) | off_t offset, int buffer_length, int *eof, void *data) | ||
{ | { | ||
int len; /* | int len; /* Il numero di bytes usati al momento*/ | ||
/* | /* | ||
* | * E' statico quindi sarà ancora in memoria. | ||
* | * quando usciremo da questa funzione | ||
*/ | */ | ||
static char my_buffer[80]; | static char my_buffer[80]; | ||
/* | /* | ||
* | * Diamo tutte le nostre informazioni in one go /da sistemare/, cosi se qualcuno chiede | ||
se abbiamo più informazioni la risposta deve essere no. | |||
*/ | |||
if (offset > 0) | |||
return 0; | return 0; | ||
contributi