LKMPG: Pianificare compiti: differenze tra le versioni

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 sia portata in esecuzione, abbiamo bisogno di metterela ancora nella lista my_workqueue ogni volta che è chiamata, per il prossimo interrupt.
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 - scheduale a function to be called on every timer interrupt.
  *  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:


/*  
/*  
  * The necessary header files
  * I file headers necessari
  */
  */


/*  
/*  
  * Standard in kernel modules
  * Standard kernel moduli
  */
  */
#include <linux/kernel.h> /* We're doing kernel work */
#include <linux/kernel.h> /* Stiamo facendo un lavoro sul kernel */
#include <linux/module.h> /* Specifically, a module */
#include <linux/module.h> /* In specifico, un modulo*/
#include <linux/proc_fs.h> /* Necessary because we use the proc fs */
#include <linux/proc_fs.h> /* Necessario perchè usiamo il processo fs */
#include <linux/workqueue.h> /* We scheduale tasks here */
#include <linux/workqueue.h> /* Qui programmiamo i tasks */
#include <linux/sched.h> /* We need to put ourselves to sleep
#include <linux/sched.h> /* Abbiamo bisogno di sospendere e attivare successivamente */
  and wake up later */
#include <linux/init.h> /* Per__init ed __exit */
#include <linux/init.h> /* For __init and __exit */
#include <linux/interrupt.h> /* Per irqreturn_t */
#include <linux/interrupt.h> /* For irqreturn_t */


struct proc_dir_entry *Our_Proc_File;
struct proc_dir_entry *Our_Proc_File;
Riga 39: Riga 38:


/*  
/*  
  * The number of times the timer interrupt has been called so far
  * Il numero di volte che il timer interrupt sono stati chiamati
  */
  */
static int TimerIntrpt = 0;
static int TimerIntrpt = 0;
Riga 48: Riga 47:


/*  
/*  
  * The work queue structure for this task, from workqueue.h  
  * 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:


/*  
/*  
  * This function will be called on every timer interrupt. Notice the void*
  * Questa funzione sarà chiamata ogni interrupt. Notare il puntatore nullo*
  * pointer - task functions can be used for more than one purpose, each time
  * - le funzioni task possono essere usate per più di un obbiettivo, tutte le volte
  * getting a different parameter.
  * che questa avrà un diverso parametro.
  */
  */
static void intrpt_routine(void *irrelevant)
static void intrpt_routine(void *irrelevant)
Riga 75: Riga 74:


/*  
/*  
  * Put data into the proc fs file.  
  * 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; /* The number of bytes actually used */
int len; /* Il numero di bytes usati al momento*/


/*  
/*  
* It's static so it will still be in memory
* E' statico quindi sarà ancora in memoria.
* when we leave this function
* quando usciremo da questa funzione
*/
*/
static char my_buffer[80];
static char my_buffer[80];


/*  
/*  
* We give all of our information in one go, so if anybody asks us
* Diamo tutte le nostre informazioni in one go /da sistemare/, cosi se qualcuno chiede
* if we have more information the answer should always be no.
            se abbiamo più informazioni la risposta deve essere no.
*/
        */
if (offset > 0)
if (offset > 0)
return 0;
return 0;


8

contributi