#define _GNU_SOURCE #define _LARGEFILE64_SOURCE #include #include #include #include #include #include #include #include "buse.h" static int fd; static void usage(void) { fprintf(stderr, "Usage: loopback \n"); } static int loopback_read(void *buf, u_int32_t len, u_int64_t offset) { int bytes_read; lseek64(fd, offset, SEEK_SET); while (len > 0) { bytes_read = read(fd, buf, len); assert(bytes_read > 0); len -= bytes_read; buf = (char *) buf + bytes_read; } return 0; } static int loopback_write(const void *buf, u_int32_t len, u_int64_t offset) { int bytes_written; lseek64(fd, offset, SEEK_SET); while (len > 0) { bytes_written = write(fd, buf, len); assert(bytes_written > 0); len -= bytes_written; buf = (char *) buf + bytes_written; } return 0; } static struct buse_operations bop = { .read = loopback_read, .write = loopback_write }; int main(int argc, char *argv[]) { struct stat buf; int err; int64_t size; if (argc != 3) { usage(); return -1; } fd = open(argv[1], O_RDWR|O_LARGEFILE); assert(fd != -1); /* Let's verify that this file is actually a block device. We could support * regular files, too, but we don't need to right now. */ fstat(fd, &buf); assert(S_ISBLK(buf.st_mode)); /* Figure out the size of the underlying block device. */ err = ioctl(fd, BLKGETSIZE64, &size); assert(err != -1); fprintf(stderr, "The size of this device is %ld bytes.\n", size); bop.size = size; buse_main(argc, argv, &bop, NULL); return 0; }