※ 引述《laughingman (笑面男)》之铭言:
: 看起来中断发生的handler是写在-/code/userprog/exception.cc里的ExceptionHandler
: 这支function中的SyscallException的switch里。所以只要再多加一个case就可以处理
: 新的system call,实作部分就写在该写的地方就好。但问题来了,其实还要在
: -/code/test/start.s里多加类似底下的程式码,
: .globl Print
: .ent Print
: Print:
: addiu $2, $0, SC_Print
: syscall
: j $31
: .end Print
: 这段看起来是MIPS的组语,我也了解意思,不过system call不是已经用c++实作了吗?
: 加这段组语的意思是什么呢?
: 我有看一下-/code/test里的makefile,看起来其他的test file都会用到start.o,而
: 这个start.o就是由MIPS组译器将start.s组译后得来的(?),这中间的逻辑其实我不是
: 很懂,有没有修过作业系统的高手可以解释一下,感谢各位拨空看小弟的问题。
syscall 有两边要实作, 一边是 syscall 的 handler本身, 依你给的资讯就是
exception.cc 那边
另一边是 call syscall. 通常要包一个给 C/C++ 使用的 wrapper,
这样才能用 C/C++ 呼叫 syscall. syscall 不同于一般 function, 无法直接呼叫
你 google 一下 mips syscall abi, 前两个 link 可以看一下
syscall abi 会规范 syscall 的流程,
正常来说 syscall abi 要看 OS 决定, 你应该要查一下 Nachos 的规定
这边我就直接用 linux 的 syscall abi 来讲
$v0 是 syscall number, 也就是 $2
$a0..$a7 是 syscall argument, $a0..$a2
return value 在 $v0
回头看你写的那三行,
Print:
addiu $2, $0, SC_Print ; $v0 = 0 + SC_Print
syscall ; 进到 syscall handler 处理 print
j $31 ; jump return return register, 即 "return"
这实是实作 Print syscall 的 caller 端本身, 应该还会有一个 C/C++ 宣告, 例如
int Print(const char *message);
之类的
因为 mips 的一般 function call abi 的 arguemnt/ret 传接法和 syscall
时是一样的, 都是例用 $a0..$a7 传, $v0 return
所以 Print 的 caller 刚好会准备好相关的 register, call 进 Print 后,
又再顺着把准备好的 register pass 给 syscall