"Robert P. J. Day" | 63fc1a9 | 2006-07-02 19:47:05 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 2 | /* |
| 3 | * Mini losetup implementation for busybox |
| 4 | * |
| 5 | * Copyright (C) 2002 Matt Kraai. |
| 6 | * |
Denys Vlasenko | 0ef64bd | 2010-08-16 20:14:46 +0200 | [diff] [blame] | 7 | * Licensed under GPLv2 or later, see file LICENSE in this source tree. |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 8 | */ |
Denys Vlasenko | dd898c9 | 2016-11-23 11:46:32 +0100 | [diff] [blame] | 9 | //config:config LOSETUP |
Denys Vlasenko | b097a84 | 2018-12-28 03:20:17 +0100 | [diff] [blame] | 10 | //config: bool "losetup (5.5 kb)" |
Denys Vlasenko | dd898c9 | 2016-11-23 11:46:32 +0100 | [diff] [blame] | 11 | //config: default y |
| 12 | //config: select PLATFORM_LINUX |
| 13 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 14 | //config: losetup is used to associate or detach a loop device with a regular |
| 15 | //config: file or block device, and to query the status of a loop device. This |
| 16 | //config: version does not currently support enabling data encryption. |
Denys Vlasenko | dd898c9 | 2016-11-23 11:46:32 +0100 | [diff] [blame] | 17 | |
Denys Vlasenko | ae84418 | 2017-08-07 23:14:49 +0200 | [diff] [blame] | 18 | //applet:IF_LOSETUP(APPLET_NOEXEC(losetup, losetup, BB_DIR_SBIN, BB_SUID_DROP, losetup)) |
Denys Vlasenko | dd898c9 | 2016-11-23 11:46:32 +0100 | [diff] [blame] | 19 | |
Denys Vlasenko | ae84418 | 2017-08-07 23:14:49 +0200 | [diff] [blame] | 20 | //kbuild:lib-$(CONFIG_LOSETUP) += losetup.o |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 21 | |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 22 | //usage:#define losetup_trivial_usage |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 23 | //usage: "[-rP] [-o OFS] {-f|LOOPDEV} FILE: associate loop devices\n" |
Denys Vlasenko | 309f5e3 | 2019-05-23 16:11:42 +0200 | [diff] [blame] | 24 | //usage: " losetup -c LOOPDEV: reread file size\n" |
| 25 | //usage: " losetup -d LOOPDEV: disassociate\n" |
| 26 | //usage: " losetup -a: show status\n" |
| 27 | //usage: " losetup -f: show next free loop device" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 28 | //usage:#define losetup_full_usage "\n\n" |
Denys Vlasenko | 6642676 | 2011-06-05 03:58:28 +0200 | [diff] [blame] | 29 | //usage: " -o OFS Start OFS bytes into FILE" |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 30 | //usage: "\n -P Scan for partitions" |
Denys Vlasenko | 13e709c | 2011-09-12 02:13:47 +0200 | [diff] [blame] | 31 | //usage: "\n -r Read-only" |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 32 | //usage: "\n -f Show/use next free loop device" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 33 | //usage: |
| 34 | //usage:#define losetup_notes_usage |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 35 | //usage: "One argument (losetup /dev/loop1) will display the current association\n" |
| 36 | //usage: "(if any), or disassociate it (with -d). The display shows the offset\n" |
| 37 | //usage: "and filename of the file the loop device is currently bound to.\n\n" |
| 38 | //usage: "Two arguments (losetup /dev/loop1 file.img) create a new association,\n" |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 39 | //usage: "with optional partition scanning (creates /dev/loop1p1, /dev/loop1p2\n" |
| 40 | //usage: "etc. with -P) and with an optional offset (-o 12345). Encryption is\n" |
| 41 | //usage: "not yet supported. losetup -f will show the first free loop device\n\n" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 42 | |
Denis Vlasenko | b6adbf1 | 2007-05-26 19:00:18 +0000 | [diff] [blame] | 43 | #include "libbb.h" |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 44 | |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 45 | /* 1048575 is a max possible minor number in Linux circa 2010 */ |
| 46 | /* for now use something less extreme */ |
| 47 | #define MAX_LOOP_NUM 1023 |
| 48 | |
Denis Vlasenko | 9b49a5e | 2007-10-11 10:05:36 +0000 | [diff] [blame] | 49 | int losetup_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
Denys Vlasenko | 9251014 | 2010-05-19 00:39:17 +0200 | [diff] [blame] | 50 | int losetup_main(int argc UNUSED_PARAM, char **argv) |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 51 | { |
Denis Vlasenko | 67b23e6 | 2006-10-03 21:00:06 +0000 | [diff] [blame] | 52 | unsigned opt; |
Denis Vlasenko | 27ee7ba | 2006-09-22 14:53:41 +0000 | [diff] [blame] | 53 | char *opt_o; |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 54 | char dev[LOOP_NAMESIZE]; |
Denys Vlasenko | 9251014 | 2010-05-19 00:39:17 +0200 | [diff] [blame] | 55 | enum { |
Denys Vlasenko | 309f5e3 | 2019-05-23 16:11:42 +0200 | [diff] [blame] | 56 | OPT_c = (1 << 0), |
| 57 | OPT_d = (1 << 1), |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 58 | OPT_P = (1 << 2), |
| 59 | OPT_o = (1 << 3), |
| 60 | OPT_f = (1 << 4), |
| 61 | OPT_a = (1 << 5), |
| 62 | OPT_r = (1 << 6), |
Denys Vlasenko | 9251014 | 2010-05-19 00:39:17 +0200 | [diff] [blame] | 63 | }; |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 64 | |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 65 | opt = getopt32(argv, "^" "cdPo:far" "\0" "?2:d--Pofar:a--Pofr", &opt_o); |
Denis Vlasenko | 27ee7ba | 2006-09-22 14:53:41 +0000 | [diff] [blame] | 66 | argv += optind; |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 67 | |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 68 | /* LOOPDEV */ |
| 69 | if (!opt && argv[0] && !argv[1]) { |
Denys Vlasenko | 9251014 | 2010-05-19 00:39:17 +0200 | [diff] [blame] | 70 | char *s; |
Denis Vlasenko | 7ae209c | 2007-09-26 17:54:18 +0000 | [diff] [blame] | 71 | |
Denis Vlasenko | 7ae209c | 2007-09-26 17:54:18 +0000 | [diff] [blame] | 72 | s = query_loop(argv[0]); |
Denis Vlasenko | c34d355 | 2007-04-19 00:09:34 +0000 | [diff] [blame] | 73 | if (!s) |
Denis Vlasenko | 94e3365 | 2007-12-22 15:44:23 +0000 | [diff] [blame] | 74 | bb_simple_perror_msg_and_die(argv[0]); |
Denis Vlasenko | 27ee7ba | 2006-09-22 14:53:41 +0000 | [diff] [blame] | 75 | printf("%s: %s\n", argv[0], s); |
Denis Vlasenko | c34d355 | 2007-04-19 00:09:34 +0000 | [diff] [blame] | 76 | if (ENABLE_FEATURE_CLEAN_UP) |
| 77 | free(s); |
Denis Vlasenko | 7ae209c | 2007-09-26 17:54:18 +0000 | [diff] [blame] | 78 | return EXIT_SUCCESS; |
| 79 | } |
| 80 | |
Denys Vlasenko | 309f5e3 | 2019-05-23 16:11:42 +0200 | [diff] [blame] | 81 | /* -c LOOPDEV */ |
| 82 | if (opt == OPT_c && argv[0]) { |
| 83 | int fd = xopen(argv[0], O_RDONLY); |
| 84 | #ifndef LOOP_SET_CAPACITY |
| 85 | # define LOOP_SET_CAPACITY 0x4C07 |
| 86 | #endif |
| 87 | xioctl(fd, LOOP_SET_CAPACITY, /*ignored:*/0); |
| 88 | return EXIT_SUCCESS; |
| 89 | } |
| 90 | |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 91 | /* -d LOOPDEV */ |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 92 | if (opt == OPT_d && argv[0]) { |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 93 | if (del_loop(argv[0])) |
| 94 | bb_simple_perror_msg_and_die(argv[0]); |
| 95 | return EXIT_SUCCESS; |
| 96 | } |
Denys Vlasenko | 9251014 | 2010-05-19 00:39:17 +0200 | [diff] [blame] | 97 | |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 98 | /* -a */ |
| 99 | if (opt == OPT_a) { |
| 100 | int n; |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 101 | for (n = 0; n < MAX_LOOP_NUM; n++) { |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 102 | char *s; |
| 103 | |
| 104 | sprintf(dev, LOOP_FORMAT, n); |
| 105 | s = query_loop(dev); |
| 106 | if (s) { |
Denis Vlasenko | 7ae209c | 2007-09-26 17:54:18 +0000 | [diff] [blame] | 107 | printf("%s: %s\n", dev, s); |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 108 | free(s); |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 109 | } |
| 110 | } |
| 111 | return EXIT_SUCCESS; |
| 112 | } |
| 113 | |
| 114 | /* contains -f */ |
| 115 | if (opt & OPT_f) { |
| 116 | char *s; |
Denys Vlasenko | 3b69ba7 | 2019-06-09 23:20:49 +0200 | [diff] [blame] | 117 | int n; |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 118 | |
Denys Vlasenko | 3b69ba7 | 2019-06-09 23:20:49 +0200 | [diff] [blame] | 119 | n = get_free_loop(); |
| 120 | if (n == -1) |
James Byrne | 6937487 | 2019-07-02 11:35:03 +0200 | [diff] [blame] | 121 | bb_simple_error_msg_and_die("no free loop devices"); |
Denys Vlasenko | 3b69ba7 | 2019-06-09 23:20:49 +0200 | [diff] [blame] | 122 | if (n < 0) /* n == -2: no /dev/loop-control, use legacy method */ |
| 123 | n = 0; |
| 124 | /* or: n >= 0: the number of next free loopdev, just verify it */ |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 125 | do { |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 126 | if (n > MAX_LOOP_NUM) |
James Byrne | 6937487 | 2019-07-02 11:35:03 +0200 | [diff] [blame] | 127 | bb_simple_error_msg_and_die("no free loop devices"); |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 128 | sprintf(dev, LOOP_FORMAT, n++); |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 129 | s = query_loop(dev); |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 130 | free(s); |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 131 | } while (s); |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 132 | /* now: dev is next free "/dev/loopN" */ |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 133 | if ((opt == OPT_f) && !argv[0]) { |
| 134 | puts(dev); |
| 135 | return EXIT_SUCCESS; |
Denis Vlasenko | 956a569 | 2006-09-27 14:51:27 +0000 | [diff] [blame] | 136 | } |
| 137 | } |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 138 | |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 139 | /* [-rP] [-o OFS] {-f|LOOPDEV} FILE */ |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 140 | if (argv[0] && ((opt & OPT_f) || argv[1])) { |
| 141 | unsigned long long offset = 0; |
| 142 | char *d = dev; |
| 143 | |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 144 | if (opt & OPT_o) |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 145 | offset = xatoull(opt_o); |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 146 | if (!(opt & OPT_f)) |
| 147 | d = *argv++; |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 148 | |
| 149 | if (argv[0]) { |
Jack O'Sullivan | 726cbb1 | 2019-05-28 15:28:27 +0100 | [diff] [blame] | 150 | unsigned flags = (opt & OPT_r) ? BB_LO_FLAGS_READ_ONLY : 0; |
| 151 | if (opt & OPT_P) { |
| 152 | flags |= BB_LO_FLAGS_PARTSCAN; |
| 153 | } |
| 154 | if (set_loop(&d, argv[0], offset, flags) < 0) |
Mandeep Singh Baines | 7991d45 | 2013-03-04 16:33:12 -0800 | [diff] [blame] | 155 | bb_simple_perror_msg_and_die(argv[0]); |
| 156 | return EXIT_SUCCESS; |
| 157 | } |
| 158 | } |
| 159 | |
Denys Vlasenko | ab518ee | 2017-03-16 16:49:37 +0100 | [diff] [blame] | 160 | /* TODO: util-linux 2.28 shows this when run w/o params: |
| 161 | * NAME SIZELIMIT OFFSET AUTOCLEAR RO BACK-FILE DIO |
| 162 | * /dev/loop0 0 0 1 0 /PATH/TO/FILE 0 |
| 163 | * |
| 164 | * implemented by reading /sys: |
| 165 | * |
| 166 | * open("/sys/block", O_RDONLY|O_NONBLOCK|O_DIRECTORY|O_CLOEXEC) = 3 |
| 167 | * newfstatat(3, "loop0/loop/backing_file", {st_mode=S_IFREG|0444, st_size=4096, ...}, 0) = 0 |
| 168 | * stat("/dev/loop0", {st_mode=S_IFBLK|0660, st_rdev=makedev(7, 0), ...}) = 0 |
| 169 | * open("/sys/dev/block/7:0/loop/offset", O_RDONLY|O_CLOEXEC) = 5 |
| 170 | * read(5, "0\n", 4096) = 2 |
| 171 | * open("/sys/dev/block/7:0/loop/sizelimit", O_RDONLY|O_CLOEXEC) = 5 |
| 172 | * read(5, "0\n", 4096) = 2 |
| 173 | * open("/sys/dev/block/7:0/loop/offset", O_RDONLY|O_CLOEXEC) = 5 |
| 174 | * read(5, "0\n", 4096) = 2 |
| 175 | * open("/sys/dev/block/7:0/loop/autoclear", O_RDONLY|O_CLOEXEC) = 5 |
| 176 | * read(5, "1\n", 4096) = 2 |
| 177 | * open("/sys/dev/block/7:0/ro", O_RDONLY|O_CLOEXEC) = 5 |
| 178 | * read(5, "0\n", 4096) = 2 |
| 179 | * open("/sys/dev/block/7:0/loop/backing_file", O_RDONLY|O_CLOEXEC) = 5 |
| 180 | * read(5, "/PATH/TO/FILE", 4096) = 37 |
| 181 | * open("/sys/dev/block/7:0/loop/dio", O_RDONLY|O_CLOEXEC) = 5 |
| 182 | * read(5, "0\n", 4096) = 2 |
| 183 | */ |
| 184 | |
Denys Vlasenko | 4928e9f | 2013-06-27 03:45:16 +0200 | [diff] [blame] | 185 | bb_show_usage(); /* does not return */ |
| 186 | /*return EXIT_FAILURE;*/ |
Matt Kraai | 83788da | 2002-03-20 17:38:37 +0000 | [diff] [blame] | 187 | } |