41 - 内核中的屏幕打印

---- 整理自狄泰软件唐佐林老师课程

查看所有文章链接:(更新中)深入浅出操作系统 - 目录

1. 课程目标

使用C语言设计一组打印函数,为后续进一步开发内核功能做准备。

2. 内核中的屏幕打印模块

在这里插入图片描述

2.1 接口设计

void ClearScreen();
void SetPrintPos(short  w, short  h);
void SetPrintColor(PrintColor  c);
void PrintChar(char  c);
int PrintString(const  char*  s);
int PrintIntDec(int  n);
int PrintIntHex(unsigned  int  n);

2.2 接口之间的关系

在这里插入图片描述

2.3 void PrintChar(char c);

功能定义:在屏幕上打印一个字符
实现原理:直接在显存对应位置写入数据

在这里插入图片描述

  • 问题:不是说好用C语言开发内核吗?

2.4 C语言中的内嵌汇编

这里的是 AT&T 汇编,之前code中写的是 Intel 汇编。

在这里插入图片描述

asm [volatile] ("Instruction list":Output:Input:Clobber/Modify)

Clobber/Modify:
指定此项会通知编译器可能会破坏寄存器/内存里面的数据,然后编译器就会将这些可能被破坏的内存和寄存器提前保护起来。

  • 内嵌汇编示例

在这里插入图片描述

  • 注意
    因为使用了gcc编译器,所以内嵌汇编时使用的是AT&T汇编格式(与 nasm汇编格式不同)

2.5 编程实验:void PrintChar(char c);

2.5.1 打印字符 “ D ”

【参看链接】:41 - 内核中的屏幕打印 / 41 / 00

在这里插入图片描述
在这里插入图片描述

2.5.2 清屏并打印字符串

【参看链接】:41 - 内核中的屏幕打印 / 41 / 01

在这里插入图片描述
在这里插入图片描述

2.5.3 打印16进制数

【参看链接】:41 - 内核中的屏幕打印 / 41 / 02

在这里插入图片描述
在这里插入图片描述

2.5.4 打印十进制数

在这里插入图片描述

【参看链接】:41 - 内核中的屏幕打印 / 41 / 03

在这里插入图片描述
在这里插入图片描述

2.6 保护模式下的光标追踪

在这里插入图片描述

【参看链接】:41 - 内核中的屏幕打印 / 41 / 04

在这里插入图片描述
在这里插入图片描述

3. 小结

  • gcc编译器只支持AT&T格式的内嵌汇编
  • 通过内嵌汇编的方式可实现PrintChar()函数
  • PrintChar()是其它屏幕打印函数的基础
  • 通过操作0x03D4和0x03D5端口对光标位置进行设置
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
将echo命令最后一行的exit()替换为`return 0`后,运行echo命令会出现如下错误提示: ``` $ echo test echo test panic: initproc: not sleeping cpuid = 0 ``` 这个错误提示是在`panic()`函数打印的,该函数定义在`/kernel/proc.c`文件,具体的打印语句是在该函数的第7行。 出现这种状况的原因是,在xv6,每个进程都有自己的用户栈和内核栈。当进程结束时,会调用`exit()`函数,该函数会在用户栈设置返回值并跳转到内核的`trap()`函数,然后进一步调用`panic()`函数,打印上述错误信息。 而我们将exit()替换为`return 0`后,虽然也会在用户栈设置返回值,但并不会跳转到内核的`trap()`函数,因此就会出现上述错误。 为了解决这个问题,我们需要在`exit()`函数,将返回值传递给`wait()`函数,并在`wait()`函数进行处理。我们可以添加一个新的全局变量`int wait_return_code`,在`exit()`函数将返回值赋值给该变量,然后在`wait()`函数对该变量进行判断。修改后的代码如下: ```c int wait_return_code = 0; void exit(void) { struct proc *curproc = myproc(); struct proc *p; int fd; if(curproc == initproc) panic("init exiting"); // Close all open files. for(fd = 0; fd < NOFILE; fd++){ if(curproc->ofile[fd]){ fileclose(curproc->ofile[fd]); curproc->ofile[fd] = 0; } } begin_op(); iput(curproc->cwd); end_op(); curproc->cwd = 0; acquire(&ptable.lock); // Parent might be sleeping in wait(). wakeup1(curproc->parent); // Pass abandoned children to init. for(p = ptable.proc; p < &ptable.proc[NPROC]; p++){ if(p->parent == curproc){ p->parent = initproc; if(p->state == ZOMBIE) wakeup1(initproc); } } // Jump into the scheduler, never to return. curproc->state = ZOMBIE; wait_return_code = curproc->xstate; sched(); panic("zombie exit"); } int wait(void) { struct proc *p; int havekids, pid; acquire(&ptable.lock); for(;;){ // Scan through table looking for zombie children. havekids = 0; for(p = ptable.proc; p < &ptable.proc[NPROC]; p++){ if(p->parent != myproc()) continue; havekids = 1; if(p->state == ZOMBIE){ // Found one. pid = p->pid; kfree(p->kstack); p->kstack = 0; freevm(p->pgdir); p->state = UNUSED; p->pid = 0; p->parent = 0; p->name[0] = 0; p->killed = 0; p->xstate = 0; wait_return_code = 0; release(&ptable.lock); return pid; } } // No point waiting if we don't have any children. if(!havekids || myproc()->killed){ release(&ptable.lock); return -1; } // Wait for children to exit. (See wakeup1 call in proc_exit.) sleep(myproc(), &ptable.lock); //DOC: wait-sleep } } ``` 我们在`exit()`函数将当前进程的返回值赋值给全局变量`wait_return_code`,在`wait()`函数根据`wait_return_code`的值进行判断,如果为0,则正常结束,不打印提示信息;否则,打印原来的错误信息。 修改后的echo命令代码如下: ```c #include "kernel/types.h" #include "user/user.h" int main(int argc, char *argv[]) { int i; for (i = 1; i < argc; i++) { printf("%s ", argv[i]); } printf("\n"); if (argc < 2) { return 0; } if (strcmp(argv[argc - 1], "0x88") == 0) { return 0x88; } else { return 0; } } ``` 修改后的代码运行结果如下: ``` $ echo test echo test panic: initproc: not sleeping cpuid = 0 $ echo test 0x88 echo test 0x88 ``` 可以看到,如果返回值为0,程序正常结束,没有任何提示信息;如果返回值为0x88,则打印原来的错误信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

uuxiang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值