Index: linux-2.6.7/fs/ext3/namei.c =================================================================== --- linux-2.6.7.orig/fs/ext3/namei.c 2004-06-15 23:19:36.000000000 -0600 +++ linux-2.6.7/fs/ext3/namei.c 2004-08-20 17:48:54.000000000 -0600 @@ -1596,11 +1596,17 @@ static int ext3_delete_entry (handle_t * static inline void ext3_inc_count(handle_t *handle, struct inode *inode) { inode->i_nlink++; + if (is_dx(inode) && inode->i_nlink > 1) { + /* limit is 16-bit i_links_count */ + if (inode->i_nlink >= EXT3_LINK_MAX || inode->i_nlink == 2) + inode->i_nlink = 1; + } } static inline void ext3_dec_count(handle_t *handle, struct inode *inode) { - inode->i_nlink--; + if (!S_ISDIR(inode->i_mode) || inode->i_nlink > 2) + inode->i_nlink--; } static int ext3_add_nondir(handle_t *handle, @@ -1693,7 +1698,7 @@ static int ext3_mkdir(struct inode * dir struct ext3_dir_entry_2 * de; int err; - if (dir->i_nlink >= EXT3_LINK_MAX) + if (EXT3_DIR_LINK_MAXED(dir)) return -EMLINK; handle = ext3_journal_start(dir, EXT3_DATA_TRANS_BLOCKS + @@ -1715,7 +1720,7 @@ static int ext3_mkdir(struct inode * dir inode->i_size = EXT3_I(inode)->i_disksize = inode->i_sb->s_blocksize; dir_block = ext3_bread (handle, inode, 0, 1, &err); if (!dir_block) { - inode->i_nlink--; /* is this nlink == 0? */ + ext3_dec_count(handle, inode); /* is this nlink == 0? */ ext3_mark_inode_dirty(handle, inode); iput (inode); goto out_stop; @@ -1747,7 +1752,7 @@ static int ext3_mkdir(struct inode * dir iput (inode); goto out_stop; } - dir->i_nlink++; + ext3_inc_count(handle, dir); ext3_update_dx_flag(dir); ext3_mark_inode_dirty(handle, dir); d_instantiate(dentry, inode); @@ -2010,10 +2015,10 @@ static int ext3_rmdir (struct inode * di retval = ext3_delete_entry(handle, dir, de, bh); if (retval) goto end_rmdir; - if (inode->i_nlink != 2) - ext3_warning (inode->i_sb, "ext3_rmdir", - "empty directory has nlink!=2 (%d)", - inode->i_nlink); + if (!EXT3_DIR_LINK_EMPTY(inode)) + ext3_warning(inode->i_sb, "ext3_rmdir", + "empty directory has too many links (%d)", + inode->i_nlink); inode->i_version++; inode->i_nlink = 0; /* There's no need to set i_disksize: the fact that i_nlink is @@ -2023,7 +2028,7 @@ static int ext3_rmdir (struct inode * di ext3_orphan_add(handle, inode); inode->i_ctime = dir->i_ctime = dir->i_mtime = CURRENT_TIME; ext3_mark_inode_dirty(handle, inode); - dir->i_nlink--; + ext3_dec_count(handle, dir); ext3_update_dx_flag(dir); ext3_mark_inode_dirty(handle, dir); @@ -2074,7 +2079,7 @@ static int ext3_unlink(struct inode * di dir->i_ctime = dir->i_mtime = CURRENT_TIME; ext3_update_dx_flag(dir); ext3_mark_inode_dirty(handle, dir); - inode->i_nlink--; + ext3_dec_count(handle, inode); if (!inode->i_nlink) ext3_orphan_add(handle, inode); inode->i_ctime = dir->i_ctime; @@ -2146,7 +2151,7 @@ static int ext3_link (struct dentry * ol struct inode *inode = old_dentry->d_inode; int err; - if (inode->i_nlink >= EXT3_LINK_MAX) + if (EXT3_DIR_LINK_MAXED(inode)) return -EMLINK; handle = ext3_journal_start(dir, EXT3_DATA_TRANS_BLOCKS + @@ -2230,8 +2235,8 @@ static int ext3_rename (struct inode * o if (le32_to_cpu(PARENT_INO(dir_bh->b_data)) != old_dir->i_ino) goto end_rename; retval = -EMLINK; - if (!new_inode && new_dir!=old_dir && - new_dir->i_nlink >= EXT3_LINK_MAX) + if (!new_inode && new_dir != old_dir && + EXT3_DIR_LINK_MAXED(new_dir)) goto end_rename; } if (!new_bh) { @@ -2288,7 +2293,7 @@ static int ext3_rename (struct inode * o } if (new_inode) { - new_inode->i_nlink--; + ext3_dec_count(handle, new_inode); new_inode->i_ctime = CURRENT_TIME; } old_dir->i_ctime = old_dir->i_mtime = CURRENT_TIME; @@ -2299,11 +2304,11 @@ static int ext3_rename (struct inode * o PARENT_INO(dir_bh->b_data) = le32_to_cpu(new_dir->i_ino); BUFFER_TRACE(dir_bh, "call ext3_journal_dirty_metadata"); ext3_journal_dirty_metadata(handle, dir_bh); - old_dir->i_nlink--; + ext3_dec_count(handle, old_dir); if (new_inode) { - new_inode->i_nlink--; + ext3_dec_count(handle, new_inode); } else { - new_dir->i_nlink++; + ext3_inc_count(handle, new_dir); ext3_update_dx_flag(new_dir); ext3_mark_inode_dirty(handle, new_dir); } Index: linux-2.6.7/include/linux/ext3_fs.h =================================================================== --- linux-2.6.7.orig/include/linux/ext3_fs.h 2004-06-15 23:19:36.000000000 -0600 +++ linux-2.6.7/include/linux/ext3_fs.h 2004-08-20 17:41:27.000000000 -0600 @@ -41,7 +41,7 @@ struct statfs; /* * Always enable hashed directories */ -#define CONFIG_EXT3_INDEX +#define CONFIG_EXT3_INDEX 1 /* * Debug code @@ -79,7 +81,7 @@ /* * Maximal count of links to a file */ -#define EXT3_LINK_MAX 32000 +#define EXT3_LINK_MAX 65000 /* * Macro-instructions used to manage several block sizes @@ -595,14 +595,15 @@ struct ext3_dir_entry_2 { */ #ifdef CONFIG_EXT3_INDEX - #define is_dx(dir) (EXT3_HAS_COMPAT_FEATURE(dir->i_sb, \ - EXT3_FEATURE_COMPAT_DIR_INDEX) && \ +#define is_dx(dir) (EXT3_HAS_COMPAT_FEATURE(dir->i_sb, \ + EXT3_FEATURE_COMPAT_DIR_INDEX) && \ (EXT3_I(dir)->i_flags & EXT3_INDEX_FL)) -#define EXT3_DIR_LINK_MAX(dir) (!is_dx(dir) && (dir)->i_nlink >= EXT3_LINK_MAX) -#define EXT3_DIR_LINK_EMPTY(dir) ((dir)->i_nlink == 2 || (dir)->i_nlink == 1) +#define EXT3_DIR_LINK_MAXED(dir) (!is_dx(dir) && (dir)->i_nlink >=EXT3_LINK_MAX) +#define EXT3_DIR_LINK_EMPTY(dir) ((dir)->i_nlink == 2 || \ + (is_dx(dir) && (dir)->i_nlink == 1)) #else #define is_dx(dir) 0 -#define EXT3_DIR_LINK_MAX(dir) ((dir)->i_nlink >= EXT3_LINK_MAX) +#define EXT3_DIR_LINK_MAXED(dir) ((dir)->i_nlink >= EXT3_LINK_MAX) #define EXT3_DIR_LINK_EMPTY(dir) ((dir)->i_nlink == 2) #endif