2005-04-16 15:20:36 -07:00
/*
* linux / fs / fat / dir . c
*
* directory handling functions for fat - based filesystems
*
* Written 1992 , 1993 by Werner Almesberger
*
* Hidden files 1995 by Albert Cahalan < albert @ ccs . neu . edu > < adc @ coe . neu . edu >
*
* VFAT extensions by Gordon Chaffee < chaffee @ plateau . cs . berkeley . edu >
* Merged with msdos fs by Henrik Storner < storner @ osiris . ping . dk >
* Rewritten for constant inumbers . Plugged buffer overrun in readdir ( ) . AV
* Short name translation 1999 , 2001 by Wolfram Pienkoss < wp @ bszh . de >
*/
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/time.h>
# include <linux/buffer_head.h>
2006-08-31 12:50:04 +02:00
# include <linux/compat.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
2010-03-16 05:48:08 +09:00
# include <linux/kernel.h>
2008-11-06 12:53:46 -08:00
# include "fat.h"
2005-04-16 15:20:36 -07:00
2009-04-30 10:08:18 -04:00
/*
* Maximum buffer size of short name .
* [ ( MSDOS_NAME + ' . ' ) * max one char + nul ]
* For msdos style , [ ' . ' ( hidden ) + MSDOS_NAME + ' . ' + nul ]
*/
# define FAT_MAX_SHORT_SIZE ((MSDOS_NAME + 1) * NLS_MAX_CHARSET_SIZE + 1)
/*
* Maximum buffer size of unicode chars from slots .
* [ ( max longname slots * 13 ( size in a slot ) + nul ) * sizeof ( wchar_t ) ]
*/
# define FAT_MAX_UNI_CHARS ((MSDOS_SLOTS - 1) * 13 + 1)
# define FAT_MAX_UNI_SIZE (FAT_MAX_UNI_CHARS * sizeof(wchar_t))
2005-04-16 15:20:36 -07:00
static inline loff_t fat_make_i_pos ( struct super_block * sb ,
struct buffer_head * bh ,
struct msdos_dir_entry * de )
{
return ( ( loff_t ) bh - > b_blocknr < < MSDOS_SB ( sb ) - > dir_per_block_bits )
| ( de - ( struct msdos_dir_entry * ) bh - > b_data ) ;
}
2005-09-06 15:17:12 -07:00
static inline void fat_dir_readahead ( struct inode * dir , sector_t iblock ,
sector_t phys )
{
struct super_block * sb = dir - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bh ;
int sec ;
/* This is not a first sector of cluster, or sec_per_clus == 1 */
if ( ( iblock & ( sbi - > sec_per_clus - 1 ) ) | | sbi - > sec_per_clus = = 1 )
return ;
/* root dir of FAT12/FAT16 */
if ( ( sbi - > fat_bits ! = 32 ) & & ( dir - > i_ino = = MSDOS_ROOT_INO ) )
return ;
2006-01-08 01:02:09 -08:00
bh = sb_find_get_block ( sb , phys ) ;
if ( bh = = NULL | | ! buffer_uptodate ( bh ) ) {
2005-09-06 15:17:12 -07:00
for ( sec = 0 ; sec < sbi - > sec_per_clus ; sec + + )
sb_breadahead ( sb , phys + sec ) ;
}
brelse ( bh ) ;
}
2005-04-16 15:20:36 -07:00
/* Returns the inode number of the directory entry at offset pos. If bh is
non - NULL , it is brelse ' d before . Pos is incremented . The buffer header is
returned in bh .
AV . Most often we do it item - by - item . Makes sense to optimize .
AV . OK , there we go : if both bh and de are non - NULL we assume that we just
AV . want the next entry ( took one explicit de = NULL in vfat / namei . c ) .
AV . It ' s done in fat_get_entry ( ) ( inlined ) , here the slow case lives .
AV . Additionally , when we return - 1 ( i . e . reached the end of directory )
AV . we make bh NULL .
*/
static int fat__get_entry ( struct inode * dir , loff_t * pos ,
struct buffer_head * * bh , struct msdos_dir_entry * * de )
{
struct super_block * sb = dir - > i_sb ;
sector_t phys , iblock ;
2006-01-08 01:02:11 -08:00
unsigned long mapped_blocks ;
int err , offset ;
2005-04-16 15:20:36 -07:00
next :
if ( * bh )
brelse ( * bh ) ;
* bh = NULL ;
iblock = * pos > > sb - > s_blocksize_bits ;
2008-11-06 12:53:57 -08:00
err = fat_bmap ( dir , iblock , & phys , & mapped_blocks , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( err | | ! phys )
return - 1 ; /* beyond EOF or error */
2005-09-06 15:17:12 -07:00
fat_dir_readahead ( dir , iblock , phys ) ;
2005-04-16 15:20:36 -07:00
* bh = sb_bread ( sb , phys ) ;
if ( * bh = = NULL ) {
2011-04-12 21:08:38 +09:00
fat_msg ( sb , KERN_ERR , " Directory bread(block %llu) failed " ,
2008-11-06 12:53:58 -08:00
( llu ) phys ) ;
2005-04-16 15:20:36 -07:00
/* skip this block */
* pos = ( iblock + 1 ) < < sb - > s_blocksize_bits ;
goto next ;
}
offset = * pos & ( sb - > s_blocksize - 1 ) ;
* pos + = sizeof ( struct msdos_dir_entry ) ;
* de = ( struct msdos_dir_entry * ) ( ( * bh ) - > b_data + offset ) ;
return 0 ;
}
static inline int fat_get_entry ( struct inode * dir , loff_t * pos ,
struct buffer_head * * bh ,
struct msdos_dir_entry * * de )
{
/* Fast stuff first */
if ( * bh & & * de & &
( * de - ( struct msdos_dir_entry * ) ( * bh ) - > b_data ) < MSDOS_SB ( dir - > i_sb ) - > dir_per_block - 1 ) {
* pos + = sizeof ( struct msdos_dir_entry ) ;
( * de ) + + ;
return 0 ;
}
return fat__get_entry ( dir , pos , bh , de ) ;
}
/*
2006-03-22 00:13:35 +01:00
* Convert Unicode 16 to UTF - 8 , translated Unicode , or ASCII .
2005-04-16 15:20:36 -07:00
* If uni_xlate is enabled and we can ' t get a 1 : 1 conversion , use a
* colon as an escape character since it is normally invalid on the vfat
* filesystem . The following four characters are the hexadecimal digits
* of Unicode value . This lets us do a full dump and restore of Unicode
* filenames . We could get into some trouble with long Unicode names ,
* but ignore that right now .
* Ahem . . . Stack smashing in ring 0 isn ' t fun . Fixed .
*/
2011-04-12 21:08:38 +09:00
static int uni16_to_x8 ( struct super_block * sb , unsigned char * ascii ,
const wchar_t * uni , int len , struct nls_table * nls )
2005-04-16 15:20:36 -07:00
{
2011-04-12 21:08:38 +09:00
int uni_xlate = MSDOS_SB ( sb ) - > options . unicode_xlate ;
2008-07-25 01:46:43 -07:00
const wchar_t * ip ;
wchar_t ec ;
2010-03-16 05:48:08 +09:00
unsigned char * op ;
2005-04-16 15:20:36 -07:00
int charlen ;
ip = uni ;
op = ascii ;
2008-04-28 02:16:29 -07:00
while ( * ip & & ( ( len - NLS_MAX_CHARSET_SIZE ) > 0 ) ) {
2005-04-16 15:20:36 -07:00
ec = * ip + + ;
2010-03-16 05:48:08 +09:00
if ( ( charlen = nls - > uni2char ( ec , op , NLS_MAX_CHARSET_SIZE ) ) > 0 ) {
2005-04-16 15:20:36 -07:00
op + = charlen ;
2008-04-28 02:16:29 -07:00
len - = charlen ;
2005-04-16 15:20:36 -07:00
} else {
if ( uni_xlate = = 1 ) {
2010-03-16 05:48:08 +09:00
* op + + = ' : ' ;
op = pack_hex_byte ( op , ec > > 8 ) ;
op = pack_hex_byte ( op , ec ) ;
2008-04-28 02:16:29 -07:00
len - = 5 ;
2005-04-16 15:20:36 -07:00
} else {
* op + + = ' ? ' ;
2008-04-28 02:16:29 -07:00
len - - ;
2005-04-16 15:20:36 -07:00
}
}
}
2008-04-28 02:16:29 -07:00
if ( unlikely ( * ip ) ) {
2011-04-12 21:08:38 +09:00
fat_msg ( sb , KERN_WARNING , " filename was truncated while "
" converting. " ) ;
2008-04-28 02:16:29 -07:00
}
2005-04-16 15:20:36 -07:00
* op = 0 ;
return ( op - ascii ) ;
}
2011-04-12 21:08:38 +09:00
static inline int fat_uni_to_x8 ( struct super_block * sb , const wchar_t * uni ,
2008-07-25 01:46:43 -07:00
unsigned char * buf , int size )
{
2011-04-12 21:08:38 +09:00
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
2008-07-25 01:46:43 -07:00
if ( sbi - > options . utf8 )
2009-04-30 10:08:18 -04:00
return utf16s_to_utf8s ( uni , FAT_MAX_UNI_CHARS ,
UTF16_HOST_ENDIAN , buf , size ) ;
2008-07-25 01:46:43 -07:00
else
2011-04-12 21:08:38 +09:00
return uni16_to_x8 ( sb , buf , uni , size , sbi - > nls_io ) ;
2008-07-25 01:46:43 -07:00
}
2005-04-16 15:20:36 -07:00
static inline int
fat_short2uni ( struct nls_table * t , unsigned char * c , int clen , wchar_t * uni )
{
int charlen ;
charlen = t - > char2uni ( c , clen , uni ) ;
if ( charlen < 0 ) {
* uni = 0x003f ; /* a question mark */
charlen = 1 ;
}
return charlen ;
}
static inline int
fat_short2lower_uni ( struct nls_table * t , unsigned char * c , int clen , wchar_t * uni )
{
int charlen ;
wchar_t wc ;
charlen = t - > char2uni ( c , clen , & wc ) ;
if ( charlen < 0 ) {
* uni = 0x003f ; /* a question mark */
charlen = 1 ;
} else if ( charlen < = 1 ) {
unsigned char nc = t - > charset2lower [ * c ] ;
if ( ! nc )
nc = * c ;
if ( ( charlen = t - > char2uni ( & nc , 1 , uni ) ) < 0 ) {
* uni = 0x003f ; /* a question mark */
charlen = 1 ;
}
} else
* uni = wc ;
return charlen ;
}
static inline int
fat_shortname2uni ( struct nls_table * nls , unsigned char * buf , int buf_size ,
wchar_t * uni_buf , unsigned short opt , int lower )
{
int len = 0 ;
if ( opt & VFAT_SFN_DISPLAY_LOWER )
len = fat_short2lower_uni ( nls , buf , buf_size , uni_buf ) ;
else if ( opt & VFAT_SFN_DISPLAY_WIN95 )
len = fat_short2uni ( nls , buf , buf_size , uni_buf ) ;
else if ( opt & VFAT_SFN_DISPLAY_WINNT ) {
if ( lower )
len = fat_short2lower_uni ( nls , buf , buf_size , uni_buf ) ;
else
len = fat_short2uni ( nls , buf , buf_size , uni_buf ) ;
} else
len = fat_short2uni ( nls , buf , buf_size , uni_buf ) ;
return len ;
}
2008-07-25 01:46:43 -07:00
static inline int fat_name_match ( struct msdos_sb_info * sbi ,
const unsigned char * a , int a_len ,
const unsigned char * b , int b_len )
{
if ( a_len ! = b_len )
return 0 ;
if ( sbi - > options . name_check ! = ' s ' )
return ! nls_strnicmp ( sbi - > nls_io , a , b , a_len ) ;
else
return ! memcmp ( a , b , a_len ) ;
}
2005-10-30 15:03:50 -08:00
enum { PARSE_INVALID = 1 , PARSE_NOT_LONGNAME , PARSE_EOF , } ;
/**
* fat_parse_long - Parse extended directory entry .
*
* This function returns zero on success , negative value on error , or one of
* the following :
*
* % PARSE_INVALID - Directory entry is invalid .
* % PARSE_NOT_LONGNAME - Directory entry does not contain longname .
* % PARSE_EOF - Directory has no more entries .
*/
static int fat_parse_long ( struct inode * dir , loff_t * pos ,
struct buffer_head * * bh , struct msdos_dir_entry * * de ,
wchar_t * * unicode , unsigned char * nr_slots )
{
struct msdos_dir_slot * ds ;
unsigned char id , slot , slots , alias_checksum ;
if ( ! * unicode ) {
2008-04-28 02:16:29 -07:00
* unicode = __getname ( ) ;
2005-10-30 15:03:50 -08:00
if ( ! * unicode ) {
brelse ( * bh ) ;
return - ENOMEM ;
}
}
parse_long :
slots = 0 ;
ds = ( struct msdos_dir_slot * ) * de ;
id = ds - > id ;
if ( ! ( id & 0x40 ) )
return PARSE_INVALID ;
slots = id & ~ 0x40 ;
if ( slots > 20 | | ! slots ) /* ceil(256 * 2 / 26) */
return PARSE_INVALID ;
* nr_slots = slots ;
alias_checksum = ds - > alias_checksum ;
slot = slots ;
while ( 1 ) {
int offset ;
slot - - ;
offset = slot * 13 ;
fat16_towchar ( * unicode + offset , ds - > name0_4 , 5 ) ;
fat16_towchar ( * unicode + offset + 5 , ds - > name5_10 , 6 ) ;
fat16_towchar ( * unicode + offset + 11 , ds - > name11_12 , 2 ) ;
if ( ds - > id & 0x40 )
( * unicode ) [ offset + 13 ] = 0 ;
if ( fat_get_entry ( dir , pos , bh , de ) < 0 )
return PARSE_EOF ;
if ( slot = = 0 )
break ;
ds = ( struct msdos_dir_slot * ) * de ;
if ( ds - > attr ! = ATTR_EXT )
return PARSE_NOT_LONGNAME ;
if ( ( ds - > id & ~ 0x40 ) ! = slot )
goto parse_long ;
if ( ds - > alias_checksum ! = alias_checksum )
goto parse_long ;
}
if ( ( * de ) - > name [ 0 ] = = DELETED_FLAG )
return PARSE_INVALID ;
if ( ( * de ) - > attr = = ATTR_EXT )
goto parse_long ;
if ( IS_FREE ( ( * de ) - > name ) | | ( ( * de ) - > attr & ATTR_VOLUME ) )
return PARSE_INVALID ;
if ( fat_checksum ( ( * de ) - > name ) ! = alias_checksum )
* nr_slots = 0 ;
return 0 ;
}
2005-04-16 15:20:36 -07:00
/*
* Return values : negative - > error , 0 - > not found , positive - > found ,
* value is the total amount of slots , including the shortname entry .
*/
int fat_search_long ( struct inode * inode , const unsigned char * name ,
int name_len , struct fat_slot_info * sinfo )
{
struct super_block * sb = inode - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bh = NULL ;
struct msdos_dir_entry * de ;
struct nls_table * nls_disk = sbi - > nls_disk ;
2008-04-28 02:16:29 -07:00
unsigned char nr_slots ;
2008-07-25 01:46:43 -07:00
wchar_t bufuname [ 14 ] ;
2005-04-16 15:20:36 -07:00
wchar_t * unicode = NULL ;
2008-04-28 02:16:29 -07:00
unsigned char work [ MSDOS_NAME ] ;
2008-07-25 01:46:44 -07:00
unsigned char bufname [ FAT_MAX_SHORT_SIZE ] ;
2005-04-16 15:20:36 -07:00
unsigned short opt_shortname = sbi - > options . shortname ;
loff_t cpos = 0 ;
2008-07-25 01:46:43 -07:00
int chl , i , j , last_u , err , len ;
2005-04-16 15:20:36 -07:00
err = - ENOENT ;
2008-07-25 01:46:43 -07:00
while ( 1 ) {
2005-04-16 15:20:36 -07:00
if ( fat_get_entry ( inode , & cpos , & bh , & de ) = = - 1 )
2008-07-25 01:46:43 -07:00
goto end_of_dir ;
2005-04-16 15:20:36 -07:00
parse_record :
nr_slots = 0 ;
if ( de - > name [ 0 ] = = DELETED_FLAG )
continue ;
if ( de - > attr ! = ATTR_EXT & & ( de - > attr & ATTR_VOLUME ) )
continue ;
if ( de - > attr ! = ATTR_EXT & & IS_FREE ( de - > name ) )
continue ;
if ( de - > attr = = ATTR_EXT ) {
2005-10-30 15:03:50 -08:00
int status = fat_parse_long ( inode , & cpos , & bh , & de ,
& unicode , & nr_slots ) ;
2008-11-06 12:53:48 -08:00
if ( status < 0 ) {
err = status ;
goto end_of_dir ;
} else if ( status = = PARSE_INVALID )
2005-04-16 15:20:36 -07:00
continue ;
2005-10-30 15:03:50 -08:00
else if ( status = = PARSE_NOT_LONGNAME )
goto parse_record ;
else if ( status = = PARSE_EOF )
2008-07-25 01:46:43 -07:00
goto end_of_dir ;
2005-04-16 15:20:36 -07:00
}
memcpy ( work , de - > name , sizeof ( de - > name ) ) ;
/* see namei.c, msdos_format_name */
if ( work [ 0 ] = = 0x05 )
work [ 0 ] = 0xE5 ;
for ( i = 0 , j = 0 , last_u = 0 ; i < 8 ; ) {
2007-07-15 23:39:56 -07:00
if ( ! work [ i ] )
break ;
2005-04-16 15:20:36 -07:00
chl = fat_shortname2uni ( nls_disk , & work [ i ] , 8 - i ,
& bufuname [ j + + ] , opt_shortname ,
de - > lcase & CASE_LOWER_BASE ) ;
if ( chl < = 1 ) {
if ( work [ i ] ! = ' ' )
last_u = j ;
} else {
last_u = j ;
}
i + = chl ;
}
j = last_u ;
fat_short2uni ( nls_disk , " . " , 1 , & bufuname [ j + + ] ) ;
2007-07-15 23:39:56 -07:00
for ( i = 8 ; i < MSDOS_NAME ; ) {
if ( ! work [ i ] )
break ;
chl = fat_shortname2uni ( nls_disk , & work [ i ] ,
MSDOS_NAME - i ,
2005-04-16 15:20:36 -07:00
& bufuname [ j + + ] , opt_shortname ,
de - > lcase & CASE_LOWER_EXT ) ;
if ( chl < = 1 ) {
2007-07-15 23:39:56 -07:00
if ( work [ i ] ! = ' ' )
2005-04-16 15:20:36 -07:00
last_u = j ;
} else {
last_u = j ;
}
i + = chl ;
}
if ( ! last_u )
continue ;
2008-07-25 01:46:43 -07:00
/* Compare shortname */
2005-04-16 15:20:36 -07:00
bufuname [ last_u ] = 0x0000 ;
2011-04-12 21:08:38 +09:00
len = fat_uni_to_x8 ( sb , bufuname , bufname , sizeof ( bufname ) ) ;
2008-07-25 01:46:43 -07:00
if ( fat_name_match ( sbi , name , name_len , bufname , len ) )
goto found ;
2005-04-16 15:20:36 -07:00
if ( nr_slots ) {
2008-07-25 01:46:44 -07:00
void * longname = unicode + FAT_MAX_UNI_CHARS ;
int size = PATH_MAX - FAT_MAX_UNI_SIZE ;
2008-07-25 01:46:43 -07:00
/* Compare longname */
2011-04-12 21:08:38 +09:00
len = fat_uni_to_x8 ( sb , unicode , longname , size ) ;
2008-07-25 01:46:44 -07:00
if ( fat_name_match ( sbi , name , name_len , longname , len ) )
2008-07-25 01:46:43 -07:00
goto found ;
2005-04-16 15:20:36 -07:00
}
}
2008-07-25 01:46:43 -07:00
found :
2005-04-16 15:20:36 -07:00
nr_slots + + ; /* include the de */
sinfo - > slot_off = cpos - nr_slots * sizeof ( * de ) ;
sinfo - > nr_slots = nr_slots ;
sinfo - > de = de ;
sinfo - > bh = bh ;
sinfo - > i_pos = fat_make_i_pos ( sb , sinfo - > bh , sinfo - > de ) ;
err = 0 ;
2008-07-25 01:46:43 -07:00
end_of_dir :
2005-04-16 15:20:36 -07:00
if ( unicode )
2008-04-28 02:16:29 -07:00
__putname ( unicode ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_search_long ) ;
2005-04-16 15:20:36 -07:00
struct fat_ioctl_filldir_callback {
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
void __user * dirent ;
2005-04-16 15:20:36 -07:00
int result ;
/* for dir ioctl */
const char * longname ;
int long_len ;
const char * shortname ;
int short_len ;
} ;
2005-10-30 15:03:50 -08:00
static int __fat_readdir ( struct inode * inode , struct file * filp , void * dirent ,
filldir_t filldir , int short_only , int both )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = inode - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bh ;
struct msdos_dir_entry * de ;
struct nls_table * nls_disk = sbi - > nls_disk ;
2008-07-25 01:46:43 -07:00
unsigned char nr_slots ;
2005-04-16 15:20:36 -07:00
wchar_t bufuname [ 14 ] ;
wchar_t * unicode = NULL ;
2008-07-25 01:46:44 -07:00
unsigned char c , work [ MSDOS_NAME ] ;
unsigned char bufname [ FAT_MAX_SHORT_SIZE ] , * ptname = bufname ;
2008-07-25 01:46:43 -07:00
unsigned short opt_shortname = sbi - > options . shortname ;
2005-04-16 15:20:36 -07:00
int isvfat = sbi - > options . isvfat ;
int nocase = sbi - > options . nocase ;
2008-07-25 01:46:44 -07:00
const char * fill_name = NULL ;
2005-04-16 15:20:36 -07:00
unsigned long inum ;
2008-07-25 01:46:43 -07:00
unsigned long lpos , dummy , * furrfu = & lpos ;
2005-04-16 15:20:36 -07:00
loff_t cpos ;
2008-07-25 01:46:44 -07:00
int chi , chl , i , i2 , j , last , last_u , dotoffset = 0 , fill_len = 0 ;
2005-04-16 15:20:36 -07:00
int ret = 0 ;
2008-05-19 19:53:01 -07:00
lock_super ( sb ) ;
2005-04-16 15:20:36 -07:00
cpos = filp - > f_pos ;
/* Fake . and .. for the root directory. */
if ( inode - > i_ino = = MSDOS_ROOT_INO ) {
while ( cpos < 2 ) {
if ( filldir ( dirent , " .. " , cpos + 1 , cpos , MSDOS_ROOT_INO , DT_DIR ) < 0 )
goto out ;
cpos + + ;
filp - > f_pos + + ;
}
if ( cpos = = 2 ) {
dummy = 2 ;
furrfu = & dummy ;
cpos = 0 ;
}
}
2008-07-25 01:46:43 -07:00
if ( cpos & ( sizeof ( struct msdos_dir_entry ) - 1 ) ) {
2005-04-16 15:20:36 -07:00
ret = - ENOENT ;
goto out ;
}
bh = NULL ;
2008-07-25 01:46:43 -07:00
get_new :
2005-04-16 15:20:36 -07:00
if ( fat_get_entry ( inode , & cpos , & bh , & de ) = = - 1 )
2008-07-25 01:46:43 -07:00
goto end_of_dir ;
2005-10-30 15:03:50 -08:00
parse_record :
2008-07-25 01:46:43 -07:00
nr_slots = 0 ;
2008-07-25 01:46:44 -07:00
/*
* Check for long filename entry , but if short_only , we don ' t
* need to parse long filename .
*/
if ( isvfat & & ! short_only ) {
2005-04-16 15:20:36 -07:00
if ( de - > name [ 0 ] = = DELETED_FLAG )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-04-16 15:20:36 -07:00
if ( de - > attr ! = ATTR_EXT & & ( de - > attr & ATTR_VOLUME ) )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-04-16 15:20:36 -07:00
if ( de - > attr ! = ATTR_EXT & & IS_FREE ( de - > name ) )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-04-16 15:20:36 -07:00
} else {
if ( ( de - > attr & ATTR_VOLUME ) | | IS_FREE ( de - > name ) )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-04-16 15:20:36 -07:00
}
if ( isvfat & & de - > attr = = ATTR_EXT ) {
2005-10-30 15:03:50 -08:00
int status = fat_parse_long ( inode , & cpos , & bh , & de ,
2008-07-25 01:46:43 -07:00
& unicode , & nr_slots ) ;
2005-10-30 15:03:50 -08:00
if ( status < 0 ) {
filp - > f_pos = cpos ;
ret = status ;
goto out ;
} else if ( status = = PARSE_INVALID )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-10-30 15:03:50 -08:00
else if ( status = = PARSE_NOT_LONGNAME )
goto parse_record ;
else if ( status = = PARSE_EOF )
2008-07-25 01:46:43 -07:00
goto end_of_dir ;
2008-07-25 01:46:44 -07:00
if ( nr_slots ) {
void * longname = unicode + FAT_MAX_UNI_CHARS ;
int size = PATH_MAX - FAT_MAX_UNI_SIZE ;
2011-04-12 21:08:38 +09:00
int len = fat_uni_to_x8 ( sb , unicode , longname , size ) ;
2008-07-25 01:46:44 -07:00
fill_name = longname ;
fill_len = len ;
/* !both && !short_only, so we don't need shortname. */
if ( ! both )
goto start_filldir ;
}
2005-04-16 15:20:36 -07:00
}
if ( sbi - > options . dotsOK ) {
ptname = bufname ;
dotoffset = 0 ;
if ( de - > attr & ATTR_HIDDEN ) {
* ptname + + = ' . ' ;
dotoffset = 1 ;
}
}
memcpy ( work , de - > name , sizeof ( de - > name ) ) ;
/* see namei.c, msdos_format_name */
if ( work [ 0 ] = = 0x05 )
work [ 0 ] = 0xE5 ;
for ( i = 0 , j = 0 , last = 0 , last_u = 0 ; i < 8 ; ) {
2007-07-15 23:39:56 -07:00
if ( ! ( c = work [ i ] ) )
break ;
2005-04-16 15:20:36 -07:00
chl = fat_shortname2uni ( nls_disk , & work [ i ] , 8 - i ,
& bufuname [ j + + ] , opt_shortname ,
de - > lcase & CASE_LOWER_BASE ) ;
if ( chl < = 1 ) {
ptname [ i + + ] = ( ! nocase & & c > = ' A ' & & c < = ' Z ' ) ? c + 32 : c ;
if ( c ! = ' ' ) {
last = i ;
last_u = j ;
}
} else {
last_u = j ;
for ( chi = 0 ; chi < chl & & i < 8 ; chi + + ) {
ptname [ i ] = work [ i ] ;
i + + ; last = i ;
}
}
}
i = last ;
j = last_u ;
fat_short2uni ( nls_disk , " . " , 1 , & bufuname [ j + + ] ) ;
ptname [ i + + ] = ' . ' ;
2007-07-15 23:39:56 -07:00
for ( i2 = 8 ; i2 < MSDOS_NAME ; ) {
if ( ! ( c = work [ i2 ] ) )
break ;
chl = fat_shortname2uni ( nls_disk , & work [ i2 ] , MSDOS_NAME - i2 ,
2005-04-16 15:20:36 -07:00
& bufuname [ j + + ] , opt_shortname ,
de - > lcase & CASE_LOWER_EXT ) ;
if ( chl < = 1 ) {
i2 + + ;
ptname [ i + + ] = ( ! nocase & & c > = ' A ' & & c < = ' Z ' ) ? c + 32 : c ;
if ( c ! = ' ' ) {
last = i ;
last_u = j ;
}
} else {
last_u = j ;
2007-07-15 23:39:56 -07:00
for ( chi = 0 ; chi < chl & & i2 < MSDOS_NAME ; chi + + ) {
ptname [ i + + ] = work [ i2 + + ] ;
2005-04-16 15:20:36 -07:00
last = i ;
}
}
}
if ( ! last )
2008-07-25 01:46:43 -07:00
goto record_end ;
2005-04-16 15:20:36 -07:00
i = last + dotoffset ;
j = last_u ;
2008-07-25 01:46:44 -07:00
if ( isvfat ) {
bufuname [ j ] = 0x0000 ;
2011-04-12 21:08:38 +09:00
i = fat_uni_to_x8 ( sb , bufuname , bufname , sizeof ( bufname ) ) ;
2008-07-25 01:46:44 -07:00
}
if ( nr_slots ) {
/* hack for fat_ioctl_filldir() */
struct fat_ioctl_filldir_callback * p = dirent ;
p - > longname = fill_name ;
p - > long_len = fill_len ;
p - > shortname = bufname ;
p - > short_len = i ;
fill_name = NULL ;
fill_len = 0 ;
} else {
fill_name = bufname ;
fill_len = i ;
}
start_filldir :
2008-07-25 01:46:43 -07:00
lpos = cpos - ( nr_slots + 1 ) * sizeof ( struct msdos_dir_entry ) ;
2005-04-16 15:20:36 -07:00
if ( ! memcmp ( de - > name , MSDOS_DOT , MSDOS_NAME ) )
inum = inode - > i_ino ;
else if ( ! memcmp ( de - > name , MSDOS_DOTDOT , MSDOS_NAME ) ) {
2006-12-08 02:36:39 -08:00
inum = parent_ino ( filp - > f_path . dentry ) ;
2005-04-16 15:20:36 -07:00
} else {
loff_t i_pos = fat_make_i_pos ( sb , bh , de ) ;
struct inode * tmp = fat_iget ( sb , i_pos ) ;
if ( tmp ) {
inum = tmp - > i_ino ;
iput ( tmp ) ;
} else
inum = iunique ( sb , MSDOS_ROOT_INO ) ;
}
if ( filldir ( dirent , fill_name , fill_len , * furrfu , inum ,
( de - > attr & ATTR_DIR ) ? DT_DIR : DT_REG ) < 0 )
2008-07-25 01:46:43 -07:00
goto fill_failed ;
2005-04-16 15:20:36 -07:00
2008-07-25 01:46:43 -07:00
record_end :
2005-04-16 15:20:36 -07:00
furrfu = & lpos ;
filp - > f_pos = cpos ;
2008-07-25 01:46:43 -07:00
goto get_new ;
end_of_dir :
2005-04-16 15:20:36 -07:00
filp - > f_pos = cpos ;
2008-07-25 01:46:43 -07:00
fill_failed :
2005-09-06 15:17:12 -07:00
brelse ( bh ) ;
2005-04-16 15:20:36 -07:00
if ( unicode )
2008-04-28 02:16:29 -07:00
__putname ( unicode ) ;
2005-04-16 15:20:36 -07:00
out :
2008-05-19 19:53:01 -07:00
unlock_super ( sb ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
static int fat_readdir ( struct file * filp , void * dirent , filldir_t filldir )
{
2006-12-08 02:36:39 -08:00
struct inode * inode = filp - > f_path . dentry - > d_inode ;
2005-10-30 15:03:50 -08:00
return __fat_readdir ( inode , filp , dirent , filldir , 0 , 0 ) ;
2005-04-16 15:20:36 -07:00
}
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
# define FAT_IOCTL_FILLDIR_FUNC(func, dirent_type) \
static int func ( void * __buf , const char * name , int name_len , \
loff_t offset , u64 ino , unsigned int d_type ) \
{ \
struct fat_ioctl_filldir_callback * buf = __buf ; \
struct dirent_type __user * d1 = buf - > dirent ; \
struct dirent_type __user * d2 = d1 + 1 ; \
\
if ( buf - > result ) \
return - EINVAL ; \
buf - > result + + ; \
\
if ( name ! = NULL ) { \
/* dirent has only short name */ \
if ( name_len > = sizeof ( d1 - > d_name ) ) \
name_len = sizeof ( d1 - > d_name ) - 1 ; \
\
if ( put_user ( 0 , d2 - > d_name ) | | \
put_user ( 0 , & d2 - > d_reclen ) | | \
copy_to_user ( d1 - > d_name , name , name_len ) | | \
put_user ( 0 , d1 - > d_name + name_len ) | | \
put_user ( name_len , & d1 - > d_reclen ) ) \
goto efault ; \
} else { \
/* dirent has short and long name */ \
const char * longname = buf - > longname ; \
int long_len = buf - > long_len ; \
const char * shortname = buf - > shortname ; \
int short_len = buf - > short_len ; \
\
if ( long_len > = sizeof ( d1 - > d_name ) ) \
long_len = sizeof ( d1 - > d_name ) - 1 ; \
if ( short_len > = sizeof ( d1 - > d_name ) ) \
short_len = sizeof ( d1 - > d_name ) - 1 ; \
\
if ( copy_to_user ( d2 - > d_name , longname , long_len ) | | \
put_user ( 0 , d2 - > d_name + long_len ) | | \
put_user ( long_len , & d2 - > d_reclen ) | | \
put_user ( ino , & d2 - > d_ino ) | | \
put_user ( offset , & d2 - > d_off ) | | \
copy_to_user ( d1 - > d_name , shortname , short_len ) | | \
put_user ( 0 , d1 - > d_name + short_len ) | | \
put_user ( short_len , & d1 - > d_reclen ) ) \
goto efault ; \
} \
return 0 ; \
efault : \
buf - > result = - EFAULT ; \
return - EFAULT ; \
}
2008-07-25 01:46:43 -07:00
FAT_IOCTL_FILLDIR_FUNC ( fat_ioctl_filldir , __fat_dirent )
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
static int fat_ioctl_readdir ( struct inode * inode , struct file * filp ,
void __user * dirent , filldir_t filldir ,
int short_only , int both )
2005-04-16 15:20:36 -07:00
{
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
struct fat_ioctl_filldir_callback buf ;
int ret ;
buf . dirent = dirent ;
buf . result = 0 ;
mutex_lock ( & inode - > i_mutex ) ;
ret = - ENOENT ;
if ( ! IS_DEADDIR ( inode ) ) {
ret = __fat_readdir ( inode , filp , & buf , filldir ,
short_only , both ) ;
2005-04-16 15:20:36 -07:00
}
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
mutex_unlock ( & inode - > i_mutex ) ;
if ( ret > = 0 )
ret = buf . result ;
return ret ;
2005-04-16 15:20:36 -07:00
}
2010-05-17 08:13:47 +09:00
static long fat_dir_ioctl ( struct file * filp , unsigned int cmd ,
unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2010-05-17 08:13:47 +09:00
struct inode * inode = filp - > f_path . dentry - > d_inode ;
2008-07-25 01:46:43 -07:00
struct __fat_dirent __user * d1 = ( struct __fat_dirent __user * ) arg ;
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
int short_only , both ;
2005-04-16 15:20:36 -07:00
switch ( cmd ) {
case VFAT_IOCTL_READDIR_SHORT :
short_only = 1 ;
both = 0 ;
break ;
case VFAT_IOCTL_READDIR_BOTH :
short_only = 0 ;
both = 1 ;
break ;
default :
2010-05-17 08:13:47 +09:00
return fat_generic_ioctl ( filp , cmd , arg ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-25 01:46:43 -07:00
if ( ! access_ok ( VERIFY_WRITE , d1 , sizeof ( struct __fat_dirent [ 2 ] ) ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
/*
* Yes , we don ' t need this put_user ( ) absolutely . However old
* code didn ' t return the right value . So , app use this value ,
* in order to check whether it is EOF .
*/
if ( put_user ( 0 , & d1 - > d_reclen ) )
return - EFAULT ;
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
return fat_ioctl_readdir ( inode , filp , d1 , fat_ioctl_filldir ,
short_only , both ) ;
2005-04-16 15:20:36 -07:00
}
2006-08-31 12:50:04 +02:00
# ifdef CONFIG_COMPAT
# define VFAT_IOCTL_READDIR_BOTH32 _IOR('r', 1, struct compat_dirent[2])
# define VFAT_IOCTL_READDIR_SHORT32 _IOR('r', 2, struct compat_dirent[2])
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
FAT_IOCTL_FILLDIR_FUNC ( fat_compat_ioctl_filldir , compat_dirent )
2006-08-31 12:50:04 +02:00
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
static long fat_compat_dir_ioctl ( struct file * filp , unsigned cmd ,
2006-08-31 12:50:04 +02:00
unsigned long arg )
{
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
struct inode * inode = filp - > f_path . dentry - > d_inode ;
struct compat_dirent __user * d1 = compat_ptr ( arg ) ;
int short_only , both ;
2006-08-31 12:50:04 +02:00
switch ( cmd ) {
case VFAT_IOCTL_READDIR_SHORT32 :
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
short_only = 1 ;
both = 0 ;
break ;
case VFAT_IOCTL_READDIR_BOTH32 :
short_only = 0 ;
both = 1 ;
2006-08-31 12:50:04 +02:00
break ;
default :
2010-05-17 08:13:47 +09:00
return fat_generic_ioctl ( filp , cmd , ( unsigned long ) arg ) ;
2006-08-31 12:50:04 +02:00
}
fat: fix VFAT compat ioctls on 64-bit systems
If you compile and run the below test case in an msdos or vfat directory on
an x86-64 system with -m32 you'll get garbage in the kernel_dirent struct
followed by a SIGSEGV.
The patch fixes this.
Reported and initial fix by Bart Oldeman
#include <sys/types.h>
#include <sys/ioctl.h>
#include <dirent.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
struct kernel_dirent {
long d_ino;
long d_off;
unsigned short d_reclen;
char d_name[256]; /* We must not include limits.h! */
};
#define VFAT_IOCTL_READDIR_BOTH _IOR('r', 1, struct kernel_dirent [2])
#define VFAT_IOCTL_READDIR_SHORT _IOR('r', 2, struct kernel_dirent [2])
int main(void)
{
int fd = open(".", O_RDONLY);
struct kernel_dirent de[2];
while (1) {
int i = ioctl(fd, VFAT_IOCTL_READDIR_BOTH, (long)de);
if (i == -1) break;
if (de[0].d_reclen == 0) break;
printf("SFN: reclen=%2d off=%d ino=%d, %-12s",
de[0].d_reclen, de[0].d_off, de[0].d_ino, de[0].d_name);
if (de[1].d_reclen)
printf("\tLFN: reclen=%2d off=%d ino=%d, %s",
de[1].d_reclen, de[1].d_off, de[1].d_ino, de[1].d_name);
printf("\n");
}
return 0;
}
Signed-off-by: Bart Oldeman <bartoldeman@users.sourceforge.net>
Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-08 00:31:28 -07:00
if ( ! access_ok ( VERIFY_WRITE , d1 , sizeof ( struct compat_dirent [ 2 ] ) ) )
return - EFAULT ;
/*
* Yes , we don ' t need this put_user ( ) absolutely . However old
* code didn ' t return the right value . So , app use this value ,
* in order to check whether it is EOF .
*/
if ( put_user ( 0 , & d1 - > d_reclen ) )
return - EFAULT ;
return fat_ioctl_readdir ( inode , filp , d1 , fat_compat_ioctl_filldir ,
short_only , both ) ;
2006-08-31 12:50:04 +02:00
}
# endif /* CONFIG_COMPAT */
2006-03-28 01:56:42 -08:00
const struct file_operations fat_dir_operations = {
2008-11-06 12:53:47 -08:00
. llseek = generic_file_llseek ,
2005-04-16 15:20:36 -07:00
. read = generic_read_dir ,
. readdir = fat_readdir ,
2010-05-17 08:13:47 +09:00
. unlocked_ioctl = fat_dir_ioctl ,
2006-08-31 12:50:04 +02:00
# ifdef CONFIG_COMPAT
. compat_ioctl = fat_compat_dir_ioctl ,
# endif
2009-06-07 13:44:36 -04:00
. fsync = fat_file_fsync ,
2005-04-16 15:20:36 -07:00
} ;
static int fat_get_short_entry ( struct inode * dir , loff_t * pos ,
struct buffer_head * * bh ,
struct msdos_dir_entry * * de )
{
while ( fat_get_entry ( dir , pos , bh , de ) > = 0 ) {
/* free entry or long name entry or volume label */
if ( ! IS_FREE ( ( * de ) - > name ) & & ! ( ( * de ) - > attr & ATTR_VOLUME ) )
return 0 ;
}
return - ENOENT ;
}
/*
* The " .. " entry can not provide the " struct fat_slot_info " informations
* for inode . So , this function provide the some informations only .
*/
int fat_get_dotdot_entry ( struct inode * dir , struct buffer_head * * bh ,
struct msdos_dir_entry * * de , loff_t * i_pos )
{
loff_t offset ;
offset = 0 ;
* bh = NULL ;
while ( fat_get_short_entry ( dir , & offset , bh , de ) > = 0 ) {
if ( ! strncmp ( ( * de ) - > name , MSDOS_DOTDOT , MSDOS_NAME ) ) {
* i_pos = fat_make_i_pos ( dir - > i_sb , * bh , * de ) ;
return 0 ;
}
}
return - ENOENT ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_get_dotdot_entry ) ;
2005-04-16 15:20:36 -07:00
/* See if directory is empty */
int fat_dir_empty ( struct inode * dir )
{
struct buffer_head * bh ;
struct msdos_dir_entry * de ;
loff_t cpos ;
int result = 0 ;
bh = NULL ;
cpos = 0 ;
while ( fat_get_short_entry ( dir , & cpos , & bh , & de ) > = 0 ) {
if ( strncmp ( de - > name , MSDOS_DOT , MSDOS_NAME ) & &
strncmp ( de - > name , MSDOS_DOTDOT , MSDOS_NAME ) ) {
result = - ENOTEMPTY ;
break ;
}
}
brelse ( bh ) ;
return result ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_dir_empty ) ;
2005-04-16 15:20:36 -07:00
/*
* fat_subdirs counts the number of sub - directories of dir . It can be run
* on directories being created .
*/
int fat_subdirs ( struct inode * dir )
{
struct buffer_head * bh ;
struct msdos_dir_entry * de ;
loff_t cpos ;
int count = 0 ;
bh = NULL ;
cpos = 0 ;
while ( fat_get_short_entry ( dir , & cpos , & bh , & de ) > = 0 ) {
if ( de - > attr & ATTR_DIR )
count + + ;
}
brelse ( bh ) ;
return count ;
}
/*
* Scans a directory for a given file ( name points to its formatted name ) .
* Returns an error code or zero .
*/
int fat_scan ( struct inode * dir , const unsigned char * name ,
struct fat_slot_info * sinfo )
{
struct super_block * sb = dir - > i_sb ;
sinfo - > slot_off = 0 ;
sinfo - > bh = NULL ;
while ( fat_get_short_entry ( dir , & sinfo - > slot_off , & sinfo - > bh ,
& sinfo - > de ) > = 0 ) {
if ( ! strncmp ( sinfo - > de - > name , name , MSDOS_NAME ) ) {
sinfo - > slot_off - = sizeof ( * sinfo - > de ) ;
sinfo - > nr_slots = 1 ;
sinfo - > i_pos = fat_make_i_pos ( sb , sinfo - > bh , sinfo - > de ) ;
return 0 ;
}
}
return - ENOENT ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_scan ) ;
2005-04-16 15:20:36 -07:00
static int __fat_remove_entries ( struct inode * dir , loff_t pos , int nr_slots )
{
struct super_block * sb = dir - > i_sb ;
struct buffer_head * bh ;
struct msdos_dir_entry * de , * endp ;
int err = 0 , orig_slots ;
while ( nr_slots ) {
bh = NULL ;
if ( fat_get_entry ( dir , & pos , & bh , & de ) < 0 ) {
err = - EIO ;
break ;
}
orig_slots = nr_slots ;
endp = ( struct msdos_dir_entry * ) ( bh - > b_data + sb - > s_blocksize ) ;
while ( nr_slots & & de < endp ) {
de - > name [ 0 ] = DELETED_FLAG ;
de + + ;
nr_slots - - ;
}
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bh , dir ) ;
2005-04-16 15:20:36 -07:00
if ( IS_DIRSYNC ( dir ) )
err = sync_dirty_buffer ( bh ) ;
brelse ( bh ) ;
if ( err )
break ;
/* pos is *next* de's position, so this does `- sizeof(de)' */
pos + = ( ( orig_slots - nr_slots ) * sizeof ( * de ) ) - sizeof ( * de ) ;
}
return err ;
}
int fat_remove_entries ( struct inode * dir , struct fat_slot_info * sinfo )
{
2011-04-12 21:08:38 +09:00
struct super_block * sb = dir - > i_sb ;
2005-04-16 15:20:36 -07:00
struct msdos_dir_entry * de ;
struct buffer_head * bh ;
int err = 0 , nr_slots ;
/*
* First stage : Remove the shortname . By this , the directory
* entry is removed .
*/
nr_slots = sinfo - > nr_slots ;
de = sinfo - > de ;
sinfo - > de = NULL ;
bh = sinfo - > bh ;
sinfo - > bh = NULL ;
while ( nr_slots & & de > = ( struct msdos_dir_entry * ) bh - > b_data ) {
de - > name [ 0 ] = DELETED_FLAG ;
de - - ;
nr_slots - - ;
}
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bh , dir ) ;
2005-04-16 15:20:36 -07:00
if ( IS_DIRSYNC ( dir ) )
err = sync_dirty_buffer ( bh ) ;
brelse ( bh ) ;
if ( err )
return err ;
dir - > i_version + + ;
if ( nr_slots ) {
/*
* Second stage : remove the remaining longname slots .
* ( This directory entry is already removed , and so return
* the success )
*/
err = __fat_remove_entries ( dir , sinfo - > slot_off , nr_slots ) ;
if ( err ) {
2011-04-12 21:08:38 +09:00
fat_msg ( sb , KERN_WARNING ,
" Couldn't remove the long name slots " ) ;
2005-04-16 15:20:36 -07:00
}
}
dir - > i_mtime = dir - > i_atime = CURRENT_TIME_SEC ;
if ( IS_DIRSYNC ( dir ) )
( void ) fat_sync_inode ( dir ) ;
else
mark_inode_dirty ( dir ) ;
return 0 ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_remove_entries ) ;
2005-04-16 15:20:36 -07:00
static int fat_zeroed_cluster ( struct inode * dir , sector_t blknr , int nr_used ,
struct buffer_head * * bhs , int nr_bhs )
{
struct super_block * sb = dir - > i_sb ;
sector_t last_blknr = blknr + MSDOS_SB ( sb ) - > sec_per_clus ;
int err , i , n ;
/* Zeroing the unused blocks on this cluster */
blknr + = nr_used ;
n = nr_used ;
while ( blknr < last_blknr ) {
bhs [ n ] = sb_getblk ( sb , blknr ) ;
if ( ! bhs [ n ] ) {
err = - ENOMEM ;
goto error ;
}
memset ( bhs [ n ] - > b_data , 0 , sb - > s_blocksize ) ;
set_buffer_uptodate ( bhs [ n ] ) ;
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bhs [ n ] , dir ) ;
2005-04-16 15:20:36 -07:00
n + + ;
blknr + + ;
if ( n = = nr_bhs ) {
if ( IS_DIRSYNC ( dir ) ) {
err = fat_sync_bhs ( bhs , n ) ;
if ( err )
goto error ;
}
for ( i = 0 ; i < n ; i + + )
brelse ( bhs [ i ] ) ;
n = 0 ;
}
}
if ( IS_DIRSYNC ( dir ) ) {
err = fat_sync_bhs ( bhs , n ) ;
if ( err )
goto error ;
}
for ( i = 0 ; i < n ; i + + )
brelse ( bhs [ i ] ) ;
return 0 ;
error :
for ( i = 0 ; i < n ; i + + )
bforget ( bhs [ i ] ) ;
return err ;
}
int fat_alloc_new_dir ( struct inode * dir , struct timespec * ts )
{
struct super_block * sb = dir - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bhs [ MAX_BUF_PER_PAGE ] ;
struct msdos_dir_entry * de ;
sector_t blknr ;
__le16 date , time ;
2008-11-06 12:53:47 -08:00
u8 time_cs ;
2005-04-16 15:20:36 -07:00
int err , cluster ;
err = fat_alloc_clusters ( dir , & cluster , 1 ) ;
if ( err )
goto error ;
blknr = fat_clus_to_blknr ( sbi , cluster ) ;
bhs [ 0 ] = sb_getblk ( sb , blknr ) ;
if ( ! bhs [ 0 ] ) {
err = - ENOMEM ;
goto error_free ;
}
2008-11-06 12:53:47 -08:00
fat_time_unix2fat ( sbi , ts , & time , & date , & time_cs ) ;
2005-04-16 15:20:36 -07:00
de = ( struct msdos_dir_entry * ) bhs [ 0 ] - > b_data ;
/* filling the new directory slots ("." and ".." entries) */
memcpy ( de [ 0 ] . name , MSDOS_DOT , MSDOS_NAME ) ;
memcpy ( de [ 1 ] . name , MSDOS_DOTDOT , MSDOS_NAME ) ;
de - > attr = de [ 1 ] . attr = ATTR_DIR ;
de [ 0 ] . lcase = de [ 1 ] . lcase = 0 ;
de [ 0 ] . time = de [ 1 ] . time = time ;
de [ 0 ] . date = de [ 1 ] . date = date ;
if ( sbi - > options . isvfat ) {
/* extra timestamps */
de [ 0 ] . ctime = de [ 1 ] . ctime = time ;
2008-11-06 12:53:47 -08:00
de [ 0 ] . ctime_cs = de [ 1 ] . ctime_cs = time_cs ;
2005-04-16 15:20:36 -07:00
de [ 0 ] . adate = de [ 0 ] . cdate = de [ 1 ] . adate = de [ 1 ] . cdate = date ;
} else {
de [ 0 ] . ctime = de [ 1 ] . ctime = 0 ;
2008-11-06 12:53:47 -08:00
de [ 0 ] . ctime_cs = de [ 1 ] . ctime_cs = 0 ;
2005-04-16 15:20:36 -07:00
de [ 0 ] . adate = de [ 0 ] . cdate = de [ 1 ] . adate = de [ 1 ] . cdate = 0 ;
}
de [ 0 ] . start = cpu_to_le16 ( cluster ) ;
de [ 0 ] . starthi = cpu_to_le16 ( cluster > > 16 ) ;
de [ 1 ] . start = cpu_to_le16 ( MSDOS_I ( dir ) - > i_logstart ) ;
de [ 1 ] . starthi = cpu_to_le16 ( MSDOS_I ( dir ) - > i_logstart > > 16 ) ;
de [ 0 ] . size = de [ 1 ] . size = 0 ;
memset ( de + 2 , 0 , sb - > s_blocksize - 2 * sizeof ( * de ) ) ;
set_buffer_uptodate ( bhs [ 0 ] ) ;
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bhs [ 0 ] , dir ) ;
2005-04-16 15:20:36 -07:00
err = fat_zeroed_cluster ( dir , blknr , 1 , bhs , MAX_BUF_PER_PAGE ) ;
if ( err )
goto error_free ;
return cluster ;
error_free :
fat_free_clusters ( dir , cluster ) ;
error :
return err ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_alloc_new_dir ) ;
2005-04-16 15:20:36 -07:00
static int fat_add_new_entries ( struct inode * dir , void * slots , int nr_slots ,
int * nr_cluster , struct msdos_dir_entry * * de ,
struct buffer_head * * bh , loff_t * i_pos )
{
struct super_block * sb = dir - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bhs [ MAX_BUF_PER_PAGE ] ;
sector_t blknr , start_blknr , last_blknr ;
unsigned long size , copy ;
int err , i , n , offset , cluster [ 2 ] ;
/*
* The minimum cluster size is 512 bytes , and maximum entry
* size is 32 * slots ( 672 bytes ) . So , iff the cluster size is
* 512 bytes , we may need two clusters .
*/
size = nr_slots * sizeof ( struct msdos_dir_entry ) ;
* nr_cluster = ( size + ( sbi - > cluster_size - 1 ) ) > > sbi - > cluster_bits ;
BUG_ON ( * nr_cluster > 2 ) ;
err = fat_alloc_clusters ( dir , cluster , * nr_cluster ) ;
if ( err )
goto error ;
/*
* First stage : Fill the directory entry . NOTE : This cluster
* is not referenced from any inode yet , so updates order is
* not important .
*/
i = n = copy = 0 ;
do {
start_blknr = blknr = fat_clus_to_blknr ( sbi , cluster [ i ] ) ;
last_blknr = start_blknr + sbi - > sec_per_clus ;
while ( blknr < last_blknr ) {
bhs [ n ] = sb_getblk ( sb , blknr ) ;
if ( ! bhs [ n ] ) {
err = - ENOMEM ;
goto error_nomem ;
}
/* fill the directory entry */
copy = min ( size , sb - > s_blocksize ) ;
memcpy ( bhs [ n ] - > b_data , slots , copy ) ;
slots + = copy ;
size - = copy ;
set_buffer_uptodate ( bhs [ n ] ) ;
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bhs [ n ] , dir ) ;
2005-04-16 15:20:36 -07:00
if ( ! size )
break ;
n + + ;
blknr + + ;
}
} while ( + + i < * nr_cluster ) ;
memset ( bhs [ n ] - > b_data + copy , 0 , sb - > s_blocksize - copy ) ;
offset = copy - sizeof ( struct msdos_dir_entry ) ;
get_bh ( bhs [ n ] ) ;
* bh = bhs [ n ] ;
* de = ( struct msdos_dir_entry * ) ( ( * bh ) - > b_data + offset ) ;
* i_pos = fat_make_i_pos ( sb , * bh , * de ) ;
/* Second stage: clear the rest of cluster, and write outs */
err = fat_zeroed_cluster ( dir , start_blknr , + + n , bhs , MAX_BUF_PER_PAGE ) ;
if ( err )
goto error_free ;
return cluster [ 0 ] ;
error_free :
brelse ( * bh ) ;
* bh = NULL ;
n = 0 ;
error_nomem :
for ( i = 0 ; i < n ; i + + )
bforget ( bhs [ i ] ) ;
fat_free_clusters ( dir , cluster [ 0 ] ) ;
error :
return err ;
}
int fat_add_entries ( struct inode * dir , void * slots , int nr_slots ,
struct fat_slot_info * sinfo )
{
struct super_block * sb = dir - > i_sb ;
struct msdos_sb_info * sbi = MSDOS_SB ( sb ) ;
struct buffer_head * bh , * prev , * bhs [ 3 ] ; /* 32*slots (672bytes) */
struct msdos_dir_entry * de ;
int err , free_slots , i , nr_bhs ;
loff_t pos , i_pos ;
sinfo - > nr_slots = nr_slots ;
/* First stage: search free direcotry entries */
free_slots = nr_bhs = 0 ;
bh = prev = NULL ;
pos = 0 ;
err = - ENOSPC ;
while ( fat_get_entry ( dir , & pos , & bh , & de ) > - 1 ) {
/* check the maximum size of directory */
if ( pos > = FAT_MAX_DIR_SIZE )
goto error ;
if ( IS_FREE ( de - > name ) ) {
if ( prev ! = bh ) {
get_bh ( bh ) ;
bhs [ nr_bhs ] = prev = bh ;
nr_bhs + + ;
}
free_slots + + ;
if ( free_slots = = nr_slots )
goto found ;
} else {
for ( i = 0 ; i < nr_bhs ; i + + )
brelse ( bhs [ i ] ) ;
prev = NULL ;
free_slots = nr_bhs = 0 ;
}
}
if ( dir - > i_ino = = MSDOS_ROOT_INO ) {
if ( sbi - > fat_bits ! = 32 )
goto error ;
} else if ( MSDOS_I ( dir ) - > i_start = = 0 ) {
2011-04-12 21:08:38 +09:00
fat_msg ( sb , KERN_ERR , " Corrupted directory (i_pos %lld) " ,
2005-04-16 15:20:36 -07:00
MSDOS_I ( dir ) - > i_pos ) ;
err = - EIO ;
goto error ;
}
found :
err = 0 ;
pos - = free_slots * sizeof ( * de ) ;
nr_slots - = free_slots ;
if ( free_slots ) {
/*
* Second stage : filling the free entries with new entries .
* NOTE : If this slots has shortname , first , we write
* the long name slots , then write the short name .
*/
int size = free_slots * sizeof ( * de ) ;
int offset = pos & ( sb - > s_blocksize - 1 ) ;
int long_bhs = nr_bhs - ( nr_slots = = 0 ) ;
/* Fill the long name slots. */
for ( i = 0 ; i < long_bhs ; i + + ) {
int copy = min_t ( int , sb - > s_blocksize - offset , size ) ;
memcpy ( bhs [ i ] - > b_data + offset , slots , copy ) ;
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bhs [ i ] , dir ) ;
2005-04-16 15:20:36 -07:00
offset = 0 ;
slots + = copy ;
size - = copy ;
}
if ( long_bhs & & IS_DIRSYNC ( dir ) )
err = fat_sync_bhs ( bhs , long_bhs ) ;
if ( ! err & & i < nr_bhs ) {
/* Fill the short name slot. */
int copy = min_t ( int , sb - > s_blocksize - offset , size ) ;
memcpy ( bhs [ i ] - > b_data + offset , slots , copy ) ;
2009-06-07 13:44:36 -04:00
mark_buffer_dirty_inode ( bhs [ i ] , dir ) ;
2005-04-16 15:20:36 -07:00
if ( IS_DIRSYNC ( dir ) )
err = sync_dirty_buffer ( bhs [ i ] ) ;
}
for ( i = 0 ; i < nr_bhs ; i + + )
brelse ( bhs [ i ] ) ;
if ( err )
goto error_remove ;
}
if ( nr_slots ) {
int cluster , nr_cluster ;
/*
* Third stage : allocate the cluster for new entries .
* And initialize the cluster with new entries , then
* add the cluster to dir .
*/
cluster = fat_add_new_entries ( dir , slots , nr_slots , & nr_cluster ,
& de , & bh , & i_pos ) ;
if ( cluster < 0 ) {
err = cluster ;
goto error_remove ;
}
err = fat_chain_add ( dir , cluster , nr_cluster ) ;
if ( err ) {
fat_free_clusters ( dir , cluster ) ;
goto error_remove ;
}
if ( dir - > i_size & ( sbi - > cluster_size - 1 ) ) {
2009-06-04 02:34:22 +09:00
fat_fs_error ( sb , " Odd directory size " ) ;
2005-04-16 15:20:36 -07:00
dir - > i_size = ( dir - > i_size + sbi - > cluster_size - 1 )
& ~ ( ( loff_t ) sbi - > cluster_size - 1 ) ;
}
dir - > i_size + = nr_cluster < < sbi - > cluster_bits ;
MSDOS_I ( dir ) - > mmu_private + = nr_cluster < < sbi - > cluster_bits ;
}
sinfo - > slot_off = pos ;
sinfo - > de = de ;
sinfo - > bh = bh ;
sinfo - > i_pos = fat_make_i_pos ( sb , sinfo - > bh , sinfo - > de ) ;
return 0 ;
error :
brelse ( bh ) ;
for ( i = 0 ; i < nr_bhs ; i + + )
brelse ( bhs [ i ] ) ;
return err ;
error_remove :
brelse ( bh ) ;
if ( free_slots )
__fat_remove_entries ( dir , pos , free_slots ) ;
return err ;
}
2006-01-08 01:02:10 -08:00
EXPORT_SYMBOL_GPL ( fat_add_entries ) ;