blob: fa085a25f1c4fe9d8dae18d762edba19295bac19 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (c) 2009 Josef Bacik
* All Rights Reserved.
*/
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <ctype.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/statfs.h>
#include <sys/vfs.h>
#include <linux/fs.h>
#include <linux/types.h>
#include <linux/fiemap.h>
/* Global for non-critical message suppression */
int quiet;
static void
usage(void)
{
printf("Usage: fiemap-tester [-m map] [-r number of runs] [-s seed] [-qS]");
printf("[-p preallocate (1/0)] ");
printf("filename\n");
printf(" -m map : generate a file with the map given and test\n");
printf(" -p 0/1 : turn block preallocation on or off\n");
printf(" -r count : number of runs to execute (default infinity)\n");
printf(" -s seed : seed for random map generator (default 1)\n");
printf(" -q : be quiet about non-errors\n");
printf(" -S : sync file before mapping (via ioctl flags)\n");
printf("-m and -r cannot be used together\n");
exit(EXIT_FAILURE);
}
static char *
generate_file_mapping(int blocks, int prealloc)
{
char *map;
int num_types = 2, cur_block = 0;
int i = 0;
map = malloc(sizeof(char) * blocks);
if (!map)
return NULL;
if (prealloc)
num_types++;
for (i = 0; i < blocks; i++) {
long num = random() % num_types;
switch (num) {
case 0:
map[cur_block] = 'D';
break;
case 1:
map[cur_block] = 'H';
break;
case 2:
map[cur_block] = 'P';
break;
}
cur_block++;
}
return map;
}
static int
create_file_from_mapping(int fd, char *map, int blocks, int blocksize)
{
int cur_offset = 0, ret = 0, bufsize;
char *buf;
int i = 0;
bufsize = sizeof(char) * blocksize;
if (posix_memalign((void **)&buf, 4096, bufsize))
return -1;
memset(buf, 'a', bufsize);
for (i = 0; i < blocks; i++) {
switch (map[i]) {
case 'D':
ret = write(fd, buf, bufsize);
if (ret < bufsize) {
printf("Short write\n");
ret = -1;
goto out;
}
break;
#ifdef HAVE_FALLOCATE
case 'P':
ret = fallocate(fd, 0, cur_offset, blocksize);
if (ret < 0) {
printf("Error fallocating\n");
goto out;
}
/* fallthrough; seek to end of prealloc space */
#endif
case 'H':
ret = lseek(fd, blocksize, SEEK_CUR);
if (ret == (off_t)-1) {
printf("Error lseeking\n");
ret = -1;
goto out;
}
break;
default:
printf("Hrm, unrecognized flag in map\n");
ret = -1;
goto out;
}
cur_offset += blocksize;
}
ret = 0;
out:
return ret;
}
static void
show_extent_block(struct fiemap_extent *extent, int blocksize)
{
__u64 logical = extent->fe_logical;
__u64 phys = extent->fe_physical;
__u64 length = extent->fe_length;
int flags = extent->fe_flags;
printf("logical: [%8llu..%8llu] phys: %8llu..%8llu "
"flags: 0x%03X tot: %llu\n",
logical / blocksize, (logical + length - 1) / blocksize,
phys / blocksize, (phys + length - 1) / blocksize,
flags,
(length / blocksize));
}
static void
show_extents(struct fiemap *fiemap, int blocksize)
{
unsigned int i;
for (i = 0; i < fiemap->fm_mapped_extents; i++)
show_extent_block(&fiemap->fm_extents[i], blocksize);
}
static int
check_flags(struct fiemap *fiemap, int blocksize)
{
struct fiemap_extent *extent;
__u64 aligned_offset, aligned_length;
int c;
for (c = 0; c < fiemap->fm_mapped_extents; c++) {
extent = &fiemap->fm_extents[c];
aligned_offset = extent->fe_physical & ~((__u64)blocksize - 1);
aligned_length = extent->fe_length & ~((__u64)blocksize - 1);
if ((aligned_offset != extent->fe_physical ||
aligned_length != extent->fe_length) &&
!(extent->fe_flags & FIEMAP_EXTENT_NOT_ALIGNED)) {
printf("ERROR: FIEMAP_EXTENT_NOT_ALIGNED is not set "
"but the extent is unaligned: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_DATA_ENCRYPTED &&
!(extent->fe_flags & FIEMAP_EXTENT_ENCODED)) {
printf("ERROR: FIEMAP_EXTENT_DATA_ENCRYPTED is set, "
"but FIEMAP_EXTENT_ENCODED is not set: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_NOT_ALIGNED &&
aligned_offset == extent->fe_physical &&
aligned_length == extent->fe_length) {
printf("ERROR: FIEMAP_EXTENT_NOT_ALIGNED is set but "
"offset and length is blocksize aligned: "
"%llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_LAST &&
c + 1 < fiemap->fm_mapped_extents) {
printf("ERROR: FIEMAP_EXTENT_LAST is set but there are"
" more extents left: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_DELALLOC &&
!(extent->fe_flags & FIEMAP_EXTENT_UNKNOWN)) {
printf("ERROR: FIEMAP_EXTENT_DELALLOC is set but "
"FIEMAP_EXTENT_UNKNOWN is not set: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_DATA_INLINE &&
!(extent->fe_flags & FIEMAP_EXTENT_NOT_ALIGNED)) {
printf("ERROR: FIEMAP_EXTENT_DATA_INLINE is set but "
"FIEMAP_EXTENT_NOT_ALIGNED is not set: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
if (extent->fe_flags & FIEMAP_EXTENT_DATA_TAIL &&
!(extent->fe_flags & FIEMAP_EXTENT_NOT_ALIGNED)) {
printf("ERROR: FIEMAP_EXTENT_DATA_TAIL is set but "
"FIEMAP_EXTENT_NOT_ALIGNED is not set: %llu\n",
(unsigned long long)
(extent->fe_logical / blocksize));
return -1;
}
}
return 0;
}
static int
check_data(struct fiemap *fiemap, __u64 logical_offset, int blocksize,
int prealloc)
{
struct fiemap_extent *extent;
__u64 orig_offset = logical_offset;
int c, found = 0;
for (c = 0; c < fiemap->fm_mapped_extents; c++) {
__u64 start, end;
extent = &fiemap->fm_extents[c];
start = extent->fe_logical;
end = extent->fe_logical + extent->fe_length;
if (logical_offset > end)
continue;
if (logical_offset + blocksize < start)
break;
if (logical_offset >= start &&
logical_offset < end) {
if (prealloc &&
!(extent->fe_flags & FIEMAP_EXTENT_UNWRITTEN)) {
printf("ERROR: preallocated extent is not "
"marked with FIEMAP_EXTENT_UNWRITTEN: "
"%llu\n",
(unsigned long long)
(start / blocksize));
return -1;
}
if (logical_offset + blocksize > end) {
logical_offset = end+1;
continue;
} else {
found = 1;
break;
}
}
}
if (!found) {
printf("ERROR: couldn't find extent at %llu\n",
(unsigned long long)(orig_offset / blocksize));
}
return (!found) ? -1 : 0;
}
static int
check_weird_fs_hole(int fd, __u64 logical_offset, int blocksize)
{
static int warning_printed = 0;
int block, i;
size_t buf_len = sizeof(char) * blocksize;
char *buf;
block = (int)(logical_offset / blocksize);
if (ioctl(fd, FIBMAP, &block) < 0) {
perror("Can't fibmap file");
return -1;
}
if (!block) {
printf("ERROR: FIEMAP claimed there was data at a block "
"which should be a hole, and FIBMAP confirmend that "
"it is in fact a hole, so FIEMAP is wrong: %llu\n",
(unsigned long long)(logical_offset / blocksize));
return -1;
}
buf = malloc(buf_len);
if (!buf) {
perror("Could not allocate temporary buffer");
return -1;
}
if (pread(fd, buf, buf_len, (off_t)logical_offset) < 0) {
perror("Error reading from file");
free(buf);
return -1;
}
for (i = 0; i < buf_len; i++) {
if (buf[i] != 0) {
printf("ERROR: FIEMAP claimed there was data (%c) at "
"block %llu that should have been a hole, and "
"FIBMAP confirmed that it was allocated, but "
"it should be filled with 0's, but it was not "
"so you have a big problem!\n",
buf[i],
(unsigned long long)(logical_offset / blocksize));
free(buf);
return -1;
}
}
if (warning_printed || quiet) {
free(buf);
return 0;
}
printf("HEY FS PERSON: your fs is weird. I specifically wanted a\n"
"hole and you allocated a block anyway. FIBMAP confirms that\n"
"you allocated a block, and the block is filled with 0's so\n"
"everything is kosher, but you still allocated a block when\n"
"didn't need to. This may or may not be what you wanted,\n"
"which is why I'm only printing this message once, in case\n"
"you didn't do it on purpose. This was at block %llu.\n",
(unsigned long long)(logical_offset / blocksize));
warning_printed = 1;
free(buf);
return 0;
}
static int
check_hole(struct fiemap *fiemap, int fd, __u64 logical_offset, int blocksize)
{
struct fiemap_extent *extent;
int c;
for (c = 0; c < fiemap->fm_mapped_extents; c++) {
__u64 start, end;
extent = &fiemap->fm_extents[c];
start = extent->fe_logical;
end = extent->fe_logical + extent->fe_length;
if (logical_offset > end)
continue;
if (logical_offset + blocksize < start)
break;
/*
* Filesystems are allowed to fill in holes with preallocated
* unwritten extents
*/
if (extent->fe_flags & FIEMAP_EXTENT_UNWRITTEN)
continue;
if (logical_offset >= start &&
logical_offset < end) {
if (check_weird_fs_hole(fd, logical_offset,
blocksize) == 0)
break;
printf("ERROR: found an allocated extent where a hole "
"should be: %llu\n",
(unsigned long long)(start / blocksize));
return -1;
}
}
return 0;
}
static int query_fiemap_count(int fd, int blocks, int blocksize)
{
struct fiemap fiemap = { 0, };
fiemap.fm_length = blocks * blocksize;
if (ioctl(fd, FS_IOC_FIEMAP, (unsigned long)&fiemap) < 0) {
perror("FIEMAP query ioctl failed");
return -1;
}
return 0;
}
static int
compare_fiemap_and_map(int fd, char *map, int blocks, int blocksize, int syncfile)
{
struct fiemap *fiemap;
char *fiebuf;
int blocks_to_map, ret, cur_extent = 0;
__u64 map_start, map_length;
int i, c;
if (query_fiemap_count(fd, blocks, blocksize) < 0)
return -1;
blocks_to_map = (random() % blocks) + 1;
fiebuf = malloc(sizeof(struct fiemap) +
(blocks_to_map * sizeof(struct fiemap_extent)));
if (!fiebuf) {
perror("Could not allocate fiemap buffers");
return -1;
}
fiemap = (struct fiemap *)fiebuf;
map_start = 0;
map_length = blocks_to_map * blocksize;
fiemap->fm_flags = syncfile ? FIEMAP_FLAG_SYNC : 0;
fiemap->fm_extent_count = blocks_to_map;
fiemap->fm_mapped_extents = 0;
do {
fiemap->fm_start = map_start;
fiemap->fm_length = map_length;
ret = ioctl(fd, FS_IOC_FIEMAP, (unsigned long)fiemap);
if (ret < 0) {
perror("FIEMAP ioctl failed");
free(fiemap);
return -1;
}
if (check_flags(fiemap, blocksize))
goto error;
for (i = cur_extent, c = 1; i < blocks; i++, c++) {
__u64 logical_offset = i * blocksize;
if (c > fiemap->fm_mapped_extents) {
i++;
break;
}
switch (map[i]) {
case 'D':
if (check_data(fiemap, logical_offset,
blocksize, 0))
goto error;
break;
case 'H':
if (check_hole(fiemap, fd, logical_offset,
blocksize))
goto error;
break;
case 'P':
if (check_data(fiemap, logical_offset,
blocksize, 1))
goto error;
break;
default:
printf("ERROR: weird value in map: %c\n",
map[i]);
goto error;
}
}
cur_extent = i;
map_start = i * blocksize;
} while (cur_extent < blocks);
ret = 0;
return ret;
error:
printf("map is '%s'\n", map);
show_extents(fiemap, blocksize);
return -1;
}
int
main(int argc, char **argv)
{
int blocksize = 0; /* filesystem blocksize */
int fd; /* file descriptor */
int opt;
int rc;
char *fname; /* filename to map */
char *map = NULL; /* file map to generate */
int runs = -1; /* the number of runs to have */
int blocks = 0; /* the number of blocks to generate */
int maxblocks = 0; /* max # of blocks to create */
int prealloc = 1; /* whether or not to do preallocation */
int syncfile = 0; /* whether fiemap should sync file first */
int seed = 1;
while ((opt = getopt(argc, argv, "m:r:s:p:qS")) != -1) {
switch(opt) {
case 'm':
map = strdup(optarg);
break;
case 'p':
prealloc = atoi(optarg);;
#ifndef HAVE_FALLOCATE
if (prealloc) {
printf("Not built with preallocation support\n");
usage();
}
#endif
break;
case 'q':
quiet = 1;
break;
case 'r':
runs = atoi(optarg);
break;
case 's':
seed = atoi(optarg);
break;
/* sync file before mapping */
case 'S':
syncfile = 1;
break;
default:
usage();
}
}
if (runs != -1 && map)
usage();
fname = argv[optind++];
if (!fname)
usage();
fd = open(fname, O_RDWR|O_CREAT|O_TRUNC|O_DIRECT, 0644);
if (fd < 0) {
perror("Can't open file");
exit(1);
}
if (ioctl(fd, FIGETBSZ, &blocksize) < 0) {
struct statfs buf;
if (fstatfs(fd, &buf) == 0) {
blocksize = buf.f_bsize;
} else {
perror("Can't get filesystem block size");
close(fd);
exit(1);
}
}
if (blocksize <= 0) {
printf("Illegal filesystem block size\n");
close(fd);
exit(1);
}
#ifdef HAVE_FALLOCATE
/* if fallocate passes, then we can do preallocation, else not */
if (prealloc) {
prealloc = !((int)fallocate(fd, 0, 0, blocksize));
if (!prealloc)
printf("preallocation not supported, disabling\n");
}
#else
prealloc = 0;
#endif
if (ftruncate(fd, 0)) {
perror("Can't truncate file");
close(fd);
exit(1);
}
if (map) {
blocks = strlen(map);
runs = 0;
}
srandom(seed);
/* max file size 2mb / block size */
maxblocks = (2 * 1024 * 1024) / blocksize;
if (runs == -1)
printf("Starting infinite run, if you don't see any output "
"then its working properly.\n");
do {
if (!map) {
blocks = random() % maxblocks;
if (blocks == 0) {
if (!quiet)
printf("Skipping 0 length file\n");
continue;
}
map = generate_file_mapping(blocks, prealloc);
if (!map) {
printf("Could not create map\n");
exit(1);
}
}
rc = create_file_from_mapping(fd, map, blocks, blocksize);
if (rc) {
perror("Could not create file\n");
free(map);
close(fd);
exit(1);
}
rc = compare_fiemap_and_map(fd, map, blocks, blocksize, syncfile);
if (rc) {
printf("Problem comparing fiemap and map\n");
free(map);
close(fd);
exit(1);
}
free(map);
map = NULL;
if (ftruncate(fd, 0)) {
perror("Could not truncate file\n");
close(fd);
exit(1);
}
if (lseek(fd, 0, SEEK_SET)) {
perror("Could not seek set\n");
close(fd);
exit(1);
}
if (runs) runs--;
} while (runs != 0);
close(fd);
return 0;
}