Commit e7c17e43 authored by Mark Fasheh's avatar Mark Fasheh

ocfs2: Introduce dir free space list

The only operation which doesn't get faster with directory indexing is
insert, which still has to walk the entire unindexed directory portion to
find a free block. This patch provides an improvement in directory insert
performance by maintaining a singly linked list of directory leaf blocks
which have space for additional dirents.
Signed-off-by: default avatarMark Fasheh <mfasheh@suse.com>
Acked-by: default avatarJoel Becker <joel.becker@oracle.com>
parent 4ed8a6bb
This diff is collapsed.
...@@ -39,11 +39,19 @@ struct ocfs2_dir_lookup_result { ...@@ -39,11 +39,19 @@ struct ocfs2_dir_lookup_result {
struct buffer_head *dl_dx_root_bh; /* Root of indexed struct buffer_head *dl_dx_root_bh; /* Root of indexed
* tree */ * tree */
struct buffer_head *dl_dx_leaf_bh; /* Indexed leaf block */ struct buffer_head *dl_dx_leaf_bh; /* Indexed leaf block */
struct ocfs2_dx_entry *dl_dx_entry; /* Target dx_entry in struct ocfs2_dx_entry *dl_dx_entry; /* Target dx_entry in
* indexed leaf */ * indexed leaf */
struct ocfs2_dx_hinfo dl_hinfo; /* Name hash results */ struct ocfs2_dx_hinfo dl_hinfo; /* Name hash results */
struct buffer_head *dl_prev_leaf_bh;/* Previous entry in
* dir free space
* list. NULL if
* previous entry is
* dx root block. */
}; };
void ocfs2_free_dir_lookup_result(struct ocfs2_dir_lookup_result *res); void ocfs2_free_dir_lookup_result(struct ocfs2_dir_lookup_result *res);
int ocfs2_find_entry(const char *name, int namelen, int ocfs2_find_entry(const char *name, int namelen,
......
...@@ -385,8 +385,8 @@ static inline int ocfs2_remove_extent_credits(struct super_block *sb) ...@@ -385,8 +385,8 @@ static inline int ocfs2_remove_extent_credits(struct super_block *sb)
} }
/* data block for new dir/symlink, 2 for bitmap updates (bitmap fe + /* data block for new dir/symlink, 2 for bitmap updates (bitmap fe +
* bitmap block for the new bit) */ * bitmap block for the new bit) dx_root update for free list */
#define OCFS2_DIR_LINK_ADDITIONAL_CREDITS (1 + 2) #define OCFS2_DIR_LINK_ADDITIONAL_CREDITS (1 + 2 + 1)
static inline int ocfs2_add_dir_index_credits(struct super_block *sb) static inline int ocfs2_add_dir_index_credits(struct super_block *sb)
{ {
...@@ -420,19 +420,19 @@ static inline int ocfs2_mknod_credits(struct super_block *sb, int is_dir, ...@@ -420,19 +420,19 @@ static inline int ocfs2_mknod_credits(struct super_block *sb, int is_dir,
#define OCFS2_SIMPLE_DIR_EXTEND_CREDITS (2) #define OCFS2_SIMPLE_DIR_EXTEND_CREDITS (2)
/* file update (nlink, etc) + directory mtime/ctime + dir entry block + quota /* file update (nlink, etc) + directory mtime/ctime + dir entry block + quota
* update on dir + index leaf */ * update on dir + index leaf + dx root update for free list */
static inline int ocfs2_link_credits(struct super_block *sb) static inline int ocfs2_link_credits(struct super_block *sb)
{ {
return 2*OCFS2_INODE_UPDATE_CREDITS + 2 + return 2*OCFS2_INODE_UPDATE_CREDITS + 3 +
ocfs2_quota_trans_credits(sb); ocfs2_quota_trans_credits(sb);
} }
/* inode + dir inode (if we unlink a dir), + dir entry block + orphan /* inode + dir inode (if we unlink a dir), + dir entry block + orphan
* dir inode link + dir inode index leaf */ * dir inode link + dir inode index leaf + dir index root */
static inline int ocfs2_unlink_credits(struct super_block *sb) static inline int ocfs2_unlink_credits(struct super_block *sb)
{ {
/* The quota update from ocfs2_link_credits is unused here... */ /* The quota update from ocfs2_link_credits is unused here... */
return 2 * OCFS2_INODE_UPDATE_CREDITS + 2 + ocfs2_link_credits(sb); return 2 * OCFS2_INODE_UPDATE_CREDITS + 3 + ocfs2_link_credits(sb);
} }
/* dinode + orphan dir dinode + inode alloc dinode + orphan dir entry + /* dinode + orphan dir dinode + inode alloc dinode + orphan dir entry +
......
...@@ -416,6 +416,7 @@ static struct ocfs2_system_inode_info ocfs2_system_inodes[NUM_SYSTEM_INODES] = { ...@@ -416,6 +416,7 @@ static struct ocfs2_system_inode_info ocfs2_system_inodes[NUM_SYSTEM_INODES] = {
#define OCFS2_DIR_REC_LEN(name_len) (((name_len) + OCFS2_DIR_MEMBER_LEN + \ #define OCFS2_DIR_REC_LEN(name_len) (((name_len) + OCFS2_DIR_MEMBER_LEN + \
OCFS2_DIR_ROUND) & \ OCFS2_DIR_ROUND) & \
~OCFS2_DIR_ROUND) ~OCFS2_DIR_ROUND)
#define OCFS2_DIR_MIN_REC_LEN OCFS2_DIR_REC_LEN(1)
#define OCFS2_LINK_MAX 32000 #define OCFS2_LINK_MAX 32000
...@@ -842,7 +843,9 @@ struct ocfs2_dx_root_block { ...@@ -842,7 +843,9 @@ struct ocfs2_dx_root_block {
__le16 dr_reserved1; __le16 dr_reserved1;
__le64 dr_dir_blkno; /* Pointer to parent inode */ __le64 dr_dir_blkno; /* Pointer to parent inode */
__le64 dr_reserved2; __le64 dr_reserved2;
__le64 dr_reserved3[16]; __le64 dr_free_blk; /* Pointer to head of free
* unindexed block list. */
__le64 dr_reserved3[15];
union { union {
struct ocfs2_extent_list dr_list; /* Keep this aligned to 128 struct ocfs2_extent_list dr_list; /* Keep this aligned to 128
* bits for maximum space * bits for maximum space
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment