lsmod - ¿Cómo detener los hilos del kernel de Linux en rmmod?
como cargar un modulo en linux (3)
Escribí el siguiente código para crear un hilo del núcleo:
#include<linux/init.h>
#include<linux/module.h>
#include<linux/kernel.h>
#include<linux/kthread.h>
#include<linux/sched.h>
struct task_struct *task;
int data;
int ret;
int thread_function(void *data)
{
int var;
var = 10;
return var;
}
static int kernel_init(void)
{
data = 20;
printk(KERN_INFO"--------------------------------------------");
task = kthread_create(&thread_function,(void *)data,"pradeep");
task = kthread_run(&thread_function,(void *)data,"pradeep");
printk(KERN_INFO"Kernel Thread : %s/n",task->comm);
return 0;
}
static void kernel_exit(void)
{
ret = kthread_stop(task);
}
module_init(kernel_init);
module_exit(kernel_exit);
Al dar el comando insmod, puedo crear un subproceso del kernel llamado "pradeep" y puedo ver el nuevo hilo usando el comando ps -ef
siguiente manera
root 6071 2 0 10:21 ? 00:00:00 [pradeep]
y su padre es kthreadd cuyo PID es 2. Pero no puedo detener este hilo al dar el comando rmmod
. Está dando la siguiente salida:
ERROR: Removing ''pradeep'': Device or resource busy.
¿Alguien por favor me puede decir cómo matar este hilo?
Debes usar solo uno de kthread_create()
o kthread_run()
:
/**
* kthread_run - create and wake a thread.
* @threadfn: the function to run until signal_pending(current).
* @data: data ptr for @threadfn.
* @namefmt: printf-style name for the thread.
*
* Description: Convenient wrapper for kthread_create() followed by
* wake_up_process(). Returns the kthread or ERR_PTR(-ENOMEM).
*/
#define kthread_run(threadfn, data, namefmt, ...) /
({ /
struct task_struct *__k /
= kthread_create(threadfn, data, namefmt, ## __VA_ARGS__); /
if (!IS_ERR(__k)) /
wake_up_process(__k); /
__k; /
})
Así que estás creando dos hilos y filtrando uno de ellos:
task = kthread_create(&thread_function,(void*) &data,"pradeep");
task = kthread_run(&thread_function,(void*) &data,"pradeep");
Además, a tu función de hilo le pueden faltar algunos detalles:
/**
* kthread_create - create a kthread.
* @threadfn: the function to run until signal_pending(current).
* @data: data ptr for @threadfn.
* @namefmt: printf-style name for the thread.
*
* Description: This helper function creates and names a kernel
* thread. The thread will be stopped: use wake_up_process() to start
* it. See also kthread_run().
*
* When woken, the thread will run @threadfn() with @data as its
* argument. @threadfn() can either call do_exit() directly if it is a
* standalone thread for which noone will call kthread_stop(), or
* return when ''kthread_should_stop()'' is true (which means
* kthread_stop() has been called). The return value should be zero
* or a negative error number; it will be passed to kthread_stop().
*
* Returns a task_struct or ERR_PTR(-ENOMEM).
*/
Creo que las dos opciones para terminar un hilo son:
- Llame a
do_exit()
cuando haya terminado. - Devuelve un valor cuando otro hilo llama a
kthread_stop()
.
Con suerte, después de solucionar estos dos pequeños problemas, tendrá un creador / segador de hilos funcional.
Espero que el siguiente programa resuelva su problema ... pulgar arriba :-)
`#include<linux/init.h>
#include<linux/module.h>
#include<linux/kernel.h>
#include<linux/kthread.h>
#include<linux/sched.h>`
struct task_struct *task;
int data;
int ret;
int thread_function(void *data)
{
int var;
var = 10;
printk(KERN_INFO "IN THREAD FUNCTION");
while(!kthread_should_stop()){
schedule();
}
/*do_exit(1);*/
return var;
}
static int kernel_init(void)
{
data = 20;
printk(KERN_INFO"--------------------------------------------");
/*task = kthread_create(&thread_function,(void *)data,"pradeep");*/
task = kthread_run(&thread_function,(void *)data,"pradeep");
printk(KERN_INFO"Kernel Thread : %s/n",task->comm);
return 0;
}
static void kernel_exit(void)
{
kthread_stop(task);
}
module_init(kernel_init);
module_exit(kernel_exit);
MODULE_AUTHOR("SHRQ");
MODULE_LICENSE("GPL");
en el código u no es necesario usar kthread_create
api como kthread_run
hace internamente ... Use cualquiera
task = kthread_create (& thread_function, (void *) data, "pradeep");
O
task = kthread_run (& thread_function, (void *) data, "pradeep");
Además, su módulo no está bajo licencia GPL. Esa podría ser una de las causas de sus problemas.