9769f4eb3f
The current isofs treatment of hidden files is flawed in two ways. First, it does not provide sufficient granularity; it hides both 'hidden' files and 'associated' files (resource fork for Mac files). Second, the default behavior to completely strip hidden files, while an admirable implementation of the spec, is a poor choice given the real world use of hidden files as a poor mans copy protection scheme for MSDOS and Windows based systems. A longer description of this is available here: http://www.uwsg.iu.edu/hypermail/linux/kernel/0205.3/0267.html This patch was originally built after a few private conversations with Alan Cox; I shamefully failed to persist in seeing it go forward, I hope to make amends now. This patch introduces granularity by allowing explicit control for both hidden and associated files. It also reverses the default so that by default, hidden files are treated as regular files on the iso9660 file system. This allow Wine to process Windows CDs, including those that are hybrid Mac/Windows CDs properly and completely, without our having to go muck up peoples fstabs as we do now. (I have tested this with such a hybrid + hidden CD and have verified that this patch works as claimed). Signed-off-by: Jeremy White <jwhite@codeweavers.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
192 lines
6.4 KiB
C
192 lines
6.4 KiB
C
#include <linux/fs.h>
|
|
#include <linux/buffer_head.h>
|
|
#include <linux/iso_fs.h>
|
|
#include <asm/unaligned.h>
|
|
|
|
enum isofs_file_format {
|
|
isofs_file_normal = 0,
|
|
isofs_file_sparse = 1,
|
|
isofs_file_compressed = 2,
|
|
};
|
|
|
|
/*
|
|
* iso fs inode data in memory
|
|
*/
|
|
struct iso_inode_info {
|
|
unsigned long i_iget5_block;
|
|
unsigned long i_iget5_offset;
|
|
unsigned int i_first_extent;
|
|
unsigned char i_file_format;
|
|
unsigned char i_format_parm[3];
|
|
unsigned long i_next_section_block;
|
|
unsigned long i_next_section_offset;
|
|
off_t i_section_size;
|
|
struct inode vfs_inode;
|
|
};
|
|
|
|
/*
|
|
* iso9660 super-block data in memory
|
|
*/
|
|
struct isofs_sb_info {
|
|
unsigned long s_ninodes;
|
|
unsigned long s_nzones;
|
|
unsigned long s_firstdatazone;
|
|
unsigned long s_log_zone_size;
|
|
unsigned long s_max_size;
|
|
|
|
unsigned char s_high_sierra; /* A simple flag */
|
|
unsigned char s_mapping;
|
|
int s_rock_offset; /* offset of SUSP fields within SU area */
|
|
unsigned char s_rock;
|
|
unsigned char s_joliet_level;
|
|
unsigned char s_utf8;
|
|
unsigned char s_cruft; /* Broken disks with high
|
|
byte of length containing
|
|
junk */
|
|
unsigned char s_unhide;
|
|
unsigned char s_nosuid;
|
|
unsigned char s_nodev;
|
|
unsigned char s_nocompress;
|
|
unsigned char s_hide;
|
|
unsigned char s_showassoc;
|
|
|
|
mode_t s_mode;
|
|
gid_t s_gid;
|
|
uid_t s_uid;
|
|
struct nls_table *s_nls_iocharset; /* Native language support table */
|
|
};
|
|
|
|
static inline struct isofs_sb_info *ISOFS_SB(struct super_block *sb)
|
|
{
|
|
return sb->s_fs_info;
|
|
}
|
|
|
|
static inline struct iso_inode_info *ISOFS_I(struct inode *inode)
|
|
{
|
|
return container_of(inode, struct iso_inode_info, vfs_inode);
|
|
}
|
|
|
|
static inline int isonum_711(char *p)
|
|
{
|
|
return *(u8 *)p;
|
|
}
|
|
static inline int isonum_712(char *p)
|
|
{
|
|
return *(s8 *)p;
|
|
}
|
|
static inline unsigned int isonum_721(char *p)
|
|
{
|
|
return le16_to_cpu(get_unaligned((__le16 *)p));
|
|
}
|
|
static inline unsigned int isonum_722(char *p)
|
|
{
|
|
return be16_to_cpu(get_unaligned((__le16 *)p));
|
|
}
|
|
static inline unsigned int isonum_723(char *p)
|
|
{
|
|
/* Ignore bigendian datum due to broken mastering programs */
|
|
return le16_to_cpu(get_unaligned((__le16 *)p));
|
|
}
|
|
static inline unsigned int isonum_731(char *p)
|
|
{
|
|
return le32_to_cpu(get_unaligned((__le32 *)p));
|
|
}
|
|
static inline unsigned int isonum_732(char *p)
|
|
{
|
|
return be32_to_cpu(get_unaligned((__le32 *)p));
|
|
}
|
|
static inline unsigned int isonum_733(char *p)
|
|
{
|
|
/* Ignore bigendian datum due to broken mastering programs */
|
|
return le32_to_cpu(get_unaligned((__le32 *)p));
|
|
}
|
|
extern int iso_date(char *, int);
|
|
|
|
struct inode; /* To make gcc happy */
|
|
|
|
extern int parse_rock_ridge_inode(struct iso_directory_record *, struct inode *);
|
|
extern int get_rock_ridge_filename(struct iso_directory_record *, char *, struct inode *);
|
|
extern int isofs_name_translate(struct iso_directory_record *, char *, struct inode *);
|
|
|
|
int get_joliet_filename(struct iso_directory_record *, unsigned char *, struct inode *);
|
|
int get_acorn_filename(struct iso_directory_record *, char *, struct inode *);
|
|
|
|
extern struct dentry *isofs_lookup(struct inode *, struct dentry *, struct nameidata *);
|
|
extern struct buffer_head *isofs_bread(struct inode *, sector_t);
|
|
extern int isofs_get_blocks(struct inode *, sector_t, struct buffer_head **, unsigned long);
|
|
|
|
extern struct inode *isofs_iget(struct super_block *sb,
|
|
unsigned long block,
|
|
unsigned long offset);
|
|
|
|
/* Because the inode number is no longer relevant to finding the
|
|
* underlying meta-data for an inode, we are free to choose a more
|
|
* convenient 32-bit number as the inode number. The inode numbering
|
|
* scheme was recommended by Sergey Vlasov and Eric Lammerts. */
|
|
static inline unsigned long isofs_get_ino(unsigned long block,
|
|
unsigned long offset,
|
|
unsigned long bufbits)
|
|
{
|
|
return (block << (bufbits - 5)) | (offset >> 5);
|
|
}
|
|
|
|
/* Every directory can have many redundant directory entries scattered
|
|
* throughout the directory tree. First there is the directory entry
|
|
* with the name of the directory stored in the parent directory.
|
|
* Then, there is the "." directory entry stored in the directory
|
|
* itself. Finally, there are possibly many ".." directory entries
|
|
* stored in all the subdirectories.
|
|
*
|
|
* In order for the NFS get_parent() method to work and for the
|
|
* general consistency of the dcache, we need to make sure the
|
|
* "i_iget5_block" and "i_iget5_offset" all point to exactly one of
|
|
* the many redundant entries for each directory. We normalize the
|
|
* block and offset by always making them point to the "." directory.
|
|
*
|
|
* Notice that we do not use the entry for the directory with the name
|
|
* that is located in the parent directory. Even though choosing this
|
|
* first directory is more natural, it is much easier to find the "."
|
|
* entry in the NFS get_parent() method because it is implicitly
|
|
* encoded in the "extent + ext_attr_length" fields of _all_ the
|
|
* redundant entries for the directory. Thus, it can always be
|
|
* reached regardless of which directory entry you have in hand.
|
|
*
|
|
* This works because the "." entry is simply the first directory
|
|
* record when you start reading the file that holds all the directory
|
|
* records, and this file starts at "extent + ext_attr_length" blocks.
|
|
* Because the "." entry is always the first entry listed in the
|
|
* directories file, the normalized "offset" value is always 0.
|
|
*
|
|
* You should pass the directory entry in "de". On return, "block"
|
|
* and "offset" will hold normalized values. Only directories are
|
|
* affected making it safe to call even for non-directory file
|
|
* types. */
|
|
static inline void
|
|
isofs_normalize_block_and_offset(struct iso_directory_record* de,
|
|
unsigned long *block,
|
|
unsigned long *offset)
|
|
{
|
|
/* Only directories are normalized. */
|
|
if (de->flags[0] & 2) {
|
|
*offset = 0;
|
|
*block = (unsigned long)isonum_733(de->extent)
|
|
+ (unsigned long)isonum_711(de->ext_attr_length);
|
|
}
|
|
}
|
|
|
|
extern struct inode_operations isofs_dir_inode_operations;
|
|
extern struct file_operations isofs_dir_operations;
|
|
extern struct address_space_operations isofs_symlink_aops;
|
|
extern struct export_operations isofs_export_ops;
|
|
|
|
/* The following macros are used to check for memory leaks. */
|
|
#ifdef LEAK_CHECK
|
|
#define free_s leak_check_free_s
|
|
#define malloc leak_check_malloc
|
|
#define sb_bread leak_check_bread
|
|
#define brelse leak_check_brelse
|
|
extern void * leak_check_malloc(unsigned int size);
|
|
extern void leak_check_free_s(void * obj, int size);
|
|
extern struct buffer_head * leak_check_bread(struct super_block *sb, int block);
|
|
extern void leak_check_brelse(struct buffer_head * bh);
|
|
#endif /* LEAK_CHECK */
|