Hi, Yuezhang Mo, > In normal, there are 19 directory entries at most for a file or > a directory. > - A file directory entry > - A stream extension directory entry > - 1~17 file name directory entry > > So the directory entries are in 3 sectors at most, it is enough > for struct exfat_entry_set_cache to pre-allocate 3 bh. > > This commit changes the size of struct exfat_entry_set_cache as: > > Before After > 32-bit system 88 32 bytes > 64-bit system 168 48 bytes > > Signed-off-by: Yuezhang Mo <Yuezhang.Mo@xxxxxxxx> > Reviewed-by: Andy Wu <Andy.Wu@xxxxxxxx> > Reviewed-by: Aoyama Wataru <wataru.aoyama@xxxxxxxx> > --- > fs/exfat/exfat_fs.h | 24 ++++++++++++++++++++---- > 1 file changed, 20 insertions(+), 4 deletions(-) > > diff --git a/fs/exfat/exfat_fs.h b/fs/exfat/exfat_fs.h > index a8f8eee4937c..7d2493cda5d8 100644 > --- a/fs/exfat/exfat_fs.h > +++ b/fs/exfat/exfat_fs.h > @@ -9,6 +9,7 @@ > #include <linux/fs.h> > #include <linux/ratelimit.h> > #include <linux/nls.h> > +#include <linux/blkdev.h> > > #define EXFAT_ROOT_INO 1 > > @@ -41,6 +42,14 @@ enum { > #define ES_2_ENTRIES 2 > #define ES_ALL_ENTRIES 0 > > +#define ES_FILE_ENTRY 0 > +#define ES_STREAM_ENTRY 1 > +#define ES_FIRST_FILENAME_ENTRY 2 New ES_ definitions seem to be an index in an entry set. However, this is confusing with definitions for specifying the range used when obtaining an entry set, such as ES_2_ENTRIES or ES_ALL_ENTRIES. Therefore, it would be better to use ES_IDX_ instead of ES_ to distinguish names such as ES_IDX_FILE, ES_IDX_STREAM and so on. (If you can think of a better prefix, it doesn't have to be ES_IDX_) > +#define EXFAT_FILENAME_ENTRY_NUM(name_len) \ > + DIV_ROUND_UP(name_len, EXFAT_FILE_NAME_LEN) > +#define ES_LAST_FILENAME_ENTRY(name_len) \ > + (ES_FIRST_FILENAME_ENTRY + EXFAT_FILENAME_ENTRY_NUM(name_len)) As with the newly defined ES_ value above, it makes sense for the ES_LAST_FILENAME_ENTRY() MACRO to return the index of the last filename entry. So let's subtract 1 from the current MACRO. > + > #define DIR_DELETED 0xFFFF0321 > > /* type values */ > @@ -68,9 +77,6 @@ enum { > #define MAX_NAME_LENGTH 255 /* max len of file name excluding NULL */ > #define MAX_VFSNAME_BUF_SIZE ((MAX_NAME_LENGTH + 1) * MAX_CHARSET_SIZE) > > -/* Enough size to hold 256 dentry (even 512 Byte sector) */ > -#define DIR_CACHE_SIZE (256*sizeof(struct exfat_dentry)/512+1) > - > #define EXFAT_HINT_NONE -1 > #define EXFAT_MIN_SUBDIR 2 > > @@ -125,6 +131,16 @@ enum { > #define BITS_PER_BYTE_MASK 0x7 > #define IGNORED_BITS_REMAINED(clu, clu_base) ((1 << ((clu) - (clu_base))) - 1) > > +/* 19 entries = 1 file entry + 1 stream entry + 17 filename entries */ > +#define ES_MAX_ENTRY_NUM ES_LAST_FILENAME_ENTRY(MAX_NAME_LENGTH) Of course, it needs to add 1 here. > + > +/* > + * 19 entries x 32 bytes/entry = 608 bytes. > + * The 608 bytes are in 3 sectors at most (even 512 Byte sector). > + */ > +#define DIR_CACHE_SIZE \ > + (DIV_ROUND_UP(EXFAT_DEN_TO_B(ES_MAX_ENTRY_NUM), SECTOR_SIZE) + 1) > + > struct exfat_dentry_namebuf { > char *lfn; > int lfnbuf_len; /* usually MAX_UNINAME_BUF_SIZE */ > @@ -166,11 +182,11 @@ struct exfat_hint { > > struct exfat_entry_set_cache { > struct super_block *sb; > - bool modified; > unsigned int start_off; > int num_bh; > struct buffer_head *bh[DIR_CACHE_SIZE]; > unsigned int num_entries; > + bool modified; > }; > > struct exfat_dir_entry {