php thread linux,Linux_linux内核函数kernel_thread,设备驱动程序中,如果需要几 - phpStudy...
linux内核函数kernel_thread
设备驱动程序中,如果需要几个并发执行的人物,可以启动内核线程,启动内和县城的函数为:
int kernel_thread (int ( * fn )( void * ), void * arg, unsigned long flags);
kernel_thread函数的作用是产生一个新的线程
内核线程实际上就是一个共享父进程地址空间的进程,它有自己的系统堆栈.
内核线程和进程都是通过do_fork()函数来产生的,系统中规定的最大进程数与
线程数由fork_init来决定:
[/arch/kernel/process.c/fork_init()]
void __init fork_init(unsigned long mempages)
{
#ifndef __HAVE_ARCH_TASK_STRUCT_ALLOCATOR
#ifndef ARCH_MIN_TASKALIGN
#define ARCH_MIN_TASKALIGN L1_CACHE_BYTES
#endif
/* 在slab高速缓存中建立task_struct结构专用的缓冲区队列 */
task_struct_cachep =
kmem_cache_create("task_struct", sizeof(struct task_struct),
ARCH_MIN_TASKALIGN, SLAB_PANIC, NULL, NULL);
#endif
/*
把默认线程数设置到一个安全值,因为内核中总的线程占用的空间
可能要内存一半还要多.
参数mempages系统中总的物理内存结构大小,它等于mempages/PAGESIZE.
比如我机器的内存是512m,那么在我的系统最多能同时产生线程数为
(512*2^20/2^12) / 2^3 = 512*2^5 = 16384
*/
max_threads = mempages / (8 * THREAD_SIZE / PAGE_SIZE);
/*
* 启动系统的时候至少需要20个线程
*/
if(max_threads < 20)
max_threads = 20;
/*
* 每个进程最多产生max_threads/2,也就是线程总数的一半,在我的机器上为8192.
*/
init_task.signal->rlim[RLIMIT_NPROC].rlim_cur = max_threads/2;
init_task.signal->rlim[RLIMIT_NPROC].rlim_max = max_threads/2;
}
kernel_thread原形在/arch/kernel/process.c中.
(*fn)(void *)为要执行的函数的指针,arg为函数参数,flags为do_fork产生线程时的标志.
int kernel_thread(int (*fn)(void *), void * arg, unsigned long flags)
{
struct pt_regs regs;
memset(®s, 0, sizeof(regs));
regs.ebx = (unsigned long) fn; /* ebx指向函数地址 */
regs.edx = (unsigned long) arg; /* edx指向参数 */
regs.xds = __USER_DS;
regs.xes = __USER_DS;
regs.orig_eax = -1;
regs.eip = (unsigned long) kernel_thread_helper;
regs.xcs = __KERNEL_CS;
regs.eflags = X86_EFLAGS_IF | X86_EFLAGS_SF | X86_EFLAGS_PF | 0x2;
/* 利用do_fork来产生一个新的线程,共享父进程地址空间,并且不允许调试子进程 */
return do_fork(flags | CLONE_VM | CLONE_UNTRACED, 0, ®s, 0, NULL, NULL);
}
[/arch/i386/kernel/process.c/kernel_thread_helper]
extern void kernel_thread_helper(void); /* 定义成全局变量 */
__asm__(".section .text\n"
".align 4\n"
"kernel_thread_helper:\n\t"
"movl %edx,%eax\n\t"
"pushl %edx\n\t" /* edx指向参数,压入堆栈 */
"call *%ebx\n\t" /* ebx指向函数地址,执行函数 */
"pushl %eax\n\t"
"call do_exit\n" /* 结束线程 */
".previous");
在kernel_thread中调用了do_fork,那么do_fork是怎样转入kernel_thread_helper去执行的呢,继续跟踪下do_fork函数.
[kernel/fork.c/do_fork()]
long do_fork(unsigned long clone_flags,
unsigned long stack_start,
struct pt_regs *regs,
unsigned long stack_size,
int __user *parent_tidptr,
int __user *child_tidptr)
{
....
....
p = copy_process(clone_flags, stack_start, regs, stack_size, parent_tidptr, child_tidptr, pid);
....
....
}
它调用copy_process函数来向子进程拷贝父进程的进程环境和全部寄存器副本.
[kernel/fork.c/do_fork()->copy_process]
static task_t *copy_process(unsigned long clone_flags,
unsigned long stack_start,
struct pt_regs *regs,
unsigned long stack_size,
int __user *parent_tidptr,
int __user *child_tidptr,
int pid)
{
...
...
retval = copy_thread(0, clone_flags, stack_start, stack_size, p, regs);
...
...
}
它又调用copy_thread来拷贝父进程的系统堆栈并做相应的调整.
[/arch/i386/kernel/process.c/copy_thread]:
int copy_thread(int nr, unsigned long clone_flags, unsigned long esp,
unsigned long unused,
struct task_struct * p, struct pt_regs * regs)
{
...
...
p->thread.eip = (unsigned long) ret_from_fork;
}
在这里把ret_from_fork的地址赋值给p->thread.eip,p->thread.eip表示当进程下一次调度时的指令开始地址,
所以当线程创建后被调度时,是从ret_from_fork地址处开始的.
[/arch/i386/kernel/entry.s]
到这里说明,新的线程已经产生了.
ENTRY(ret_from_fork)
pushl %eax
call schedule_tail
GET_THREAD_INFO(%ebp)
popl %eax
jmp syscall_exit
syscall_exit:
...
work_resched:
call schedule
...
当它从ret_from_fork退出时,会从堆栈中弹出原来保存的ip,而ip指向kernel_thread_helper,
至此kernel_thread_helper被调用,它就可以运行我们的指定的函数了相关阅读:
CSS的书写有怎样的功能
10g中构建“高”索引
驯服CSS选择器--健壮我们的样式表
收藏一些不常用,但是有用的代码
Oracle数据库中创建合理的数据库索引
HTML表格标记教程(29):单元格的亮边框色属性BORDERCOLORLIGHT
HTML网页的基本组成概述
两个vbs脚本利用了wscirpt.network
HTML V5 与 XHTML V2
ASP SQL防注入的方法
CssGaga教程:自动同步文件和哀悼日网站一键变灰
HttpWebRequest实现浏览器请求(Cookie)
关注于使用常用对象来加快Access 2007数据库下
PHP 工厂模式使用方法
php thread linux,Linux_linux内核函数kernel_thread,设备驱动程序中,如果需要几 - phpStudy...相关推荐
- linux设备驱动程序中的阻塞机制
阻塞与非阻塞是设备访问的两种方式.在写阻塞与非阻塞的驱动程序时,经常用到等待队列. 一.阻塞与非阻塞 阻塞调用是指调用结果返回之前,当前线程会被挂起,函数只有在得到结果之后才会返回. 非阻塞指不能立刻 ...
- Unix/Linux操作系统分析实验四 设备驱动: Linux系统下的字符设备驱动程序编程
Unix/Linux操作系统分析实验一 进程控制与进程互斥 Unix/Linux操作系统分析实验二 内存分配与回收:Linux系统下利用链表实现动态内存分配 Unix/Linux操作系统分析实验三 文 ...
- Linux设备驱动程序中创建线程的方法
参考博客文章来源:https://blog.csdn.net/ezimu/article/details/60467017 第一种方法:kernel_thread #include <linux ...
- linux kernel 2.6 i2c设备驱动程序框架介绍,linux设备驱动程序-i2c(2)-adapter和设备树的解析...
linux设备驱动程序-i2c(2)-adapter和设备树的解析 (注: 基于beagle bone green开发板,linux4.14内核版本) 而在linux设备驱动程序--串行通信驱动框架分 ...
- Linux驱动(六)设备驱动中的阻塞与非阻塞IO
我们在Linux学习(二十三)IO模型中了解了LINUX中IO模型,IO模型最简单的可以分为阻塞IO和非阻塞IO.并且学习了一个用如何使用阻塞操作和非阻塞操作.而应用层之所以能实现阻塞操作和非阻塞操作 ...
- linux 驱动 文件计数,字符设备驱动程序1
一.主设备号和此设备号主设备号表示设备对应的驱动程序:次设备号由内核使用,用于正确确定设备文件所指的设备.内核用dev_t类型()来保存设备编号,dev_t是一个32位的数,12位表示主设备号,20为 ...
- linux驱动 打印变量,linux驱动 内核函数 变量 宏定义
insmod modprobe(自动检测 模块加载时需要的别的模块) rmmod 用户空间工具, 加载模块到运行中的内核以及去除它们. #include module_init(init_functi ...
- linux do_irq 报错 代码,linux - 内核函数asm_do_IRQ()中的irq与我在模块中请求的不同 - 堆栈内存溢出...
我做了一些皮质-A9开发板的实验. 我使用gpio_to_irq()获取irq num并且我请求了irq并用它写了一个小驱动程序,它在syslog中是196. 我在asm_do_IRQ中添加了一些pr ...
- linux ubuntu内核安装位置,在Ubuntu中安装或升级内核 linux kernel
主要参考:https://www.linux.cn/article-6137-1.html 1. 从 http://kernel.ubuntu.com/~kernel-ppa/mainline/ 中找 ...
最新文章
- 兼容浏览器_你知道什么是跨浏览器兼容吗?
- java ee中javamail注解_JavaEE之注解
- python bottle学习(四)request.quest/query_string/params/body等方法介绍
- 基于VS Code创建Java command-line app
- php soap function,从PHP SoapServer返回PHP数组
- 论ul、ol和dl的区别
- 鸟哥的 Linux 私房菜13-- 学习 BASH Shell 咯
- Spring异步切面源码解析
- prinect pdf toolbox 2021中文版
- 图灵机器人Web API
- TM4C123G学习记录(4)--关于ROM前缀函数和HWREG函数
- JavaEE | 语言基础部分、对象与类
- 51单片机小白零基础教程——独立键盘检测原理以及实际应用
- 听吐的微信提示音终于能改了
- mysql 初始化数据库报错
- 电脑接两个屏幕,其中一个偶尔闪灭
- 汽车UDS诊断详解及Vector相关工具链使用说明——2.1.3 初步了解CDD(以10服务为例)
- 毛星云opencv之8.4.4查找和绘制图像轮廓矩
- python界面颜色-python界面颜色
- git将本地代码推到远程仓库
热门文章
- guava集合操作类的使用_使用Guava进行测试集合的实现
- php cdi_CDI和EJB:在事务成功时发送异步邮件
- jdbc和jdbc驱动_JDBC布尔兼容性列表
- java中iterator_如何在Java中读取CSV文件-Iterator和Decorator的案例研究
- 试用期java_Java 7试用资源
- Java编程语言的历史和未来
- eclipse pmd使用_使用您自己的规则在Eclipse中自定义PMD
- javafx2_JavaFX 2 GameTutorial第5部分
- Java 9对可选的补充
- 确定活动的热点垃圾收集器