C:如何使用mount(2)挂载使用mkfs创建的常规文件系统?

时间:2015-11-20 20:00:22

标签: c unix mount

这就是我经常"定期"文件系统:

$ touch container
$ mkfs.ext4 container 20000

如何使用C函数mount(2)安装它?

实际上mount告诉我:

  

需要阻止设备

有没有办法在C中创建文件系统?

2 个答案:

答案 0 :(得分:1)

问题是mount库调用没有设置循环设备。要安装容器,您需要将-o loop传递给mount命令,该命令实际上包括losetup所做的事情(即使/dev/loopN指向您的文件,坐骑)。使用库调用时,需要模拟它,即首先需要设置循环设备。

要准确确定您需要做什么,请尝试:

strace mount -t ext4 -o loop container /mnt

并查看系统调用。您可以看到发生的情况是,首先它创建一个循环设备(例如/dev/loop0使用相当于losetup的系统调用),然后安装该循环设备。

以下是strace输出的关键行:

首先分配循环设备:

stat("/dev/loop", 0x7fffe90b00b0)       = -1 ENOENT (No such file or directory)
open("/dev/loop0", O_RDONLY)            = 3
fstat(3, {st_mode=S_IFBLK|0660, st_rdev=makedev(7, 0), ...}) = 0
ioctl(3, LOOP_GET_STATUS, {number=39736224, offset=0x6f726763, encrypt_type=0x732f7075 /* LO_CRYPT_??? */, encrypt_key_size=1702130553, flags=LO_FLAGS_READ_ONLY|LO_FLAGS_AUTOCLEAR|LO_FLAGS_PARTSCAN|0x63206460, name="group rw,noexec,nosuid,nodev,none,name=systemd 0 0", encrypt_key="s 0 0\0\0e-agent.devices 0 0\0\0\320\177\0\0", ...}) = -1 ENXIO (No such device or address)
close(3)                                = 0
open("/home/amb/so/container", O_RDWR)  = 3
open("/dev/loop0", O_RDWR)              = 4
readlink("/home", 0x7fffe90adfa0, 4096) = -1 EINVAL (Invalid argument)
readlink("/home/amb", 0x7fffe90adfa0, 4096) = -1 EINVAL (Invalid argument)
readlink("/home/amb/so", 0x7fffe90adfa0, 4096) = -1 EINVAL (Invalid argument)
readlink("/home/amb/so/container", 0x7fffe90adfa0, 4096) = -1 EINVAL (Invalid argument)
ioctl(4, LOOP_SET_FD, 0x3)              = 0
close(3)                                = 0
ioctl(4, LOOP_SET_STATUS64, {offset=0, number=0, flags=LO_FLAGS_AUTOCLEAR, file_name="/home/amb/so/container", ...}) = 0
ioctl(4, LOOP_GET_STATUS64, {offset=0, number=0, flags=LO_FLAGS_AUTOCLEAR, file_name="/home/amb/so/container", ...}) = 0

然后才进行mount

mount("/dev/loop0", "/mnt", "ext4", MS_MGC_VAL, NULL) = 0

我会建议你向mount致意,除非你有充分理由不这样做。这样可以确保维持/etc/mtab等。

答案 1 :(得分:0)

作为abligh答案的后续内容,设置循环设备的代码是

int device = open(device_path, O_RDWR);
/* omitting error checking on device for clarity */
char buf[20];
int success = 0;
for (i = 0; i < 999999999) {
    sprintf(buf, "/dev/loop%d", i);
    int h = open(buf, O_RDWR);
    if (h > -1) {
        int r= ioctl(h, LOOP_SET_FD, device);
        close(h);
        if (!r) {
           success = 1;
           break;
        }
    } else if (errno == ENOENT)
        break;
}
if (!success) {
     /* no more free loop devices or you're not root */
}