3.11 函数:pid_vnr( )
文件包含:
#include <linux/pid.h>
函数定义:
在内核源码中的位置:linux-3.19.3/kernel/pid.c
函数定义格式:pid_t pid_vnr(struct pid *pid)
函数功能描述:
函数pid_vnr( )根据输入参数,获取进程的局部进程号。
输入参数说明:
参数pid是struct pid类型的指针变量,保存进程描述符信息,其定义及详细解释请读者自行参考本章函数f ind_get_pid( )分析文档的返回参数说明部分。
返回参数说明:
此函数的返回值是pid_t类型的变量,在此代表获取的进程的全局进程号,pid_t的定义见文件linux-3.19.3/include/linux/types.h,如下:
typedef __kernel_pid_t pid_t。
其中__kernel_pid_t的定义见文件linux-3.19.3/include/uapi/asm-generic/posix_types.h,如下:
#ifndef __kernel_pid_t typedef int __kernel_pid_t; #endif
实例解析:
编写测试文件:pid_vnr.c
头文件引用:
#include <linux/module.h> #include <linux/sched.h> #include <linux/pid.h> MODULE_LICENSE("GPL");
模块加载函数定义:
static int __init pid_vnr_init(void) { printk("into pid_vnr_init.\n"); struct pid * kpid=find_get_pid(current->pid); //获取当前进程的描述符 printk("the level of the pid is:%d\n", kpid->level); //显示进程描述符的level的值 printk("the pid of the pid is:%d\n", kpid->numbers[kpid->level].nr); //显示进程的进程号 int vnr = pid_vnr(kpid); //获取进程描述符的局部进程号 printk("the pid_vnr result is:%d\n", vnr); //显示局部进程号 printk("the current pid is:%d\n", current->pid); //显示当前进程的进程号 printk("the current tgid is:%d\n", current->tgid); //显示当前进程的线程组号 printk("out pid_vnr_init.\n"); return 0; }
模块退出函数定义:
static void __exit pid_vnr_exit(void) { printk("Goodbye pid_vnr\n"); }
模块加载、退出函数调用:
module_init(pid_vnr_init); module_exit(pid_vnr_exit);
实例运行结果及分析:
首先编译模块,执行命令insmod pid_vnr.ko插入模块,然后执行命令dmesg -c查看系统输出信息,会出现如图3-11所示的结果。
图3-11 插入pid_vnr模块系统输出信息
结果分析:
图3-11输出结果显示函数pid_vnr( )获取的局部进程号是16558,而当前进程的进程号和组进程号都是16558,一般进程的进程号和进程的局部进程号是相同的。
函数分析比较:
函数pid_nr( )和函数pid_vnr( )都能获得进程的进程号,但一个是全局进程号,一个是局部进程号,获取全局进程号比较简单,因为全局进程号就是保存在进程描述符中的进程的PID值,与当前进程无关。而局部进程号的获取相对比较复杂,需要调用函数pid_nr_ns( )去获取进程的局部进程号,在获取局部进程号时还需要判断进程的命名空间信息。
全局进程号和局部进程号的区别:
全局ID是在内核本身和初始命名空间中的唯一ID号,在系统启动期间开始的init进程即属于初始命名空间。对每个ID类型,都有一个给定的全局ID,保证在整个系统中是唯一的。
局部ID属于某个特定的命名空间,不具备全局有效性。对于每个ID类型,它们在所属的命名空间内部有效,但类型相同、值也相同的ID可能出现在不同的命名空间中。