【发布时间】:2021-08-26 00:27:04
【问题描述】:
我面临着在不使用任何库的情况下(在控制台中)打印值的问题。我试图通过主函数的返回值来做到这一点,但似乎没有办法在不使用 printf() 的情况下在控制台中显示函数的返回值。我正在寻找一条建议,所以请随意“集思广益”。欢迎任何想法,希望您能帮助我并在此之前感谢您。
【问题讨论】:
-
只有实现提供的功能才有可能。
标签: c printf c-standard-library
我面临着在不使用任何库的情况下(在控制台中)打印值的问题。我试图通过主函数的返回值来做到这一点,但似乎没有办法在不使用 printf() 的情况下在控制台中显示函数的返回值。我正在寻找一条建议,所以请随意“集思广益”。欢迎任何想法,希望您能帮助我并在此之前感谢您。
【问题讨论】:
标签: c printf c-standard-library
从技术上讲,write(注意 write 而不是 fwrite)不是库函数,而是系统调用。 stdout(将出现在屏幕上),如果您能够使用write 找到它的文件描述符编号(提示提示,STDOUT_FILENO),则可以写入。
希望对您有所帮助!如果您需要更多指导,请随时发表评论!
【讨论】:
STDIN_FILENO, STDOUT_FILENO, STDERR_FILENO 提供标准文件描述符编号 0, 1, 2。您暗示了这一点,但始终要明确文件描述符和文件流指针之间的区别(例如FILE*)。话虽如此,你肯定为他提供了很好的信息来跟进。
void* 到write() 的参数时,这可能会令人生畏@ 987654332@
好吧,当您说 write() 时,您所说的“技术上”仍然是标准库中的“技术上”(尽管它可能在 .h 文件中具有一些宏魔法,具体取决于您的编译器)。存在调用每个系统调用的存根函数。
“技术上”你需要知道你的 ABI 并且你需要知道如何在你的编译器中调用那个 ABI。然后你需要(通常,对于大多数 ABIs)当前写入的系统调用号是多少等等。
在 linux 上查找它(这不是我的家乡,但它有可能是你的) SYS_WRITE 是 linux 上的系统调用 #1。你把那个放在 %rax 中,fd 放在 %rdi 中,缓冲区指针放在 %rsi 中,字符数放在 %rdx 中。
所以我们开始:
mov edx,4 ; message length
mov ecx,msg ; message to write
mov ebx,1 ; file descriptor (stdout)
mov eax,4 ; system call number (sys_write)
int 0x80 ; call kernel
那些在家阅读的人可能记得你的 BIOS 曾经说过一些关于 INT 80 的内容。 INT 80 相同。
现在您只需将该程序集放入您的 C 文件中。看起来像 gcc 你使用 asm
所以...
main()
{
__asm__("mov edx,4 ; message length"
"mov ecx,msg ; message to write"
"mov ebx,1 ; file descriptor (stdout)"
"mov eax,4 ; system call number (sys_write)"
"int 0x80 ; call kernel");
}
现在...我不会在这里告诉您如何将您的 C 字符串放入寄存器,但我再次说“技术上”...
注意:另外,请注意,我从这里的两个不同示例中复制了一些代码来说明我的观点。 X86 上的寄存器名称一团糟,我认为该段落使用的命名约定与代码 sn-p 不同。
NB2:为了更加迂腐,“技术上”crt.o 可能被认为是标准库。这取决于。有时它也是一个链接器脚本。这当然是您的编译器工具链为您提供的东西。我不打算在这里复制它,但对于您感兴趣的任何平台,谷歌“最小二进制”或类似的东西。很多人都为他们的“频道”做过这种事情......
【讨论】: