0

我正在用 C 编写自己的小 shell,或者至少我要从今天开始。我想实现三个功能。

  • remove file1

我需要什么系统调用?只需打开文件并删除其内容,或者除了使用之外还有系统调用删除文件rm吗?

  • mycopy source destination

为此,我正在考虑有一个缓冲区并打开 file1 读入缓冲区并写出到 file2 但我不知道如何将其实际放入代码中,如果有人可以将一个小例子放在一起,这将有助于一堆。

  • move source destination

这只是实现复制和删除吗?

我的主要问题是使用我从未做过的 argv[] 参数,所以它对我来说是新的。

4

2 回答 2

5

您可以使用strace查看 linux coreutils 使用的系统调用。系统调用也有man页面,您可以查找这些页面(例如man unlink)。

例如...

jason@io /tmp $ strace rm foo
execve("/bin/rm", ["rm", "foo"], [/* 66 vars */]) = 0
brk(0)                                  = 0x20dd000
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7f2a46c76000
access("/etc/ld.so.preload", R_OK)      = -1 ENOENT (No such file or directory)
open("/etc/ld.so.cache", O_RDONLY|O_CLOEXEC) = 3
fstat(3, {st_mode=S_IFREG|0644, st_size=135313, ...}) = 0
mmap(NULL, 135313, PROT_READ, MAP_PRIVATE, 3, 0) = 0x7f2a46c54000
close(3)                                = 0
open("/lib64/libc.so.6", O_RDONLY|O_CLOEXEC) = 3
read(3, "\177ELF\2\1\1\3\0\0\0\0\0\0\0\0\3\0>\0\1\0\0\0\300\370\1\0\0\0\0\0"..., 832) = 832
fstat(3, {st_mode=S_IFREG|0755, st_size=1677176, ...}) = 0
mmap(NULL, 3784472, PROT_READ|PROT_EXEC, MAP_PRIVATE|MAP_DENYWRITE, 3, 0) = 0x7f2a466ba000
mprotect(0x7f2a4684d000, 2093056, PROT_NONE) = 0
mmap(0x7f2a46a4c000, 24576, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_DENYWRITE, 3, 0x192000) = 0x7f2a46a4c000
mmap(0x7f2a46a52000, 16152, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_ANONYMOUS, -1, 0) = 0x7f2a46a52000
close(3)                                = 0
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7f2a46c53000
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7f2a46c52000
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7f2a46c51000
arch_prctl(ARCH_SET_FS, 0x7f2a46c52700) = 0
mprotect(0x7f2a46a4c000, 16384, PROT_READ) = 0
mprotect(0x7f2a46c77000, 4096, PROT_READ) = 0
munmap(0x7f2a46c54000, 135313)          = 0
brk(0)                                  = 0x20dd000
brk(0x20fe000)                          = 0x20fe000
open("/usr/lib64/locale/locale-archive", O_RDONLY|O_CLOEXEC) = 3
fstat(3, {st_mode=S_IFREG|0644, st_size=1607712, ...}) = 0
mmap(NULL, 1607712, PROT_READ, MAP_PRIVATE, 3, 0) = 0x7f2a46ac8000
close(3)                                = 0
ioctl(0, TCGETS, {B38400 opost isig icanon echo ...}) = 0
newfstatat(AT_FDCWD, "foo", {st_mode=S_IFREG|0644, st_size=0, ...}, AT_SYMLINK_NOFOLLOW) = 0
geteuid()                               = 1000
newfstatat(AT_FDCWD, "foo", {st_mode=S_IFREG|0644, st_size=0, ...}, AT_SYMLINK_NOFOLLOW) = 0
faccessat(AT_FDCWD, "foo", W_OK)        = 0
unlinkat(AT_FDCWD, "foo", 0)            = 0
lseek(0, 0, SEEK_CUR)                   = -1 ESPIPE (Illegal seek)
close(0)                                = 0
close(1)                                = 0
close(2)                                = 0
exit_group(0)                           = ?
+++ exited with 0 +++
于 2015-11-17T17:51:54.440 回答
4

我正在用 C 编写自己的小 shell,或者至少我要从今天开始。

如果你正在编写一个shell,你不需要实现这些功能;它们已经作为 coreutils 可执行文件rmcpmv. 这些通常不是外壳的一部分,也不需要是。

不管!

./remove file1 我需要什么系统调用?只需打开文件并删除其内容,或者除了使用 rm 之外是否还有系统调用来删除文件?

unlink().

./mycopy source destination 为此,我正在考虑让一个缓冲区和打开的 file1 读入缓冲区并写出到 file2 但我不知道如何将它实际放入代码中,如果有人可以拼凑一个小例子来帮助束。

正确的。没有“一步”要求。只是open()(两次)read()、 和write()

然后 ./move 只会实现复制和删除,对吗?

不!这对于大文件来说真的很慢,而且对于目录根本不起作用。您想要的系统调用是rename().

(但是,请注意,这rename()不适用于跨文件系统。在这种情况下,您确实需要复制和删除文件……如果您要移动目录,甚至可能递归。)

于 2015-11-17T18:28:03.577 回答