File: [Development] / xfs-cmds / xfsprogs / io / bmap.c (download)
Revision 1.1, Tue Mar 25 03:48:48 2003 UTC (14 years, 6 months ago) by nathans
Branch: MAIN
Add xfs_io into xfsprogs, an xfs_db-alike program for the XFS IO path
which knows about direct IO, space preallocation, realtime files, etc
which the usual tools like dd don't unfortunately. This is a fixed-up
version of a hacked tool that I found useful for coding, debugging and
testing unwritten extents; hopefully it'll be useful for realtime too.
xfsprogs/bmap/xfs_bmap.c 1.13 Renamed to xfsprogs/io/bmap.c
|
/*
* Copyright (c) 2000-2003 Silicon Graphics, Inc. All Rights Reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* Further, this software is distributed without any warranty that it is
* free of the rightful claim of any third person regarding infringement
* or the like. Any license provided herein, whether implied or
* otherwise, applies only to this software file. Patent licenses, if
* any, provided herein do not apply to combinations of this program with
* other software, or any other product whatsoever.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write the Free Software Foundation, Inc., 59
* Temple Place - Suite 330, Boston MA 02111-1307, USA.
*
* Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
* Mountain View, CA 94043, or:
*
* http://www.sgi.com
*
* For further information regarding this notice, see:
*
* http://oss.sgi.com/projects/GenInfo/SGIGPLNoticeExplan/
*/
#include <libxfs.h>
#include "command.h"
#include "init.h"
static cmdinfo_t bmap_cmd;
static int
usage(void)
{
printf("%s %s\n", bmap_cmd.name, bmap_cmd.oneline);
return 0;
}
static void
bmap_help(void)
{
printf(_(
"\n"
" prints the block mapping for an XFS file's data or attribute forks"
"\n"
" Example:\n"
" 'bmap -vp' - tabular format verbose map, including unwritten extents\n"
"\n"
" bmap prints the map of disk blocks used by the current file.\n"
" The map lists each extent used by the file, as well as regions in the\n"
" file that do not have any corresponding blocks (holes).\n"
" By default, each line of the listing takes the following form:\n"
" extent: [startoffset..endoffset]: startblock..endblock\n"
" Holes are marked by replacing the startblock..endblock with 'hole'.\n"
" All the file offsets and disk blocks are in units of 512-byte blocks.\n"
" -a -- prints the attribute fork map instead of the data fork.\n"
" -d -- suppresses a DMAPI read event, offline portions shown as holes.\n"
" -l -- also displays the length of each extent in 512-byte blocks.\n"
" Note: the bmap for non-regular files can be obtained provided the file\n"
" was opened appropriately (in particular, must be opened read-only).\n"
"\n"));
}
static int
numlen(
off64_t val)
{
off64_t tmp;
int len;
for (len = 0, tmp = val; tmp > 0; tmp = tmp/10)
len++;
return (len == 0 ? 1 : len);
}
int
bmap_f(
int argc,
char **argv)
{
struct fsxattr fsx;
struct getbmapx *map;
struct xfs_fsop_geom_v1 fsgeo;
int map_size;
int loop = 0;
int flg = 0;
int aflag = 0;
int lflag = 0;
int nflag = 0;
int vflag = 0;
int bmv_iflags = 0; /* flags for XFS_IOC_GETBMAPX */
int i = 0;
int c;
while ((c = getopt(argc, argv, "adln:pv")) != EOF) {
switch (c) {
case 'a': /* Attribute fork. */
bmv_iflags |= BMV_IF_ATTRFORK;
aflag = 1;
break;
case 'l': /* list number of blocks with each extent */
lflag = 1;
break;
case 'n': /* number of extents specified */
nflag = atoi(optarg);
break;
case 'd':
/* do not recall possibly offline DMAPI files */
bmv_iflags |= BMV_IF_NO_DMAPI_READ;
break;
case 'p':
/* report unwritten preallocated blocks */
bmv_iflags |= BMV_IF_PREALLOC;
break;
case 'v': /* Verbose output */
vflag++;
break;
default:
return usage();
}
}
if (aflag)
bmv_iflags &= ~(BMV_IF_PREALLOC|BMV_IF_NO_DMAPI_READ);
if (vflag) {
if (ioctl(fdesc, XFS_IOC_FSGEOMETRY_V1, &fsgeo) < 0) {
fprintf(stderr,
_("%s: can't get geometry [\"%s\"]: %s\n"),
progname, fname, strerror(errno));
exitcode = 1;
return 0;
}
if ((ioctl(fdesc, XFS_IOC_FSGETXATTR, &fsx)) < 0) {
fprintf(stderr,
_("%s: cannot read attrs on \"%s\": %s\n"),
progname, fname, strerror(errno));
exitcode = 1;
return 0;
}
if (fsx.fsx_xflags == XFS_XFLAG_REALTIME) {
/*
* ag info not applicable to rt, continue
* without ag output.
*/
vflag = 0;
}
}
map_size = nflag ? nflag+1 : 32; /* initial guess - 256 */
map = malloc(map_size*sizeof(*map));
if (map == NULL) {
fprintf(stderr, _("%s: malloc of %d bytes failed.\n"),
progname, (int)(map_size * sizeof(*map)));
exitcode = 1;
return 0;
}
/* Try the ioctl(XFS_IOC_GETBMAPX) for the number of extents specified by
* nflag, or the initial guess number of extents (256).
*
* If there are more extents than we guessed, use ioctl
* (XFS_IOC_FSGETXATTR[A]) to get the extent count, realloc some more
* space based on this count, and try again.
*
* If the initial FGETBMAPX attempt returns EINVAL, this may mean
* that we tried the FGETBMAPX on a zero length file. If we get
* EINVAL, check the length with fstat() and return "no extents"
* if the length == 0.
*
* Why not do the ioctl(XFS_IOC_FSGETXATTR[A]) first? Two reasons:
* (1) The extent count may be wrong for a file with delayed
* allocation blocks. The XFS_IOC_GETBMAPX forces the real
* allocation and fixes up the extent count.
* (2) For XFS_IOC_GETBMAP[X] on a DMAPI file that has been moved
* offline by a DMAPI application (e.g., DMF) the
* XFS_IOC_FSGETXATTR only reflects the extents actually online.
* Doing XFS_IOC_GETBMAPX call first forces that data blocks online
* and then everything proceeds normally (see PV #545725).
*
* If you don't want this behavior on a DMAPI offline file,
* try the "-d" option which sets the BMV_IF_NO_DMAPI_READ
* iflag for XFS_IOC_GETBMAPX.
*/
do { /* loop a miximum of two times */
bzero(map, sizeof(*map)); /* zero header */
map->bmv_length = -1;
map->bmv_count = map_size;
map->bmv_iflags = bmv_iflags;
i = ioctl(fdesc, XFS_IOC_GETBMAPX, map);
if (i < 0) {
if ( errno == EINVAL
&& !aflag && filesize() == 0) {
break;
} else {
fprintf(stderr, _("%s: ioctl(XFS_IOC_GETBMAPX) "
"iflags=0x%x [\"%s\"]: %s\n"),
progname, map->bmv_iflags, fname,
strerror(errno));
free(map);
exitcode = 1;
return 0;
}
}
if (nflag)
break;
if (map->bmv_entries < map->bmv_count-1)
break;
/* Get number of extents from ioctl XFS_IOC_FSGETXATTR[A]
* syscall.
*/
i = ioctl(fdesc, aflag ?
XFS_IOC_FSGETXATTRA : XFS_IOC_FSGETXATTR, &fsx);
if (i < 0) {
fprintf(stderr, _("%s: ioctl(XFS_IOC_FSGETXATTR%s) "
"[\"%s\"]: %s\n"), progname, aflag ? "A" : "",
fname, strerror(errno));
free(map);
exitcode = 1;
return 0;
}
if (fsx.fsx_nextents >= map_size-1) {
map_size = 2*(fsx.fsx_nextents+1);
map = realloc(map, map_size*sizeof(*map));
if (map == NULL) {
fprintf(stderr,
_("%s: cannot realloc %d bytes\n"),
progname, (int)(map_size*sizeof(*map)));
exitcode = 1;
return 0;
}
}
} while (++loop < 2);
if (!nflag) {
if (map->bmv_entries <= 0) {
printf(_("%s: no extents\n"), fname);
free(map);
return 0;
}
}
printf("%s:\n", fname);
if (!vflag) {
for (i = 0; i < map->bmv_entries; i++) {
printf("\t%d: [%lld..%lld]: ", i,
(long long) map[i + 1].bmv_offset,
(long long)(map[i + 1].bmv_offset +
map[i + 1].bmv_length - 1LL));
if (map[i + 1].bmv_block == -1)
printf(_("hole"));
else {
printf("%lld..%lld",
(long long) map[i + 1].bmv_block,
(long long)(map[i + 1].bmv_block +
map[i + 1].bmv_length - 1LL));
}
if (lflag)
printf(_(" %lld blocks\n"),
(long long)map[i+1].bmv_length);
else
printf("\n");
}
} else {
/*
* Verbose mode displays:
* extent: [startoffset..endoffset]: startblock..endblock \
* ag# (agoffset..agendoffset) totalbbs
*/
#define MINRANGE_WIDTH 16
#define MINAG_WIDTH 2
#define MINTOT_WIDTH 5
#define max(a,b) (a > b ? a : b)
#define FLG_NULL 00000 /* Null flag */
#define FLG_BSU 01000 /* Not on begin of stripe unit */
#define FLG_ESU 00100 /* Not on end of stripe unit */
#define FLG_BSW 00010 /* Not on begin of stripe width */
#define FLG_ESW 00001 /* Not on end of stripe width */
int agno;
off64_t agoff, bbperag;
int foff_w, boff_w, aoff_w, tot_w, agno_w;
char rbuf[32], bbuf[32], abuf[32];
int sunit, swidth;
foff_w = boff_w = aoff_w = MINRANGE_WIDTH;
tot_w = MINTOT_WIDTH;
bbperag = (off64_t)fsgeo.agblocks *
(off64_t)fsgeo.blocksize / BBSIZE;
sunit = fsgeo.sunit;
swidth = fsgeo.swidth;
/*
* Go through the extents and figure out the width
* needed for all columns.
*/
for (i = 0; i < map->bmv_entries; i++) {
snprintf(rbuf, sizeof(rbuf), "[%lld..%lld]:",
(long long) map[i + 1].bmv_offset,
(long long)(map[i + 1].bmv_offset +
map[i + 1].bmv_length - 1LL));
if (map[i + 1].bmv_block == -1) {
foff_w = max(foff_w, strlen(rbuf));
tot_w = max(tot_w,
numlen(map[i+1].bmv_length));
} else {
snprintf(bbuf, sizeof(bbuf), "%lld..%lld",
(long long) map[i + 1].bmv_block,
(long long)(map[i + 1].bmv_block +
map[i + 1].bmv_length - 1LL));
agno = map[i + 1].bmv_block / bbperag;
agoff = map[i + 1].bmv_block - (agno * bbperag);
snprintf(abuf, sizeof(abuf), "(%lld..%lld)",
(long long)agoff, (long long)
(agoff + map[i + 1].bmv_length - 1LL));
foff_w = max(foff_w, strlen(rbuf));
boff_w = max(boff_w, strlen(bbuf));
aoff_w = max(aoff_w, strlen(abuf));
tot_w = max(tot_w,
numlen(map[i+1].bmv_length));
}
}
agno_w = max(MINAG_WIDTH, numlen(fsgeo.agcount));
printf("%4s: %-*s %-*s %*s %-*s %*s\n",
_("EXT"),
foff_w, _("FILE-OFFSET"),
boff_w, _("BLOCK-RANGE"),
agno_w, _("AG"),
aoff_w, _("AG-OFFSET"),
tot_w, _("TOTAL"));
for (i = 0; i < map->bmv_entries; i++) {
if (sunit) {
flg = FLG_NULL;
if (map[i + 1].bmv_block % sunit != 0) {
flg |= FLG_BSU;
}
if (((map[i + 1].bmv_block +
map[i + 1].bmv_length ) % sunit ) != 0) {
flg |= FLG_ESU;
}
if (map[i + 1].bmv_block % swidth != 0) {
flg |= FLG_BSW;
}
if (((map[i + 1].bmv_block +
map[i + 1].bmv_length ) % swidth ) != 0) {
flg |= FLG_ESW;
}
}
snprintf(rbuf, sizeof(rbuf), "[%lld..%lld]:",
(long long) map[i + 1].bmv_offset,
(long long)(map[i + 1].bmv_offset +
map[i + 1].bmv_length - 1LL));
if (map[i + 1].bmv_block == -1) {
printf("%4d: %-*s %-*s %*s %-*s %*lld\n",
i,
foff_w, rbuf,
boff_w, _("hole"),
agno_w, "",
aoff_w, "",
tot_w, (long long)map[i+1].bmv_length);
} else {
snprintf(bbuf, sizeof(bbuf), "%lld..%lld",
(long long) map[i + 1].bmv_block,
(long long)(map[i + 1].bmv_block +
map[i + 1].bmv_length - 1LL));
agno = map[i + 1].bmv_block / bbperag;
agoff = map[i + 1].bmv_block - (agno * bbperag);
snprintf(abuf, sizeof(abuf), "(%lld..%lld)",
(long long)agoff, (long long)
(agoff + map[i + 1].bmv_length - 1LL));
printf("%4d: %-*s %-*s %*d %-*s %*lld",
i,
foff_w, rbuf,
boff_w, bbuf,
agno_w, agno,
aoff_w, abuf,
tot_w, (long long)map[i+1].bmv_length);
if (flg == FLG_NULL) {
printf("\n");
} else {
printf(" %-4.4o\n", flg);
}
}
}
if ( flg ) {
printf(_(" FLG Values:\n"));
printf(_(" %5.5o Doesn't begin on stripe unit\n"),
FLG_BSU);
printf(_(" %5.5o Doesn't end on stripe unit\n"),
FLG_ESU);
printf(_(" %5.5o Doesn't begin on stripe width\n"),
FLG_BSW);
printf(_(" %5.5o Doesn't end on stripe width\n"),
FLG_ESW);
}
}
free(map);
return 0;
}
void
bmap_init(void)
{
bmap_cmd.name = _("bmap");
bmap_cmd.cfunc = bmap_f;
bmap_cmd.argmin = 0;
bmap_cmd.argmax = -1;
bmap_cmd.args = _("[-adlpv] [-n nx]");
bmap_cmd.oneline = _("print block mapping for an XFS file");
bmap_cmd.help = bmap_help;
add_command(&bmap_cmd);
}