一尘不染

可以在文件名中使用“ /”吗?

linux

我知道这不是应该做的事情,但是有没有办法使用斜线字符来分隔Linux中文件名中的目录?


阅读 772

收藏
2020-06-02

共1个答案

一尘不染

答案是,除非文件系统有错误,否则您将无法这样做。原因如下:

有一个系统调用,用于重命名fs/namei.c名为的文件renameat

SYSCALL_DEFINE4(renameat, int, olddfd, const char __user *, oldname,
                int, newdfd, const char __user *, newname)

当系统调用被调用时,它将do_path_lookup对名称进行路径查找()。继续跟踪,我们可以看到link_path_walk

static int link_path_walk(const char *name, struct nameidata *nd)
{
       struct path next;
       int err;
       unsigned int lookup_flags = nd->flags;

       while (*name=='/')
              name++;
       if (!*name)
              return 0;
...

此代码适用于任何文件系统。什么意思
这意味着,如果您尝试'/'使用传统方式将带有实际字符的参数作为文件名传递,它将无法执行您想要的操作。无法逃脱角色。如果文件系统“支持”此文件,则是因为它们要么:

  • 使用unicode字符或 看起来 像斜杠但不是的字符。
  • 他们有一个错误。

此外,如果 确实
要编辑字节以在文件名中添加斜杠字符,则可能会发生不良情况。那是因为您永远无法使用名称:(来引用该文件,因为无论何时,Linux都会假定您所引用的目录不存在。使用’rm
*’技术也不起作用,因为bash只是将其扩展为文件名。甚至rm -rf都行不通,因为一个简单的strace揭示了事情在幕后进行的过程(简称):

$ ls testdir
myfile2 out
$ strace -vf rm -rf testdir
...
unlinkat(3, "myfile2", 0)               = 0
unlinkat(3, "out", 0)                   = 0
fcntl(3, F_GETFD)                       = 0x1 (flags FD_CLOEXEC)
close(3)                                = 0
unlinkat(AT_FDCWD, "testdir", AT_REMOVEDIR) = 0
...

请注意,对这些的调用unlinkat将失败,因为它们需要按名称引用文件。

2020-06-02