Linux API调用实现rm,mv和cp

时间:2015-11-17 17:41:34

标签: c linux system-calls gnu-coreutils

我用C写自己的小贝壳,或者至少我今天要开始。我想实现三个功能。

  • remove file1

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

  • mycopy source destination

为此,我想要一个缓冲区并打开file1读入缓冲区并写出文件2,但我不知道如何将这个实际放入代码中,如果有人可以把一个小例子放在一起帮助一个束。

  • move source destination

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

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

2 个答案:

答案 0 :(得分:4)

您可以使用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 +++

答案 1 :(得分:3)

  

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

如果您正在编写 shell ,则无需实现这些功能;它们已作为coreutils可执行文件rmcpmv存在。这些通常不是shell的一部分,也不一定是。

不管!

  

./ remove file1我需要进行哪些系统调用?只是打开文件并删除其内容,或者是否有系统调用删除文件,除了使用rm?

unlink()

  

./ mycopy source destination为此我想到有一个缓冲区并打开file1读入缓冲区并写出到file2但我不知道如何将其实际放入代码中如果有人可以把一个小例子放在一起会有所帮助。

正确。对此没有“单步”的呼吁;只需open()(两次),read()write()

  

然后./move只会实现复制并删除相当多的权利吗?

没有!这对于大型文件来说确实很慢,并且根本无法用于目录。您想要的系统调用是rename()

(但请注意,rename()在文件系统中不起作用。在这种情况下,如果要移动目录,则需要复制和删除文件......甚至可能是递归的。)