ext4_extents.h 9.3 KB
Newer Older
A
Alex Tomas 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * Copyright (c) 2003-2006, Cluster File Systems, Inc, info@clusterfs.com
 * Written by Alex Tomas <alex@clusterfs.com>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public Licens
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-
 */

19 20
#ifndef _EXT4_EXTENTS
#define _EXT4_EXTENTS
A
Alex Tomas 已提交
21

22
#include "ext4.h"
A
Alex Tomas 已提交
23 24

/*
25
 * With AGGRESSIVE_TEST defined, the capacity of index/leaf blocks
26 27 28
 * becomes very small, so index split, in-depth growing and
 * other hard changes happen much more often.
 * This is for debug purposes only.
A
Alex Tomas 已提交
29
 */
30
#define AGGRESSIVE_TEST_
A
Alex Tomas 已提交
31 32

/*
33 34 35
 * With EXTENTS_STATS defined, the number of blocks and extents
 * are collected in the truncate path. They'll be shown at
 * umount time.
A
Alex Tomas 已提交
36 37 38 39
 */
#define EXTENTS_STATS__

/*
40 41
 * If CHECK_BINSEARCH is defined, then the results of the binary search
 * will also be checked by linear search.
A
Alex Tomas 已提交
42 43 44 45
 */
#define CHECK_BINSEARCH__

/*
46
 * Turn on EXT_DEBUG to get lots of info about extents operations.
A
Alex Tomas 已提交
47 48 49
 */
#define EXT_DEBUG__
#ifdef EXT_DEBUG
50
#define ext_debug(fmt, ...)	printk(fmt, ##__VA_ARGS__)
A
Alex Tomas 已提交
51
#else
52
#define ext_debug(fmt, ...)	no_printk(fmt, ##__VA_ARGS__)
A
Alex Tomas 已提交
53 54 55
#endif

/*
56 57
 * If EXT_STATS is defined then stats numbers are collected.
 * These number will be displayed at umount time.
A
Alex Tomas 已提交
58 59 60 61 62
 */
#define EXT_STATS_


/*
63 64 65
 * ext4_inode has i_block array (60 bytes total).
 * The first 12 bytes store ext4_extent_header;
 * the remainder stores an array of ext4_extent.
A
Alex Tomas 已提交
66 67 68
 */

/*
69 70
 * This is the extent on-disk structure.
 * It's used at the bottom of the tree.
A
Alex Tomas 已提交
71 72 73 74 75
 */
struct ext4_extent {
	__le32	ee_block;	/* first logical block extent covers */
	__le16	ee_len;		/* number of blocks covered by extent */
	__le16	ee_start_hi;	/* high 16 bits of physical block */
76
	__le32	ee_start_lo;	/* low 32 bits of physical block */
A
Alex Tomas 已提交
77 78 79
};

/*
80 81
 * This is index on-disk structure.
 * It's used at all the levels except the bottom.
A
Alex Tomas 已提交
82 83 84
 */
struct ext4_extent_idx {
	__le32	ei_block;	/* index covers logical blocks from 'block' */
85
	__le32	ei_leaf_lo;	/* pointer to the physical block of the next *
86
				 * level. leaf or next index could be there */
A
Alex Tomas 已提交
87 88 89 90 91
	__le16	ei_leaf_hi;	/* high 16 bits of physical block */
	__u16	ei_unused;
};

/*
92
 * Each block (leaves and indexes), even inode-stored has header.
A
Alex Tomas 已提交
93 94 95 96 97
 */
struct ext4_extent_header {
	__le16	eh_magic;	/* probably will support different formats */
	__le16	eh_entries;	/* number of valid entries */
	__le16	eh_max;		/* capacity of store in entries */
98
	__le16	eh_depth;	/* has tree real underlying blocks? */
A
Alex Tomas 已提交
99 100 101 102 103 104
	__le32	eh_generation;	/* generation of the tree */
};

#define EXT4_EXT_MAGIC		cpu_to_le16(0xf30a)

/*
105 106 107
 * Array of ext4_ext_path contains path to some extent.
 * Creation/lookup routines use it for traversal/splitting/etc.
 * Truncate uses it to simulate recursive walking.
A
Alex Tomas 已提交
108 109
 */
struct ext4_ext_path {
110
	ext4_fsblk_t			p_block;
A
Alex Tomas 已提交
111 112 113 114 115 116 117 118 119 120 121
	__u16				p_depth;
	struct ext4_extent		*p_ext;
	struct ext4_extent_idx		*p_idx;
	struct ext4_extent_header	*p_hdr;
	struct buffer_head		*p_bh;
};

