内核线程
2012-06-18 08:59:06
分类: LINUX
在linux内核中,所谓的内核线程实际上是一个共享父进程地址空间的进程,它有自己的系统堆栈;所以它们依然是一个进程,只不过这些进程可以与其他进程共享某些资源,这里的其他进程也是所谓的线程。
1 内核线程与普通进程的异同
1.1 内核线程没有自己的地址空间,所以它们的"current->mm"都是空的;
1.2 内核线程只能在内核空间操作,不能与用户空间交互;
1.3 跟普通进程一样,内核线程也有优先级和被调度。
2 内核线程创建
在内核中,有两种方法可以生成内核线程,一种是使用kernel_thread()接口,另一种是用kthread_create()接口。
先说kernel_thread接口,使用该接口创建的线程,必须在该线程中调用daemonize()函数,这是因为只有当线程的父进程指向"Kthreadd"时,该线程才算是内核线程,而恰好daemonize()函数主要工作便是将该线程的父进程改成“kthreadd"内核线程;默认情况下,调用deamonize()后,会阻塞所有信号,如果想操作某个信号可以调用allow_signal()函数。
而kthread_create接口,则是标准的内核线程创建接口,只须调用该接口便可创建内核线程;默认创建的线程是存于不可运行的状态,所以需要在父进程中通过调用wake_up_process()函数来启动该线程。
3 内核线程的退出
当线程执行到函数末尾时会自动调用内核中do_exit()函数来退出或其他线程调用kthread_stop()来指定线程退出。
4 内核线程接口
4.1 kernel_thread接口
注:这里的flags的取值为
4.2 kthread_create接口
注:因为线程也是进程,所以其结构体也是使用进程的结构体"struct task_struct"。
5 示例代码 kthread.rar
6 执行结果
从结果可以看出,子线程的地址空间是没有的。
转载请注明出处: add358.blog.chinaunix.net
1 内核线程与普通进程的异同
1.1 内核线程没有自己的地址空间,所以它们的"current->mm"都是空的;
1.2 内核线程只能在内核空间操作,不能与用户空间交互;
1.3 跟普通进程一样,内核线程也有优先级和被调度。
2 内核线程创建
在内核中,有两种方法可以生成内核线程,一种是使用kernel_thread()接口,另一种是用kthread_create()接口。
先说kernel_thread接口,使用该接口创建的线程,必须在该线程中调用daemonize()函数,这是因为只有当线程的父进程指向"Kthreadd"时,该线程才算是内核线程,而恰好daemonize()函数主要工作便是将该线程的父进程改成“kthreadd"内核线程;默认情况下,调用deamonize()后,会阻塞所有信号,如果想操作某个信号可以调用allow_signal()函数。
而kthread_create接口,则是标准的内核线程创建接口,只须调用该接口便可创建内核线程;默认创建的线程是存于不可运行的状态,所以需要在父进程中通过调用wake_up_process()函数来启动该线程。
3 内核线程的退出
当线程执行到函数末尾时会自动调用内核中do_exit()函数来退出或其他线程调用kthread_stop()来指定线程退出。
4 内核线程接口
4.1 kernel_thread接口
- int kernel_thread(int (*fn)(void *), void *arg, unsigned long flags);
-
// fn为线程函数,arg为线程函数参数,flags为标记
- void daemonize(const char * name,...); // name为内核线程的名称
点击(此处)折叠或打开
- CSIGNAL /* signal mask to be sent at exit */
- CLONE_VM /* set if VM shared between processes */
- CLONE_FS /* set if fs info shared between processes */
- CLONE_FILES /* set if open files shared between processes */
- CLONE_SIGHAND /* set if signal handlers and blocked signals shared */
- CLONE_PTRACE /* set if we want to let tracing continue on the child too */
- CLONE_VFORK /* set if the parent wants the child to wake it up on mm_release */
- CLONE_PARENT /* set if we want to have the same parent as the cloner */
- CLONE_THREAD /* Same thread group? */
- CLONE_NEWNS /* New namespace group? */
- CLONE_SYSVSEM /* share system V SEM_UNDO semantics */
- CLONE_SETTLS /* create a new TLS for the child */
- CLONE_PARENT_SETTID /* set the TID in the parent */
- CLONE_CHILD_CLEARTID /* clear the TID in the child */
- CLONE_DETACHED /* Unused, ignored */
- CLONE_UNTRACED /* set if the tracing process can't force CLONE_PTRACE
- on this clone */
- CLONE_CHILD_SETTID /* set the TID in the child */
- CLONE_NEWUTS /* New utsname group? */
- CLONE_NEWIPC /* New ipcs */
- CLONE_NEWUSER /* New user namespace */
- CLONE_NEWPID /* New pid namespace */
- CLONE_NEWNET /* New network namespace */
- CLONE_IO /* Clone io context */
4.2 kthread_create接口
- struct task_struct *kthread_create(int (*threadfn)(void *data),void *data,
- const char namefmt[], ...);
-
/ /threadfn为线程函数;data为线程函数参数;namefmt为线程名称,可被格式化的
- int wake_up_process(struct task_struct *p); //唤醒线程
- struct task_struct *kthread_run(int (*threadfn)(void *data),void *data,
- const char namefmt[], ...);//是以上两个函数的功能的总和
5 示例代码 kthread.rar
点击(此处)折叠或打开
- #include <linux/module.h>
- #include <linux/init.h>
- #include <linux/kernel.h>
- #include <linux/kthread.h>
-
- MODULE_LICENSE("Dual BSD/GPL");
- MODULE_AUTHOR("Kozo");
-
- static int noop(void *dummy)
- {
- int i = 0;
- daemonize("mythread"); /*使用kernel_thread创建的线程,需要调用该函数*/
- while(i++ < 5)
- {
- printk("my thread:current->mm = %p\n",current->mm);
- printk("my thread:current->active_mm = %p\n",current->active_mm);
- set_current_state(TASK_INTERRUPTIBLE);
- schedule_timeout(10*HZ);
- }
- return 0;
- }
-
- static int noop2(void *dummy)
- {
- int i = 0;
- while(i++ < 5)
- {
- printk("my thread:current->mm = %p\n",current->mm);
- printk("my thread:current->active_mm = %p\n",current->active_mm);
- set_current_state(TASK_INTERRUPTIBLE);
- schedule_timeout(10*HZ);
- }
- return 0;
- }
- static __init int demo_init(void)
- {
-
- printk(KERN_INFO"demo init\n");
- printk("demo init:current->mm = %p\n",current->mm);
- printk("demo init:current->active_mm = %p\n",current->active_mm);
- //kernel_thread(noop, NULL, CLONE_KERNEL|SIGCHLD);
- kthread_run(noop2,NULL,"mythread");
- return 0;
- }
-
- static __exit void demo_exit(void)
- {
- printk(KERN_INFO"demo exit\n");
- }
-
- module_init(demo_init);
- module_exit(demo_exit);
从结果可以看出,子线程的地址空间是没有的。
转载请注明出处: add358.blog.chinaunix.net