1/* vi: set sw=4 ts=4: */ 2/* 3 * Copyright (C) 2017 Denys Vlasenko <vda.linux@googlemail.com> 4 * 5 * Licensed under GPLv2, see file LICENSE in this source tree. 6 */ 7//config:config FALLOCATE 8//config: bool "fallocate (5 kb)" 9//config: default y 10//config: help 11//config: Preallocate space for files. 12 13//applet:IF_FALLOCATE(APPLET(fallocate, BB_DIR_USR_BIN, BB_SUID_DROP)) 14 15//kbuild:lib-$(CONFIG_FALLOCATE) += fallocate.o 16 17//usage:#define fallocate_trivial_usage 18//usage: "[-o OFS] -l LEN FILE" 19// fallocate [-c|-p|-z] [-n] [-o OFS] -l LEN FILE 20// fallocate -d [-o OFS] [-l LEN] FILE 21//usage:#define fallocate_full_usage "\n\n" 22//usage: "Preallocate space for FILE\n" 23// "\n -c Remove range" 24// "\n -p Make hole" 25// "\n -z Zero and allocate range" 26// "\n -d Convert zeros to holes" 27// "\n -n Keep size" 28//usage: "\n -o OFS Offset of range" 29//usage: "\n -l LEN Length of range" 30 31//Upstream options: 32//The options --collapse-range, --dig-holes, --punch-hole and --zero-range 33//are mutually exclusive. 34//-c, --collapse-range 35// Removes a byte range from a file, without leaving a hole. The byte range 36// to be collapsed starts at offset and continues for length bytes. 37// At the completion of the operation, the contents of the file starting 38// at the location offset+length will be appended at the location offset, 39// and the file will be length bytes smaller. The option --keep-size may 40// not be specified for the collapse-range operation. 41//-d, --dig-holes 42// Detect and dig holes. This makes the file sparse in-place, without using 43// extra disk space. The minimum size of the hole depends on filesystem I/O 44// block size (usually 4096 bytes). Also, 45//-l, --length length 46// Specifies the length of the range, in bytes. 47//-n, --keep-size 48// Do not modify the apparent length of the file. This may effectively 49// allocate blocks past EOF, which can be removed with a truncate. 50//-o, --offset offset 51// Specifies the beginning offset of the range, in bytes. 52//-p, --punch-hole 53// Deallocates space (i.e., creates a hole) in the byte range starting 54// at offset and continuing for length bytes. Within the specified range, 55// partial filesystem blocks are zeroed, and whole 56// filesystem blocks are removed from the file. After a successful call, 57// subsequent reads from this range will return zeroes. This option may not 58// be specified at the same time as the 59// --zero-range option. Also, when using this option, --keep-size is implied. 60//-z, --zero-range 61// Zeroes space in the byte range starting at offset and continuing for 62// length bytes. Within the specified range, blocks are preallocated for 63// the regions that span the holes in the file. After 64// a successful call, subsequent reads from this range will return zeroes. 65// Zeroing is done within the filesystem preferably by converting the range 66// into unwritten extents. This approach means that the specified range 67// will not be physically zeroed out on the device (except for partial 68// blocks at the either end of the range), and I/O is (otherwise) required 69// only to update metadata. 70// Option --keep-size can be specified to prevent file length modification. 71 72#include "libbb.h" 73 74int fallocate_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; 75int fallocate_main(int argc UNUSED_PARAM, char **argv) 76{ 77 const char *str_l; 78 const char *str_o = "0"; 79 off_t ofs, len; 80 unsigned opts; 81 int fd; 82 83 /* exactly one non-option arg */ 84 opts = getopt32(argv, "^" "l:o:" "\0" "=1", &str_l, &str_o); 85 if (!(opts & 1)) 86 bb_show_usage(); 87 88 ofs = xatoull_sfx(str_o, kmg_i_suffixes); 89 len = xatoull_sfx(str_l, kmg_i_suffixes); 90 91 argv += optind; 92 fd = xopen3(*argv, O_RDWR | O_CREAT, 0666); 93 94 /* posix_fallocate has unusual method of returning error */ 95 /* maybe use Linux-specific fallocate(int fd, int mode, off_t offset, off_t len) instead? */ 96 if ((errno = posix_fallocate(fd, ofs, len)) != 0) 97 bb_perror_msg_and_die("fallocate '%s'", *argv); 98 99 /* util-linux also performs fsync(fd); */ 100 101 return EXIT_SUCCESS; 102} 103