|  | // SPDX-License-Identifier: GPL-2.0+ | 
|  | /* | 
|  | * t_stripealign.c | 
|  | * Print whether the file start block is stripe-aligned. | 
|  | * Copyright (c) 2010 Eric Sandeen <sandeen@sandeen.net> | 
|  | */ | 
|  | #include <unistd.h> | 
|  | #include <stdlib.h> | 
|  | #include <sys/types.h> | 
|  | #include <sys/vfs.h> | 
|  | #include <fcntl.h> | 
|  | #include <stdio.h> | 
|  | #include <sys/ioctl.h> | 
|  | #include <linux/fiemap.h> | 
|  | #include <linux/fs.h> | 
|  |  | 
|  | #ifndef FIEMAP_EXTENT_SHARED | 
|  | # define FIEMAP_EXTENT_SHARED	0x00002000 | 
|  | #endif | 
|  |  | 
|  | #define FIEMAP_EXTENT_ACCEPTABLE	(FIEMAP_EXTENT_LAST | \ | 
|  | FIEMAP_EXTENT_DATA_ENCRYPTED | FIEMAP_EXTENT_ENCODED | \ | 
|  | FIEMAP_EXTENT_UNWRITTEN | FIEMAP_EXTENT_MERGED | \ | 
|  | FIEMAP_EXTENT_SHARED) | 
|  |  | 
|  | /* | 
|  | * If only filename given, print first block. | 
|  | * | 
|  | * If filename & sunit (in blocks) given, print whether we are well-aligned | 
|  | */ | 
|  |  | 
|  | int main(int argc, char ** argv) | 
|  | { | 
|  | struct statfs		sb; | 
|  | struct fiemap		*fie; | 
|  | struct fiemap_extent	*fe; | 
|  | int			fd; | 
|  | int			ret; | 
|  | int			sunit = 0;	/* in blocks */ | 
|  | char			*filename; | 
|  | unsigned long long	block; | 
|  |  | 
|  | if (argc < 3) { | 
|  | printf("Usage: %s <filename> <sunit in blocks>\n", argv[0]); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | filename = argv[1]; | 
|  | sunit = atoi(argv[2]); | 
|  |  | 
|  | fd = open(filename, O_RDONLY); | 
|  | if (fd < 0) { | 
|  | perror("can't open file\n"); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | ret = fstatfs(fd, &sb); | 
|  | if (ret) { | 
|  | perror(filename); | 
|  | close(fd); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | fie = calloc(1, sizeof(struct fiemap) + sizeof(struct fiemap_extent)); | 
|  | if (!fie) { | 
|  | close(fd); | 
|  | perror("malloc"); | 
|  | return 1; | 
|  | } | 
|  | fie->fm_length = 1; | 
|  | fie->fm_flags = FIEMAP_FLAG_SYNC; | 
|  | fie->fm_extent_count = 1; | 
|  |  | 
|  | ret = ioctl(fd, FS_IOC_FIEMAP, fie); | 
|  | if (ret < 0) { | 
|  | unsigned int	bmap = 0; | 
|  |  | 
|  | ret = ioctl(fd, FIBMAP, &bmap); | 
|  | if (ret <= 0) { | 
|  | if (ret < 0) | 
|  | perror("fibmap"); | 
|  | else | 
|  | fprintf(stderr, "fibmap returned no result\n"); | 
|  | free(fie); | 
|  | close(fd); | 
|  | return 1; | 
|  | } | 
|  | block = bmap; | 
|  | goto check; | 
|  | } | 
|  |  | 
|  |  | 
|  | if (fie->fm_mapped_extents != 1) { | 
|  | printf("%s: no extents?\n", filename); | 
|  | free(fie); | 
|  | close(fd); | 
|  | return 1; | 
|  | } | 
|  | fe = &fie->fm_extents[0]; | 
|  | if (fe->fe_flags & ~FIEMAP_EXTENT_ACCEPTABLE) { | 
|  | printf("%s: bad flags 0x%x\n", filename, fe->fe_flags); | 
|  | free(fie); | 
|  | close(fd); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | block = fie->fm_extents[0].fe_physical / sb.f_bsize; | 
|  | check: | 
|  | if (block % sunit) { | 
|  | printf("%s: Start block %llu not multiple of sunit %u\n", | 
|  | filename, block, sunit); | 
|  | return 1; | 
|  | } else | 
|  | printf("%s: well-aligned\n", filename); | 
|  | free(fie); | 
|  | close(fd); | 
|  |  | 
|  | return 0; | 
|  | } |