/*
 * structure for external API
 */

122 123 124 125 126 127
/*
 * to be called by ext4_ext_walk_space()
 * negative retcode - error
 * positive retcode - signal for ext4_ext_walk_space(), see below
 * callback must return valid extent (passed or newly created)
 */
128
typedef int (*ext_prepare_callback)(struct inode *, ext4_lblk_t,
129 130 131 132 133 134
					struct ext4_ext_cache *,
					struct ext4_extent *, void *);

#define EXT_CONTINUE   0
#define EXT_BREAK      1
#define EXT_REPEAT     2
A
Alex Tomas 已提交
135

136 137 138 139 140
/*
 * Maximum number of logical blocks in a file; ext4_extent's ee_block is
 * __le32.
 */
#define EXT_MAX_BLOCKS	0xffffffff
A
Alex Tomas 已提交
141

142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
/*
 * EXT_INIT_MAX_LEN is the maximum number of blocks we can have in an
 * initialized extent. This is 2^15 and not (2^16 - 1), since we use the
 * MSB of ee_len field in the extent datastructure to signify if this
 * particular extent is an initialized extent or an uninitialized (i.e.
 * preallocated).
 * EXT_UNINIT_MAX_LEN is the maximum number of blocks we can have in an
 * uninitialized extent.
 * If ee_len is <= 0x8000, it is an initialized extent. Otherwise, it is an
 * uninitialized one. In other words, if MSB of ee_len is set, it is an
 * uninitialized extent with only one special scenario when ee_len = 0x8000.
 * In this case we can not have an uninitialized extent of zero length and
 * thus we make it as a special case of initialized extent with 0x8000 length.
 * This way we get better extent-to-group alignment for initialized extents.
 * Hence, the maximum number of blocks we can have in an *initialized*
 * extent is 2^15 (32768) and in an *uninitialized* extent is 2^15-1 (32767).
 */
#define EXT_INIT_MAX_LEN	(1UL << 15)
#define EXT_UNINIT_MAX_LEN	(EXT_INIT_MAX_LEN - 1)
161

A
Alex Tomas 已提交
162 163 164 165 166 167 168 169

#define EXT_FIRST_EXTENT(__hdr__) \
	((struct ext4_extent *) (((char *) (__hdr__)) +		\
				 sizeof(struct ext4_extent_header)))
#define EXT_FIRST_INDEX(__hdr__) \
	((struct ext4_extent_idx *) (((char *) (__hdr__)) +	\
				     sizeof(struct ext4_extent_header)))
#define EXT_HAS_FREE_INDEX(__path__) \
D
Dave Kleikamp 已提交
170 171
	(le16_to_cpu((__path__)->p_hdr->eh_entries) \
				     < le16_to_cpu((__path__)->p_hdr->eh_max))
A
Alex Tomas 已提交
172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
#define EXT_LAST_EXTENT(__hdr__) \
	(EXT_FIRST_EXTENT((__hdr__)) + le16_to_cpu((__hdr__)->eh_entries) - 1)
#define EXT_LAST_INDEX(__hdr__) \
	(EXT_FIRST_INDEX((__hdr__)) + le16_to_cpu((__hdr__)->eh_entries) - 1)
#define EXT_MAX_EXTENT(__hdr__) \
	(EXT_FIRST_EXTENT((__hdr__)) + le16_to_cpu((__hdr__)->eh_max) - 1)
#define EXT_MAX_INDEX(__hdr__) \
	(EXT_FIRST_INDEX((__hdr__)) + le16_to_cpu((__hdr__)->eh_max) - 1)

static inline struct ext4_extent_header *ext_inode_hdr(struct inode *inode)
{
	return (struct ext4_extent_header *) EXT4_I(inode)->i_data;
}

static inline struct ext4_extent_header *ext_block_hdr(struct buffer_head *bh)
{
	return (struct ext4_extent_header *) bh->b_data;
}

static inline unsigned short ext_depth(struct inode *inode)
{
	return le16_to_cpu(ext_inode_hdr(inode)->eh_depth);
}

static inline void
ext4_ext_invalidate_cache(struct inode *inode)
{
199
	EXT4_I(inode)->i_cached_extent.ec_len = 0;
A
Alex Tomas 已提交
200 201
}

