diff options
Diffstat (limited to 'fs')
-rw-r--r-- | fs/qnx4/inode.c | 23 |
1 files changed, 15 insertions, 8 deletions
diff --git a/fs/qnx4/inode.c b/fs/qnx4/inode.c index 7900a813d90f..552e994e3aa1 100644 --- a/fs/qnx4/inode.c +++ b/fs/qnx4/inode.c | |||
@@ -66,23 +66,31 @@ static int qnx4_get_block( struct inode *inode, sector_t iblock, struct buffer_h | |||
66 | return 0; | 66 | return 0; |
67 | } | 67 | } |
68 | 68 | ||
69 | static inline u32 try_extent(qnx4_xtnt_t *extent, u32 *offset) | ||
70 | { | ||
71 | u32 size = le32_to_cpu(extent->xtnt_size); | ||
72 | if (*offset < size) | ||
73 | return le32_to_cpu(extent->xtnt_blk) + *offset - 1; | ||
74 | *offset -= size; | ||
75 | return 0; | ||
76 | } | ||
77 | |||
69 | unsigned long qnx4_block_map( struct inode *inode, long iblock ) | 78 | unsigned long qnx4_block_map( struct inode *inode, long iblock ) |
70 | { | 79 | { |
71 | int ix; | 80 | int ix; |
72 | long offset, i_xblk; | 81 | long i_xblk; |
73 | unsigned long block = 0; | ||
74 | struct buffer_head *bh = NULL; | 82 | struct buffer_head *bh = NULL; |
75 | struct qnx4_xblk *xblk = NULL; | 83 | struct qnx4_xblk *xblk = NULL; |
76 | struct qnx4_inode_entry *qnx4_inode = qnx4_raw_inode(inode); | 84 | struct qnx4_inode_entry *qnx4_inode = qnx4_raw_inode(inode); |
77 | u16 nxtnt = le16_to_cpu(qnx4_inode->di_num_xtnts); | 85 | u16 nxtnt = le16_to_cpu(qnx4_inode->di_num_xtnts); |
86 | u32 offset = iblock; | ||
87 | u32 block = try_extent(&qnx4_inode->di_first_xtnt, &offset); | ||
78 | 88 | ||
79 | if ( iblock < le32_to_cpu(qnx4_inode->di_first_xtnt.xtnt_size) ) { | 89 | if (block) { |
80 | // iblock is in the first extent. This is easy. | 90 | // iblock is in the first extent. This is easy. |
81 | block = le32_to_cpu(qnx4_inode->di_first_xtnt.xtnt_blk) + iblock - 1; | ||
82 | } else { | 91 | } else { |
83 | // iblock is beyond first extent. We have to follow the extent chain. | 92 | // iblock is beyond first extent. We have to follow the extent chain. |
84 | i_xblk = le32_to_cpu(qnx4_inode->di_xblk); | 93 | i_xblk = le32_to_cpu(qnx4_inode->di_xblk); |
85 | offset = iblock - le32_to_cpu(qnx4_inode->di_first_xtnt.xtnt_size); | ||
86 | ix = 0; | 94 | ix = 0; |
87 | while ( --nxtnt > 0 ) { | 95 | while ( --nxtnt > 0 ) { |
88 | if ( ix == 0 ) { | 96 | if ( ix == 0 ) { |
@@ -98,12 +106,11 @@ unsigned long qnx4_block_map( struct inode *inode, long iblock ) | |||
98 | return -EIO; | 106 | return -EIO; |
99 | } | 107 | } |
100 | } | 108 | } |
101 | if ( offset < le32_to_cpu(xblk->xblk_xtnts[ix].xtnt_size) ) { | 109 | block = try_extent(&xblk->xblk_xtnts[ix], &offset); |
110 | if (block) { | ||
102 | // got it! | 111 | // got it! |
103 | block = le32_to_cpu(xblk->xblk_xtnts[ix].xtnt_blk) + offset - 1; | ||
104 | break; | 112 | break; |
105 | } | 113 | } |
106 | offset -= le32_to_cpu(xblk->xblk_xtnts[ix].xtnt_size); | ||
107 | if ( ++ix >= xblk->xblk_num_xtnts ) { | 114 | if ( ++ix >= xblk->xblk_num_xtnts ) { |
108 | i_xblk = le32_to_cpu(xblk->xblk_next_xblk); | 115 | i_xblk = le32_to_cpu(xblk->xblk_next_xblk); |
109 | ix = 0; | 116 | ix = 0; |