mirror of
https://github.com/Fishwaldo/linux-bl808.git
synced 2025-04-04 13:24:45 +00:00
GFS2: Implement iomap for block_map
This patch implements iomap for block mapping, and switches the block_map function to use it under the covers. The additional IOMAP_F_BOUNDARY iomap flag indicates when iomap has reached a "metadata boundary" and fetching the next mapping is likely to incur an additional I/O. This flag is used for setting the bh buffer boundary flag. Signed-off-by: Bob Peterson <rpeterso@redhat.com> Signed-off-by: Andreas Gruenbacher <agruenba@redhat.com>
This commit is contained in:
parent
5f8bd4440d
commit
3974320ca6
4 changed files with 307 additions and 100 deletions
335
fs/gfs2/bmap.c
335
fs/gfs2/bmap.c
|
@ -13,6 +13,7 @@
|
||||||
#include <linux/blkdev.h>
|
#include <linux/blkdev.h>
|
||||||
#include <linux/gfs2_ondisk.h>
|
#include <linux/gfs2_ondisk.h>
|
||||||
#include <linux/crc32.h>
|
#include <linux/crc32.h>
|
||||||
|
#include <linux/iomap.h>
|
||||||
|
|
||||||
#include "gfs2.h"
|
#include "gfs2.h"
|
||||||
#include "incore.h"
|
#include "incore.h"
|
||||||
|
@ -505,10 +506,8 @@ static inline unsigned int hptrs(struct gfs2_sbd *sdp, const unsigned int hgt)
|
||||||
* Returns: errno on error
|
* Returns: errno on error
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int gfs2_bmap_alloc(struct inode *inode, const sector_t lblock,
|
static int gfs2_iomap_alloc(struct inode *inode, struct iomap *iomap,
|
||||||
bool zero_new, struct metapath *mp,
|
unsigned flags, struct metapath *mp)
|
||||||
const size_t maxlen, sector_t *dblock,
|
|
||||||
unsigned *dblks)
|
|
||||||
{
|
{
|
||||||
struct gfs2_inode *ip = GFS2_I(inode);
|
struct gfs2_inode *ip = GFS2_I(inode);
|
||||||
struct gfs2_sbd *sdp = GFS2_SB(inode);
|
struct gfs2_sbd *sdp = GFS2_SB(inode);
|
||||||
|
@ -516,36 +515,37 @@ static int gfs2_bmap_alloc(struct inode *inode, const sector_t lblock,
|
||||||
struct buffer_head *dibh = mp->mp_bh[0];
|
struct buffer_head *dibh = mp->mp_bh[0];
|
||||||
u64 bn;
|
u64 bn;
|
||||||
unsigned n, i, blks, alloced = 0, iblks = 0, branch_start = 0;
|
unsigned n, i, blks, alloced = 0, iblks = 0, branch_start = 0;
|
||||||
|
unsigned dblks = 0;
|
||||||
unsigned ptrs_per_blk;
|
unsigned ptrs_per_blk;
|
||||||
const unsigned end_of_metadata = mp->mp_fheight - 1;
|
const unsigned end_of_metadata = mp->mp_fheight - 1;
|
||||||
int ret;
|
int ret;
|
||||||
int eob = 0;
|
|
||||||
enum alloc_state state;
|
enum alloc_state state;
|
||||||
__be64 *ptr;
|
__be64 *ptr;
|
||||||
__be64 zero_bn = 0;
|
__be64 zero_bn = 0;
|
||||||
|
size_t maxlen = iomap->length >> inode->i_blkbits;
|
||||||
|
|
||||||
BUG_ON(mp->mp_aheight < 1);
|
BUG_ON(mp->mp_aheight < 1);
|
||||||
BUG_ON(dibh == NULL);
|
BUG_ON(dibh == NULL);
|
||||||
|
|
||||||
*dblock = 0;
|
|
||||||
*dblks = 0;
|
|
||||||
gfs2_trans_add_meta(ip->i_gl, dibh);
|
gfs2_trans_add_meta(ip->i_gl, dibh);
|
||||||
|
|
||||||
if (mp->mp_fheight == mp->mp_aheight) {
|
if (mp->mp_fheight == mp->mp_aheight) {
|
||||||
struct buffer_head *bh;
|
struct buffer_head *bh;
|
||||||
|
int eob;
|
||||||
|
|
||||||
/* Bottom indirect block exists, find unalloced extent size */
|
/* Bottom indirect block exists, find unalloced extent size */
|
||||||
ptr = metapointer(end_of_metadata, mp);
|
ptr = metapointer(end_of_metadata, mp);
|
||||||
bh = mp->mp_bh[end_of_metadata];
|
bh = mp->mp_bh[end_of_metadata];
|
||||||
*dblks = gfs2_extent_length(bh->b_data, bh->b_size, ptr,
|
dblks = gfs2_extent_length(bh->b_data, bh->b_size, ptr,
|
||||||
maxlen, &eob);
|
maxlen, &eob);
|
||||||
BUG_ON(*dblks < 1);
|
BUG_ON(dblks < 1);
|
||||||
state = ALLOC_DATA;
|
state = ALLOC_DATA;
|
||||||
} else {
|
} else {
|
||||||
/* Need to allocate indirect blocks */
|
/* Need to allocate indirect blocks */
|
||||||
ptrs_per_blk = mp->mp_fheight > 1 ? sdp->sd_inptrs :
|
ptrs_per_blk = mp->mp_fheight > 1 ? sdp->sd_inptrs :
|
||||||
sdp->sd_diptrs;
|
sdp->sd_diptrs;
|
||||||
*dblks = min(maxlen, (size_t)(ptrs_per_blk -
|
dblks = min(maxlen, (size_t)(ptrs_per_blk -
|
||||||
mp->mp_list[end_of_metadata]));
|
mp->mp_list[end_of_metadata]));
|
||||||
if (mp->mp_fheight == ip->i_height) {
|
if (mp->mp_fheight == ip->i_height) {
|
||||||
/* Writing into existing tree, extend tree down */
|
/* Writing into existing tree, extend tree down */
|
||||||
iblks = mp->mp_fheight - mp->mp_aheight;
|
iblks = mp->mp_fheight - mp->mp_aheight;
|
||||||
|
@ -561,7 +561,7 @@ static int gfs2_bmap_alloc(struct inode *inode, const sector_t lblock,
|
||||||
|
|
||||||
/* start of the second part of the function (state machine) */
|
/* start of the second part of the function (state machine) */
|
||||||
|
|
||||||
blks = *dblks + iblks;
|
blks = dblks + iblks;
|
||||||
i = mp->mp_aheight;
|
i = mp->mp_aheight;
|
||||||
do {
|
do {
|
||||||
int error;
|
int error;
|
||||||
|
@ -618,32 +618,219 @@ static int gfs2_bmap_alloc(struct inode *inode, const sector_t lblock,
|
||||||
break;
|
break;
|
||||||
/* Tree complete, adding data blocks */
|
/* Tree complete, adding data blocks */
|
||||||
case ALLOC_DATA:
|
case ALLOC_DATA:
|
||||||
BUG_ON(n > *dblks);
|
BUG_ON(n > dblks);
|
||||||
BUG_ON(mp->mp_bh[end_of_metadata] == NULL);
|
BUG_ON(mp->mp_bh[end_of_metadata] == NULL);
|
||||||
gfs2_trans_add_meta(ip->i_gl, mp->mp_bh[end_of_metadata]);
|
gfs2_trans_add_meta(ip->i_gl, mp->mp_bh[end_of_metadata]);
|
||||||
*dblks = n;
|
dblks = n;
|
||||||
ptr = metapointer(end_of_metadata, mp);
|
ptr = metapointer(end_of_metadata, mp);
|
||||||
*dblock = bn;
|
iomap->addr = bn << inode->i_blkbits;
|
||||||
|
iomap->flags |= IOMAP_F_NEW;
|
||||||
while (n-- > 0)
|
while (n-- > 0)
|
||||||
*ptr++ = cpu_to_be64(bn++);
|
*ptr++ = cpu_to_be64(bn++);
|
||||||
if (zero_new) {
|
if (flags & IOMAP_ZERO) {
|
||||||
ret = sb_issue_zeroout(sb, *dblock, *dblks,
|
ret = sb_issue_zeroout(sb, iomap->addr >> inode->i_blkbits,
|
||||||
GFP_NOFS);
|
dblks, GFP_NOFS);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
fs_err(sdp,
|
fs_err(sdp,
|
||||||
"Failed to zero data buffers\n");
|
"Failed to zero data buffers\n");
|
||||||
|
flags &= ~IOMAP_ZERO;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} while ((state != ALLOC_DATA) || !(*dblock));
|
} while (iomap->addr == IOMAP_NULL_ADDR);
|
||||||
|
|
||||||
|
iomap->length = (u64)dblks << inode->i_blkbits;
|
||||||
ip->i_height = mp->mp_fheight;
|
ip->i_height = mp->mp_fheight;
|
||||||
gfs2_add_inode_blocks(&ip->i_inode, alloced);
|
gfs2_add_inode_blocks(&ip->i_inode, alloced);
|
||||||
gfs2_dinode_out(ip, mp->mp_bh[0]->b_data);
|
gfs2_dinode_out(ip, mp->mp_bh[0]->b_data);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* hole_size - figure out the size of a hole
|
||||||
|
* @inode: The inode
|
||||||
|
* @lblock: The logical starting block number
|
||||||
|
* @mp: The metapath
|
||||||
|
*
|
||||||
|
* Returns: The hole size in bytes
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static u64 hole_size(struct inode *inode, sector_t lblock, struct metapath *mp)
|
||||||
|
{
|
||||||
|
struct gfs2_inode *ip = GFS2_I(inode);
|
||||||
|
struct gfs2_sbd *sdp = GFS2_SB(inode);
|
||||||
|
struct metapath mp_eof;
|
||||||
|
u64 factor = 1;
|
||||||
|
int hgt;
|
||||||
|
u64 holesz = 0;
|
||||||
|
const __be64 *first, *end, *ptr;
|
||||||
|
const struct buffer_head *bh;
|
||||||
|
u64 lblock_stop = (i_size_read(inode) - 1) >> inode->i_blkbits;
|
||||||
|
int zeroptrs;
|
||||||
|
bool done = false;
|
||||||
|
|
||||||
|
/* Get another metapath, to the very last byte */
|
||||||
|
find_metapath(sdp, lblock_stop, &mp_eof, ip->i_height);
|
||||||
|
for (hgt = ip->i_height - 1; hgt >= 0 && !done; hgt--) {
|
||||||
|
bh = mp->mp_bh[hgt];
|
||||||
|
if (bh) {
|
||||||
|
zeroptrs = 0;
|
||||||
|
first = metapointer(hgt, mp);
|
||||||
|
end = (const __be64 *)(bh->b_data + bh->b_size);
|
||||||
|
|
||||||
|
for (ptr = first; ptr < end; ptr++) {
|
||||||
|
if (*ptr) {
|
||||||
|
done = true;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
zeroptrs++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
zeroptrs = sdp->sd_inptrs;
|
||||||
|
}
|
||||||
|
if (factor * zeroptrs >= lblock_stop - lblock + 1) {
|
||||||
|
holesz = lblock_stop - lblock + 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
holesz += factor * zeroptrs;
|
||||||
|
|
||||||
|
factor *= sdp->sd_inptrs;
|
||||||
|
if (hgt && (mp->mp_list[hgt - 1] < mp_eof.mp_list[hgt - 1]))
|
||||||
|
(mp->mp_list[hgt - 1])++;
|
||||||
|
}
|
||||||
|
return holesz << inode->i_blkbits;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gfs2_stuffed_iomap(struct inode *inode, struct iomap *iomap)
|
||||||
|
{
|
||||||
|
struct gfs2_inode *ip = GFS2_I(inode);
|
||||||
|
|
||||||
|
iomap->addr = (ip->i_no_addr << inode->i_blkbits) +
|
||||||
|
sizeof(struct gfs2_dinode);
|
||||||
|
iomap->offset = 0;
|
||||||
|
iomap->length = i_size_read(inode);
|
||||||
|
iomap->type = IOMAP_MAPPED;
|
||||||
|
iomap->flags = IOMAP_F_DATA_INLINE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gfs2_iomap_begin - Map blocks from an inode to disk blocks
|
||||||
|
* @inode: The inode
|
||||||
|
* @pos: Starting position in bytes
|
||||||
|
* @length: Length to map, in bytes
|
||||||
|
* @flags: iomap flags
|
||||||
|
* @iomap: The iomap structure
|
||||||
|
*
|
||||||
|
* Returns: errno
|
||||||
|
*/
|
||||||
|
int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length,
|
||||||
|
unsigned flags, struct iomap *iomap)
|
||||||
|
{
|
||||||
|
struct gfs2_inode *ip = GFS2_I(inode);
|
||||||
|
struct gfs2_sbd *sdp = GFS2_SB(inode);
|
||||||
|
struct metapath mp = { .mp_aheight = 1, };
|
||||||
|
unsigned int factor = sdp->sd_sb.sb_bsize;
|
||||||
|
const u64 *arr = sdp->sd_heightsize;
|
||||||
|
__be64 *ptr;
|
||||||
|
sector_t lblock;
|
||||||
|
sector_t lend;
|
||||||
|
int ret;
|
||||||
|
int eob;
|
||||||
|
unsigned int len;
|
||||||
|
struct buffer_head *bh;
|
||||||
|
u8 height;
|
||||||
|
|
||||||
|
trace_gfs2_iomap_start(ip, pos, length, flags);
|
||||||
|
if (!length) {
|
||||||
|
ret = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((flags & IOMAP_REPORT) && gfs2_is_stuffed(ip)) {
|
||||||
|
gfs2_stuffed_iomap(inode, iomap);
|
||||||
|
if (pos >= iomap->length)
|
||||||
|
return -ENOENT;
|
||||||
|
ret = 0;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
lblock = pos >> inode->i_blkbits;
|
||||||
|
lend = (pos + length + sdp->sd_sb.sb_bsize - 1) >> inode->i_blkbits;
|
||||||
|
|
||||||
|
iomap->offset = lblock << inode->i_blkbits;
|
||||||
|
iomap->addr = IOMAP_NULL_ADDR;
|
||||||
|
iomap->type = IOMAP_HOLE;
|
||||||
|
iomap->length = (u64)(lend - lblock) << inode->i_blkbits;
|
||||||
|
iomap->flags = IOMAP_F_MERGED;
|
||||||
|
bmap_lock(ip, 0);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Directory data blocks have a struct gfs2_meta_header header, so the
|
||||||
|
* remaining size is smaller than the filesystem block size. Logical
|
||||||
|
* block numbers for directories are in units of this remaining size!
|
||||||
|
*/
|
||||||
|
if (gfs2_is_dir(ip)) {
|
||||||
|
factor = sdp->sd_jbsize;
|
||||||
|
arr = sdp->sd_jheightsize;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = gfs2_meta_inode_buffer(ip, &mp.mp_bh[0]);
|
||||||
|
if (ret)
|
||||||
|
goto out_release;
|
||||||
|
|
||||||
|
height = ip->i_height;
|
||||||
|
while ((lblock + 1) * factor > arr[height])
|
||||||
|
height++;
|
||||||
|
find_metapath(sdp, lblock, &mp, height);
|
||||||
|
if (height > ip->i_height || gfs2_is_stuffed(ip))
|
||||||
|
goto do_alloc;
|
||||||
|
|
||||||
|
ret = lookup_metapath(ip, &mp);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out_release;
|
||||||
|
|
||||||
|
if (mp.mp_aheight != ip->i_height)
|
||||||
|
goto do_alloc;
|
||||||
|
|
||||||
|
ptr = metapointer(ip->i_height - 1, &mp);
|
||||||
|
if (*ptr == 0)
|
||||||
|
goto do_alloc;
|
||||||
|
|
||||||
|
iomap->type = IOMAP_MAPPED;
|
||||||
|
iomap->addr = be64_to_cpu(*ptr) << inode->i_blkbits;
|
||||||
|
|
||||||
|
bh = mp.mp_bh[ip->i_height - 1];
|
||||||
|
len = gfs2_extent_length(bh->b_data, bh->b_size, ptr, lend - lblock, &eob);
|
||||||
|
if (eob)
|
||||||
|
iomap->flags |= IOMAP_F_BOUNDARY;
|
||||||
|
iomap->length = (u64)len << inode->i_blkbits;
|
||||||
|
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
out_release:
|
||||||
|
release_metapath(&mp);
|
||||||
|
bmap_unlock(ip, 0);
|
||||||
|
out:
|
||||||
|
trace_gfs2_iomap_end(ip, iomap, ret);
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
do_alloc:
|
||||||
|
if (!(flags & IOMAP_WRITE)) {
|
||||||
|
if (pos >= i_size_read(inode)) {
|
||||||
|
ret = -ENOENT;
|
||||||
|
goto out_release;
|
||||||
|
}
|
||||||
|
ret = 0;
|
||||||
|
iomap->length = hole_size(inode, lblock, &mp);
|
||||||
|
goto out_release;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = gfs2_iomap_alloc(inode, iomap, flags, &mp);
|
||||||
|
goto out_release;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* gfs2_block_map - Map a block from an inode to a disk block
|
* gfs2_block_map - Map a block from an inode to a disk block
|
||||||
* @inode: The inode
|
* @inode: The inode
|
||||||
|
@ -662,93 +849,43 @@ int gfs2_block_map(struct inode *inode, sector_t lblock,
|
||||||
struct buffer_head *bh_map, int create)
|
struct buffer_head *bh_map, int create)
|
||||||
{
|
{
|
||||||
struct gfs2_inode *ip = GFS2_I(inode);
|
struct gfs2_inode *ip = GFS2_I(inode);
|
||||||
struct gfs2_sbd *sdp = GFS2_SB(inode);
|
struct iomap iomap;
|
||||||
unsigned int factor = sdp->sd_sb.sb_bsize;
|
int ret, flags = 0;
|
||||||
const size_t maxlen = bh_map->b_size >> inode->i_blkbits;
|
|
||||||
const u64 *arr = sdp->sd_heightsize;
|
|
||||||
__be64 *ptr;
|
|
||||||
u64 size;
|
|
||||||
struct metapath mp;
|
|
||||||
int ret;
|
|
||||||
int eob;
|
|
||||||
unsigned int len;
|
|
||||||
struct buffer_head *bh;
|
|
||||||
u8 height;
|
|
||||||
bool zero_new = false;
|
|
||||||
sector_t dblock = 0;
|
|
||||||
unsigned dblks;
|
|
||||||
|
|
||||||
BUG_ON(maxlen == 0);
|
|
||||||
|
|
||||||
memset(&mp, 0, sizeof(mp));
|
|
||||||
bmap_lock(ip, create);
|
|
||||||
clear_buffer_mapped(bh_map);
|
clear_buffer_mapped(bh_map);
|
||||||
clear_buffer_new(bh_map);
|
clear_buffer_new(bh_map);
|
||||||
clear_buffer_boundary(bh_map);
|
clear_buffer_boundary(bh_map);
|
||||||
trace_gfs2_bmap(ip, bh_map, lblock, create, 1);
|
trace_gfs2_bmap(ip, bh_map, lblock, create, 1);
|
||||||
|
|
||||||
/*
|
if (create)
|
||||||
* Directory data blocks have a struct gfs2_meta_header header, so the
|
flags |= IOMAP_WRITE;
|
||||||
* remaining size is smaller than the filesystem block size. Logical
|
|
||||||
* block numbers for directories are in units of this remaining size!
|
|
||||||
*/
|
|
||||||
if (gfs2_is_dir(ip)) {
|
|
||||||
factor = sdp->sd_jbsize;
|
|
||||||
arr = sdp->sd_jheightsize;
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = gfs2_meta_inode_buffer(ip, &mp.mp_bh[0]);
|
|
||||||
if (ret)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
height = ip->i_height;
|
|
||||||
size = (lblock + 1) * factor;
|
|
||||||
while (size > arr[height])
|
|
||||||
height++;
|
|
||||||
find_metapath(sdp, lblock, &mp, height);
|
|
||||||
mp.mp_aheight = 1;
|
|
||||||
if (height > ip->i_height || gfs2_is_stuffed(ip))
|
|
||||||
goto do_alloc;
|
|
||||||
ret = lookup_metapath(ip, &mp);
|
|
||||||
if (ret < 0)
|
|
||||||
goto out;
|
|
||||||
if (mp.mp_aheight != ip->i_height)
|
|
||||||
goto do_alloc;
|
|
||||||
ptr = metapointer(ip->i_height - 1, &mp);
|
|
||||||
if (*ptr == 0)
|
|
||||||
goto do_alloc;
|
|
||||||
map_bh(bh_map, inode->i_sb, be64_to_cpu(*ptr));
|
|
||||||
bh = mp.mp_bh[ip->i_height - 1];
|
|
||||||
len = gfs2_extent_length(bh->b_data, bh->b_size, ptr, maxlen, &eob);
|
|
||||||
bh_map->b_size = (len << inode->i_blkbits);
|
|
||||||
if (eob)
|
|
||||||
set_buffer_boundary(bh_map);
|
|
||||||
ret = 0;
|
|
||||||
out:
|
|
||||||
release_metapath(&mp);
|
|
||||||
trace_gfs2_bmap(ip, bh_map, lblock, create, ret);
|
|
||||||
bmap_unlock(ip, create);
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
do_alloc:
|
|
||||||
/* All allocations are done here, firstly check create flag */
|
|
||||||
if (!create) {
|
|
||||||
BUG_ON(gfs2_is_stuffed(ip));
|
|
||||||
ret = 0;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* At this point ret is the tree depth of already allocated blocks */
|
|
||||||
if (buffer_zeronew(bh_map))
|
if (buffer_zeronew(bh_map))
|
||||||
zero_new = true;
|
flags |= IOMAP_ZERO;
|
||||||
ret = gfs2_bmap_alloc(inode, lblock, zero_new, &mp, maxlen, &dblock,
|
ret = gfs2_iomap_begin(inode, (loff_t)lblock << inode->i_blkbits,
|
||||||
&dblks);
|
bh_map->b_size, flags, &iomap);
|
||||||
if (ret == 0) {
|
if (ret) {
|
||||||
map_bh(bh_map, inode->i_sb, dblock);
|
if (!create && ret == -ENOENT) {
|
||||||
bh_map->b_size = dblks << inode->i_blkbits;
|
/* Return unmapped buffer beyond the end of file. */
|
||||||
set_buffer_new(bh_map);
|
ret = 0;
|
||||||
|
}
|
||||||
|
goto out;
|
||||||
}
|
}
|
||||||
goto out;
|
|
||||||
|
if (iomap.length > bh_map->b_size) {
|
||||||
|
iomap.length = bh_map->b_size;
|
||||||
|
iomap.flags &= ~IOMAP_F_BOUNDARY;
|
||||||
|
}
|
||||||
|
if (iomap.addr != IOMAP_NULL_ADDR)
|
||||||
|
map_bh(bh_map, inode->i_sb, iomap.addr >> inode->i_blkbits);
|
||||||
|
bh_map->b_size = iomap.length;
|
||||||
|
if (iomap.flags & IOMAP_F_BOUNDARY)
|
||||||
|
set_buffer_boundary(bh_map);
|
||||||
|
if (iomap.flags & IOMAP_F_NEW)
|
||||||
|
set_buffer_new(bh_map);
|
||||||
|
|
||||||
|
out:
|
||||||
|
trace_gfs2_bmap(ip, bh_map, lblock, create, ret);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -10,6 +10,8 @@
|
||||||
#ifndef __BMAP_DOT_H__
|
#ifndef __BMAP_DOT_H__
|
||||||
#define __BMAP_DOT_H__
|
#define __BMAP_DOT_H__
|
||||||
|
|
||||||
|
#include <linux/iomap.h>
|
||||||
|
|
||||||
#include "inode.h"
|
#include "inode.h"
|
||||||
|
|
||||||
struct inode;
|
struct inode;
|
||||||
|
@ -47,6 +49,8 @@ static inline void gfs2_write_calc_reserv(const struct gfs2_inode *ip,
|
||||||
extern int gfs2_unstuff_dinode(struct gfs2_inode *ip, struct page *page);
|
extern int gfs2_unstuff_dinode(struct gfs2_inode *ip, struct page *page);
|
||||||
extern int gfs2_block_map(struct inode *inode, sector_t lblock,
|
extern int gfs2_block_map(struct inode *inode, sector_t lblock,
|
||||||
struct buffer_head *bh, int create);
|
struct buffer_head *bh, int create);
|
||||||
|
extern int gfs2_iomap_begin(struct inode *inode, loff_t pos, loff_t length,
|
||||||
|
unsigned flags, struct iomap *iomap);
|
||||||
extern int gfs2_extent_map(struct inode *inode, u64 lblock, int *new,
|
extern int gfs2_extent_map(struct inode *inode, u64 lblock, int *new,
|
||||||
u64 *dblock, unsigned *extlen);
|
u64 *dblock, unsigned *extlen);
|
||||||
extern int gfs2_setattr_size(struct inode *inode, u64 size);
|
extern int gfs2_setattr_size(struct inode *inode, u64 size);
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
#include <linux/gfs2_ondisk.h>
|
#include <linux/gfs2_ondisk.h>
|
||||||
#include <linux/writeback.h>
|
#include <linux/writeback.h>
|
||||||
#include <linux/ktime.h>
|
#include <linux/ktime.h>
|
||||||
|
#include <linux/iomap.h>
|
||||||
#include "incore.h"
|
#include "incore.h"
|
||||||
#include "glock.h"
|
#include "glock.h"
|
||||||
#include "rgrp.h"
|
#include "rgrp.h"
|
||||||
|
@ -469,6 +470,70 @@ TRACE_EVENT(gfs2_bmap,
|
||||||
__entry->errno)
|
__entry->errno)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
TRACE_EVENT(gfs2_iomap_start,
|
||||||
|
|
||||||
|
TP_PROTO(const struct gfs2_inode *ip, loff_t pos, ssize_t length,
|
||||||
|
u16 flags),
|
||||||
|
|
||||||
|
TP_ARGS(ip, pos, length, flags),
|
||||||
|
|
||||||
|
TP_STRUCT__entry(
|
||||||
|
__field( dev_t, dev )
|
||||||
|
__field( u64, inum )
|
||||||
|
__field( loff_t, pos )
|
||||||
|
__field( ssize_t, length )
|
||||||
|
__field( u16, flags )
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_fast_assign(
|
||||||
|
__entry->dev = ip->i_gl->gl_name.ln_sbd->sd_vfs->s_dev;
|
||||||
|
__entry->inum = ip->i_no_addr;
|
||||||
|
__entry->pos = pos;
|
||||||
|
__entry->length = length;
|
||||||
|
__entry->flags = flags;
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_printk("%u,%u bmap %llu iomap start %llu/%lu flags:%08x",
|
||||||
|
MAJOR(__entry->dev), MINOR(__entry->dev),
|
||||||
|
(unsigned long long)__entry->inum,
|
||||||
|
(unsigned long long)__entry->pos,
|
||||||
|
(unsigned long)__entry->length, (u16)__entry->flags)
|
||||||
|
);
|
||||||
|
|
||||||
|
TRACE_EVENT(gfs2_iomap_end,
|
||||||
|
|
||||||
|
TP_PROTO(const struct gfs2_inode *ip, struct iomap *iomap, int ret),
|
||||||
|
|
||||||
|
TP_ARGS(ip, iomap, ret),
|
||||||
|
|
||||||
|
TP_STRUCT__entry(
|
||||||
|
__field( dev_t, dev )
|
||||||
|
__field( u64, inum )
|
||||||
|
__field( loff_t, offset )
|
||||||
|
__field( ssize_t, length )
|
||||||
|
__field( u16, flags )
|
||||||
|
__field( u16, type )
|
||||||
|
__field( int, ret )
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_fast_assign(
|
||||||
|
__entry->dev = ip->i_gl->gl_name.ln_sbd->sd_vfs->s_dev;
|
||||||
|
__entry->inum = ip->i_no_addr;
|
||||||
|
__entry->offset = iomap->offset;
|
||||||
|
__entry->length = iomap->length;
|
||||||
|
__entry->flags = iomap->flags;
|
||||||
|
__entry->type = iomap->type;
|
||||||
|
__entry->ret = ret;
|
||||||
|
),
|
||||||
|
|
||||||
|
TP_printk("%u,%u bmap %llu iomap end %llu/%lu ty:%d flags:%08x rc:%d",
|
||||||
|
MAJOR(__entry->dev), MINOR(__entry->dev),
|
||||||
|
(unsigned long long)__entry->inum,
|
||||||
|
(unsigned long long)__entry->offset,
|
||||||
|
(unsigned long)__entry->length, (u16)__entry->type,
|
||||||
|
(u16)__entry->flags, __entry->ret)
|
||||||
|
);
|
||||||
|
|
||||||
/* Keep track of blocks as they are allocated/freed */
|
/* Keep track of blocks as they are allocated/freed */
|
||||||
TRACE_EVENT(gfs2_block_alloc,
|
TRACE_EVENT(gfs2_block_alloc,
|
||||||
|
|
||||||
|
|
|
@ -21,7 +21,8 @@ struct vm_fault;
|
||||||
/*
|
/*
|
||||||
* Flags for all iomap mappings:
|
* Flags for all iomap mappings:
|
||||||
*/
|
*/
|
||||||
#define IOMAP_F_NEW 0x01 /* blocks have been newly allocated */
|
#define IOMAP_F_NEW 0x01 /* blocks have been newly allocated */
|
||||||
|
#define IOMAP_F_BOUNDARY 0x02 /* mapping ends at metadata boundary */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Flags that only need to be reported for IOMAP_REPORT requests:
|
* Flags that only need to be reported for IOMAP_REPORT requests:
|
||||||
|
|
Loading…
Add table
Reference in a new issue