学号尾号:155
基于ubuntu kylin 18.10虚拟机
原创作品转载请注明出处https://github.com/mengning/linuxkernel/
wget https://cdn.kernel.org/pub/linux/kernel/v5.x/linux-5.0.tar.xz # 下载内核
# 解压文件
xz -d linux-5.0.1.tar.xz
tar -xvf linux-5.0.tar.xz
cd linux-5.0
# 编译内核
make i386_defconfig
此处出现报错:
执行以下命令
apt-get install flex
再次执行
make i386_defconfig
出现报错
执行以下命令
apt-get install bison
再次执行
make i386_defconfig
make
出现报错
执行以下命令
apt-get install libssl-dev
再次执行
make
等待一段时间后,控制台会输出如下信息,表示编译完成
cd ..
mkdir rootfs
git clone https://github.com/mengning/menu.git
cd menu
gcc -pthread -o init linktable.c menu.c test.c -m32 -static
此处出现问题:
apt install gcc-multilib
gcc -pthread -o init linktable.c menu.c test.c -m32 -static
再次出现问题
apt-get install gcc-4.8 gcc-4.8-multilib g++-4.8 g++-4.8-multilib
gcc -pthread -o init linktable.c menu.c test.c -m32 -static
成功了
cd ../rootfs
cp ../menu/init ./
find . | cpio -o -Hnewc |gzip -9 > ../rootfs.img
制作成功!
cd ..
qemu -kernel linux-5.0/arch/x86/boot/bzImage -initrd rootfs.img
启动成功!
---
我的学号尾号是155,在查阅系统调用表后,发现155号系统调用是
#define __NR_sched_getparam 155
函数的原型是
#include <sched.h>
//该函数用于根据进程号获取进程的调度参数,pid用于指定要获取调度参数的进程号,为0时表示获取当前进程的调度参数
//param用于存储获得的进程调度参数
//返回0时表示成功获得进程的调度参数,返回-1时表示出错,并设置errno
//errno有两种,EPERM:调用进程没有足够的权限来获取调度参数。ESRCH:进程pid不存在。
int sched_getparam(pid_t pid,struct sched_param * param);
struct sched_param
{
int32_t sched_priority;//获取调度参数时,此成员将反映分配给线程或进程的优先级
int32_t sched_curpriority;//获取调度参数时,此成员将设置为线程或进程当前运行的优先级。这是内核在进行调度决策时使用的值。
union
{
int32_t reserved[8];
struct
{
int32_t __ss_low_priority;
int32_t __ss_max_repl;
struct timespec __ss_repl_period;
struct timespec __ss_init_budget;
} __ss;
} __ss_un;
}
#define sched_ss_low_priority __ss_un.__ss.__ss_low_priority
#define sched_ss_max_repl __ss_un.__ss.__ss_max_repl
#define sched_ss_repl_period __ss_un.__ss.__ss_repl_period
#define sched_ss_init_budget __ss_un.__ss.__ss_init_budget
对于该系统调用,我们可以写出如下代码对其进行测试
//在menu文件夹中的test.c文件中加入如下代码
#include <sched.h>
int test_get_param()
{
struct sched_param param;
sched_getparam(0,¶m);
printf("The original priority of the process is %d\n",param.sched_priority);
}
int main()
{
PrintMenuOS();
SetPrompt("MenuOS>>");
MenuConfig("version","MenuOS V1.0(Based on Linux 3.18.6)",NULL);
MenuConfig("quit","Quit from MenuOS",Quit);
MenuConfig("time","Show System Time",Time);
MenuConfig("time-asm","Show System Time(asm)",TimeAsm);
MenuConfig("get_param","根据进程号获取进程的优先级",test_get_param);
ExecuteMenu();
}
重新制作根文件系统
gcc -pthread -o init linktable.c menu.c test.c -m32 -static
cd ../rootfs
cp ../menu/init ./
find . | cpio -o -Hnewc |gzip -9 > ../rootfs.img
使用gdb跟踪调试内核
cd ..
qemu -kernel linux-5.0/arch/x86/boot/bzImage -initrd rootfs.img -S -s -append nokaslr
此时启动的qemu窗口是stopped的状态
另开一个shell窗口
gdb
file vmlinux # target remote之前加载符号表
target remote:1234 # 建立gdb和gdbserver之间的连接,按c让qemu上的Linux系统继续运行
b sys_sched_getparam # 打上断点
在gdb窗口中输入
c
在qemu窗口中输入
get_param
在gdb窗口中输入
c
可以看到,sched_getparam系统调用的流程如下:
由此可以看出,用户态进程是无法访问内核的内存空间的,只有内核态进程才能访问内核。因此当执行系统调用时,必须要先将进程转换成内核态才能执行系统调用,执行完毕后再恢复到用户态。
原文:https://www.cnblogs.com/jlcs/p/10561678.html