- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在使用 qemu 运行 Debian Squeeze mips 架构,只是想弄清楚我可以放入 $v0 中执行的系统调用是什么。我找到了这个 webpage .尽管此页面确实告诉我写命令使用 4。我执行以下操作:
我正在使用 this image安装 Debian Squeeze。
是的,它使用 mips 作为系统架构:
uname -a
Linux debian 2.6.32-5-4kc-malta #1 Sat Feb 16 12:43:42 UTC 2013 mips GNU/Linux
--测试.asm--
.global __start
.text
__start:
li $v0, 4
li $a0, 1
la $a1, out_string
li $a2, 12
syscall
.data
out_string: .asciiz "Hello World\n"
现在我使用“as”进行组装,使用“ld”进行链接
as -march=mips32 -o test.o test.asm
ld -o test.out test.o
chmod +x test.out
./test.out
不幸的是,这会导致错误:Illegal Instruction
。
我编译了以下 c 程序以确定写入函数使用的偏移量及其参数以进行双重检查。
--写.c--
#include <stdio.h>
int main(int argc, char *argv[]){
printf("Hello World\n");
return 0;
}
运行 strace -i ./write.out
给出:
[2abbcbe0] write(1, "Hello World\n", 12Hello World)
write 的手册页说明如下:
#include <unistd.h>
ssize_t write(int fd, const void *buf, size_t count);
我尝试使用 write 命令输出“Hello World”,它本身就可以正常工作。
这里描述了 write 函数的作用以及我是如何得出我编写 test.asm 的方式的:
fd 或 $a0 - 从打开调用中获得的文件描述符。它是一个整数值。也可以分别为标准输入、标准输出和标准错误给出值 0、1、2
buff 或 $a1 - 指向一个字符数组,可用于存储从 fd 指向的文件中获取的内容。
nbytes 或 $a2 - 指定要从字符数组写入 fd 指向的文件的字节数。
因此我们得出顺序:
#0x4 being what unistd.h for mips tells us the system call code is 4
$v0, 4
#Set first argument to the std output 1 for the file descriptor
$a0, 1
#Set the address for the string or buff
$a1, out_string
#Set nbytes or total bytes of the string
$a2, 12
这个例子在 Mars 和 Spim 中工作得很好,尽管只有一个参数而不是 3 个:
--example.asm--
.data
out_string: .asciiz "\nHello, World!\n"
.text
main:
li $v0, 4
la $a0, out_string
syscall
li $v0, 10
syscall
因为我在 test.asm 中得到了 Illegal Instruction
,这意味着我必须指向一个无效的指令。尽管之前观察到的系统调用代码 here .不正确,即使它是 mips 架构的 unistd.h 文件,我们确认查看写入的手册页及其参数。所以现在我尝试了 strace,认为 $v0 之前指向了错误的指令,并使用 0x2abbcbe0
找到了指令指针。也许这会奏效?让我们试试。
在 strace 中,-i 选项在系统调用时打印指令指针。
所以我编辑asm文件
--测试.asm--
.global __start
.text
__start:
li $v0, 0x2abbcbe0
li $a0, 1
la $a1, out_string
li $a2, 12
syscall
.data
out_string: .asciiz "Hello World\n"
再次运行组装和链接过程:
as -march=mips32 -o test.o test.asm
ld -o test.out test.o
chmod +x test.out
./test.out
不幸的是,这导致了错误:Illegal Instruction
again。
我还颠倒了 $a 寄存器,以为我是向后做的,我多次运行 strace 并确认指令指针中的值没有改变,这表明实际上禁用了 aslr。我在/usr/include/中查找了 unistd.h 文件,它看起来与开头显示的网站上的文件完全不同。有趣的是,这种结构在 spim 和 Mars 中都能完美运行,它们都使用 li $v0, 4 模拟它们的打印/写入系统调用。我知道 Mars 和 Spim 使用它们自己的模拟系统调用。以为 strace 会帮我找到它们,但这似乎行不通。
我在哪里可以找到 $v0 的正确值?我觉得我已经将 Google 搜索到了现在毫无意义的地方。
也许我弄乱了 asm 文件及其语法?
让我们尝试一些更简单的方法,只需运行 exit:
.global __start
.text
__start:
li $v0, 1
li $a0, 0
syscall
组装链接并运行
as -march=mips32 -o test.o test.asm
ld -o test.out test.o
chmod +x test.out
./test.out
Illegal Instruction
也试过不用-march=mips32
as -o test.o test.asm
ld -o test.out test.o
chmod +x test.out
./test.out
Illegal Instruction
这应该是最容易执行的事情。
我在这里不知所措,请大家帮忙。
最佳答案
如果您愿意使用 Linux 系统调用,那么在 MIPS 平台(马耳他)上您可以找到系统调用列表 here .在此体系结构上,Linux 将其系统调用从 4000 开始映射。所使用的 ABI 是 MIPS o32。该文件的片段如下。请注意,您可以在系统上的 /usr/include/asm/unistd.h
中找到该文件的副本:
#define __NR_Linux 4000
#define __NR_exit (__NR_Linux + 1)
#define __NR_write (__NR_Linux + 4)
#define __NR_exit_group (__NR_Linux + 246)
下面的代码应该打印Hello World
然后退出:
.text
.globl __start
__start:
li $v0, 4004
li $a0, 1
la $a1, out_string
li $a2, 12
syscall
li $v0, 4001
la $a0, 1
syscall
.data
out_string: .asciiz "Hello World\n"
__NR_exit_group
(4246) 在 Linux man pages 中定义并且在一个主要方面不同于普通的 exit
(4001):
This system call is equivalent to exit(2) except that it terminates not only the calling thread, but all threads in the calling process's thread group.
关于linux - Debian Mips 系统调用 unistd.h 丢失了吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32062807/
这两个文件有什么区别?我真的无法理解。我应该提到第一个文件应该是 arch/x86/include/asm/unistd_32.h(或 _64.h)。以下是它们包含的内容的快速预览: arch/x86
在下面的代码片段中,无论我输入多长时间(编辑:我复制并粘贴到一个随机字符串),假设一个包含 9998 个字符的字符串,read() 在 i = 4095 时停止。它声明它读取的是 EOF 字符,但我的
标准unix C具有此功能: ssize_t read(int fd, void *buf, size_t count); 但是,此“读取”功能可以读取1次的最大字节数是多少? 最佳答案 通常,它可以
我已经在学校用 Mac 编程一个月了。我们在 UNIX shell 上进行 C 编程。 现在我回到家,我使用自己的 Windows 10 电脑。我尝试过进行类似的设置,但我对此不太了解。我安装了 gi
我正在尝试使用 unistd 库写入/读取文本文件。稍后我会把它部署到/dev路径下的一个文件中。我正在尝试使用 std::vector 类型而不是 char buffer[]。所以我写了两个实现,一
如何查看函数 execve 的实现(在 x86_64 Linux 下),它在库 unistd 中?我想要这个是因为我想知道如何在不调用 execve 的情况下使用汇编程序调用外部程序。我知道有一个名为
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
假设我无法访问任何 C 标准库,我尝试使用 write(x, x, x) 函数打印 2 位整数。我只包含 根据规范。我当前的函数仅提供传递给它的值的 ASCII 表示形式。 void my_print
除此之外,我的程序应该将任何小于 32 且大于 126 的字符存储为句点 (.)。然后应该打印出这个时期。 例如,如果将换行符传递给我的打印函数,它应该将其存储为句点,并在其位置打印出句点。 (我不允
我正在编写一个支持键盘/光标的小内核。一切都工作得很好,直到我添加了 unistd.h 中的函数 内核.c #include "keyboard_map.h" #include "keyboard.c
我想用 unix 域套接字实现一个服务器/客户端程序。该程序1. 启动时,在抽象命名空间中创建一个套接字和一个 unix 地址2. 尝试将套接字绑定(bind)到该地址,如果绑定(bind)失败并返回
我正在从命名管道读取/写入数据。在写入方面,它表示它正在写入一个常量 110 字节。在读取方面,大部分时间它说它正在读取 110 个字节,这是正确的,但其他时候它说它正在读取 220 个字节或 330
我一直在尝试通过使用系统调用(read()、write()、open()、close())在 unix/linux 中执行 cp 命令的 c 编程实现。 但是当我通过终端运行我的程序时,通过将该程序的
我正在学习 UNIX 编程并尝试读/写系统调用。我有一个包含一对整数的文件: 4 5 我写了这段代码来读取数字: #include #include #include #include #in
我使用的是 VisualStudio 2012 并在 windows 7 计算机上运行。 我想在我的Windows机器下执行命令make all。为此我安装了Make for Windows 。 此外
我出于教育目的探索这些库并遇到文件 os_defines.h ,它在我的 MacBoook 上位于 usr/include/c++/4.2.1/bits 。我在那里找到#include ,所以我快速
在我的代码中,我需要执行 /bin/bash,但我不想在不向它传递任何参数的情况下执行它。我写了这个: execl("/bin/bash", NULL); 然后,通过一些研究,我意识到我还需要添加类型
我试图通过管道进入读取,但它在第二个输入后一直出现段错误。我究竟做错了什么?提前致谢。 $ ./read #include void prompt_name(char *name, char *
据说在C++中包含C头文件时,要去掉“.h”后缀,然后在开头加上“c”。例如,#include 而不是 #include .但是当我使用 sleep()在我的代码中,#include 不起作用,但是
这是我的代码(只是为了测试 fork() 而创建): #include #include #include #include #include #include int main() {
我是一名优秀的程序员,十分优秀!