30天自制操作系统day27

本文介绍了如何使用LDT来隔离应用程序内存空间,并详细解释了LDT的注册过程及如何利用库来优化应用程序的编译流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

LDT

之前通过设置段时将权限+0x60的方法防止应用程序访问操作系统内存空间。但是现在应用程序还是可以修改别的应用程序的内存。为了防止应用程序被攻击,需要使用LDT。
LDT是只对一个应用程序有效的,容量也是64KB。因为现在我们每个程序只需要2个段(数据段,代码段),所以只用了里面的16字节。LDT每个程序一个,放在task结构中。
要使用LDT,需要在GDT表中注册一个LDT段。每个程序注册一个。

#define AR_LDT      0x0082

在注册LDT段时传进去的访问权限,和其它段有区别。
在task_init()函数中,为task数组中的所有task注册LDT,还要将LDT段的段号写入tss.ldtr中。set_segmdesc()中的limit是15,因为我们的LDT表只有2个段。

        taskctl->tasks0[i].tss.ldtr = (TASK_GDT0 + MAX_TASKS + i) * 8;
        set_segmdesc(gdt + TASK_GDT0 + i, 103, (int) &taskctl->tasks0[i].tss, AR_TSS32);
        set_segmdesc(gdt + TASK_GDT0 + MAX_TASKS + i, 15, (int) taskctl->tasks0[i].ldt, AR_LDT);

在cmd_app()中,把程序的数据段和代码段注册到LDT中。


        set_segmdesc(task->ldt + 0, finfo->size - 1, (int) p, AR_CODE32_ER + 0x60);
        set_segmdesc(task->ldt + 1, segsiz - 1,      (int) q, AR_DATA32_RW + 0x60);

start_app()也和之前不同了。

        start_app(0x1b, 0 * 8 + 4, esp, 1 * 8 + 4, &(task->tss.esp0));

乘以8和GDT一样,但是最后要加上4,表示这是注册到LDT的。因为GDT一般情况下后三位都是0,所以可以区别开。

之前每个应用程序,编译时都要引用a_nask.nas,但是里面有很多函数是用不到的。全部引用进来会使可执行文件体积很大。
可以通过把每个函数单独编译成一个.obj文件,然后在应用程序的makefile中只引用它调用到的那些函数。不过更好的方法是使用库。
库的作用是把刚才所有的.obj文件合并成一个apilib.lib文件。在编译应用程序时,可以这样写makefile:

.bim : a.obj apilib.lib Makefile 
    $(OBJ2BIM) @$(RULEFILE) out:a.bim map:a.map a.obj apilib.lib

这相当于Linux里面的静态库吧,在编译时就把程序中需要调用的函数编译进来。
然后,再写一个apilib.h文件,所有的函数声明放在这里,在应用程序中就可以include这个头文件。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值