I am new to kernel module programming and for my work I need to write a multithreaded kernel module. So I tried some primary uses of kernel threads. I wrote the following. It is supposed to print 1 in one thread and 2 in another thread, both 10 times.
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/kernel.h>
#include <linux/udp.h>
#include <linux/mm.h>
#include <linux/init.h>
#include <linux/kthread.h>
struct task_struct *task1;
struct task_struct *task2;
static void thread_func(void* data)
{
int *n;
n = (int *)data;
int i = 0;
while(i < 10){
printk("%d\n", *n);
i++;
}
//do_exit();
}
static int t_start(void)
{
printk("Module starting ... ... ..\n");
int *p1, *p2;
int one = 1, two = 2;
p1 = &one;
p2 = &two;
task1 = kthread_run(&thread_func, (void*)p1, "thread_func_1");
task2 = kthread_run(&thread_func, (void*)p2, "thread_func_2");
return 0;
}
static void t_end (void)
{
printk("Module terminating ... ... ...\n");
kthread_stop(task1);
kthread_stop(task2);
}
module_init(t_start);
module_exit(t_end);
MODULE_AUTHOR("Md. Taufique Hussain");
MODULE_DESCRIPTION("Testing kernel threads");
MODULE_LICENSE("GPL");
But I am facing following problems. -
What are the problems? How can I solve them.
1.Don't use these code, delete them
// kthread_stop(task1);
// kthread_stop(task2);
it seems after the termination of the thread, task will be set to null, then calling kthread_stop() will cause a null pointer error
2.Don't pass local variable to threads, use global variable instead.
3.If you want two threads switch between each other, use "wait_event" and "wake_up" function. Here is my code that work.
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/sched.h>
#include <linux/kthread.h>
#include <linux/wait.h>
MODULE_LICENSE("GPL");
int pid1 = 1;
int pid2 = 2;
DECLARE_WAIT_QUEUE_HEAD(wq);
int condition;
struct task_struct *task1;
struct task_struct *task2;
static int thread_function(void *data){
int *thread_id = (int*)data;
int i = 0;
while(i < 10){
printk(KERN_INFO "install kernel thread: %d\n", *thread_id);
i++;
if(*thread_id == 1)
{
wait_event(wq, condition == 0xA);
condition = 0xB;
wake_up(&wq);
}
else{
wait_event(wq, condition == 0xB);
condition = 0xA;
wake_up(&wq);
}
}
return 0;
}
static int kernel_init(void)
{
condition = 0xA;
task1 = kthread_create(&thread_function, (void *)&pid1, "pradeep");
task2 = kthread_create(&thread_function, (void *)&pid2, "pradeep");
//printk(KERN_INFO "After create kernel thread\n");
wake_up_process(task1);
wake_up_process(task2);
return 0;
}
int init_module(void)
{
kernel_init();
return 0;
}
void cleanup_module(void)
{
return;
}
Add schedule
call to force thread scheduling.
static void thread_func(void* data)
{
int *n;
n = (int *)data;
int i = 0;
while(i < 10){
printk("%d\n", *n);
schedule();
i++;
}
//do_exit();
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With