#include <sys/mount.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <linux/loop.h>
#include <errno.h>
#define LOOPDEV_MAXLEN 64
#define LOOP_MAJOR 7
static int is_loop(char *dev)
{
struct stat st;
int ret = 0;
if (stat(dev, &st) == 0) {
if (S_ISBLK(st.st_mode) && (major(st.st_rdev) == LOOP_MAJOR)) {
ret = 1;
}
}
return ret;
}
static int is_loop_mount(const char* path, char *loopdev)
{
FILE* f;
int count;
char device[256];
char mount_path[256];
char rest[256];
int result = 0;
int path_length = strlen(path);
f = fopen("/proc/mounts", "r");
if (!f) {
fprintf(stdout, "could not open /proc/mounts: %s\n", strerror(errno));
return -1;
}
do {
count = fscanf(f, "%255s %255s %255s\n", device, mount_path, rest);
if (count == 3) {
if (is_loop(device) && strcmp(path, mount_path) == 0) {
strlcpy(loopdev, device, LOOPDEV_MAXLEN);
result = 1;
break;
}
}
} while (count == 3);
fclose(f);
return result;
}
int umount_main(int argc, char *argv[])
{
int loop, loop_fd;
char loopdev[LOOPDEV_MAXLEN];
if(argc != 2) {
fprintf(stderr,"umount <path>\n");
return 1;
}
loop = is_loop_mount(argv[1], loopdev);
if (umount(argv[1])) {
fprintf(stderr, "failed: %s\n", strerror(errno));
return 1;
}
if (loop) {
// free the loop device
loop_fd = open(loopdev, O_RDONLY);
if (loop_fd < 0) {
fprintf(stderr, "open loop device failed: %s\n", strerror(errno));
return 1;
}
if (ioctl(loop_fd, LOOP_CLR_FD, 0) < 0) {
fprintf(stderr, "ioctl LOOP_CLR_FD failed: %s\n", strerror(errno));
return 1;
}
close(loop_fd);
}
return 0;
}