A
Amit Arora 已提交
202 203
static inline void ext4_ext_mark_uninitialized(struct ext4_extent *ext)
{
204 205 206
	/* We can not have an uninitialized extent of zero length! */
	BUG_ON((le16_to_cpu(ext->ee_len) & ~EXT_INIT_MAX_LEN) == 0);
	ext->ee_len |= cpu_to_le16(EXT_INIT_MAX_LEN);
A
Amit Arora 已提交
207 208 209 210
}

static inline int ext4_ext_is_uninitialized(struct ext4_extent *ext)
{
211 212
	/* Extent with ee_len of 0x8000 is treated as an initialized extent */
	return (le16_to_cpu(ext->ee_len) > EXT_INIT_MAX_LEN);
A
Amit Arora 已提交
213 214 215 216
}

static inline int ext4_ext_get_actual_len(struct ext4_extent *ext)
{
217 218 219
	return (le16_to_cpu(ext->ee_len) <= EXT_INIT_MAX_LEN ?
		le16_to_cpu(ext->ee_len) :
		(le16_to_cpu(ext->ee_len) - EXT_INIT_MAX_LEN));
A
Amit Arora 已提交
220 221
}

222 223 224 225 226
static inline void ext4_ext_mark_initialized(struct ext4_extent *ext)
{
	ext->ee_len = cpu_to_le16(ext4_ext_get_actual_len(ext));
}

227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
/*
 * ext4_ext_pblock:
 * combine low and high parts of physical block number into ext4_fsblk_t
 */
static inline ext4_fsblk_t ext4_ext_pblock(struct ext4_extent *ex)
{
	ext4_fsblk_t block;

	block = le32_to_cpu(ex->ee_start_lo);
	block |= ((ext4_fsblk_t) le16_to_cpu(ex->ee_start_hi) << 31) << 1;
	return block;
}

/*
 * ext4_idx_pblock:
 * combine low and high parts of a leaf physical block number into ext4_fsblk_t
 */
static inline ext4_fsblk_t ext4_idx_pblock(struct ext4_extent_idx *ix)
{
	ext4_fsblk_t block;

	block = le32_to_cpu(ix->ei_leaf_lo);
	block |= ((ext4_fsblk_t) le16_to_cpu(ix->ei_leaf_hi) << 31) << 1;
	return block;
}

/*
 * ext4_ext_store_pblock:
 * stores a large physical block number into an extent struct,
 * breaking it into parts
 */
static inline void ext4_ext_store_pblock(struct ext4_extent *ex,
					 ext4_fsblk_t pb)
{
	ex->ee_start_lo = cpu_to_le32((unsigned long) (pb & 0xffffffff));
	ex->ee_start_hi = cpu_to_le16((unsigned long) ((pb >> 31) >> 1) &
				      0xffff);
}

/*
 * ext4_idx_store_pblock:
 * stores a large physical block number into an index struct,
 * breaking it into parts
 */
static inline void ext4_idx_store_pblock(struct ext4_extent_idx *ix,
					 ext4_fsblk_t pb)
{
	ix->ei_leaf_lo = cpu_to_le32((unsigned long) (pb & 0xffffffff));
	ix->ei_leaf_hi = cpu_to_le16((unsigned long) ((pb >> 31) >> 1) &
				     0xffff);
}

279
extern int ext4_ext_calc_metadata_amount(struct inode *inode,
280
					 ext4_lblk_t lblocks);
A
Alex Tomas 已提交
281
extern int ext4_extent_tree_init(handle_t *, struct inode *);
282 283 284
extern int ext4_ext_calc_credits_for_single_extent(struct inode *inode,
						   int num,
						   struct ext4_ext_path *path);
285 286 287
extern int ext4_can_extents_be_merged(struct inode *inode,
				      struct ext4_extent *ex1,
				      struct ext4_extent *ex2);
288
extern int ext4_ext_insert_extent(handle_t *, struct inode *, struct ext4_ext_path *, struct ext4_extent *, int);
A
Aneesh Kumar K.V 已提交
289 290
extern struct ext4_ext_path *ext4_ext_find_extent(struct inode *, ext4_lblk_t,
							struct ext4_ext_path *);
291
extern void ext4_ext_drop_refs(struct ext4_ext_path *);
292
extern int ext4_ext_check_inode(struct inode *inode);
293 294
extern int ext4_find_delalloc_cluster(struct inode *inode, ext4_lblk_t lblk,
				      int search_hint_reverse);
295
#endif /* _EXT4_EXTENTS */
A
Alex Tomas 已提交
296