mirror of
https://github.com/deater/dos33fsprogs.git
synced 2025-01-07 12:31:57 +00:00
4f96a4e326
This was a dos33fs filesystem driver for Linux 2.4 It is included for historical reasons.
551 lines
17 KiB
C
551 lines
17 KiB
C
/*****************************************************************************
|
|
* super.c
|
|
* Superblock operations and basic interface to the Linux VFS.
|
|
*
|
|
* Apple II DOS 3.3 Filesystem Driver for Linux 2.4.x
|
|
* Copyright (c) 2001 Vince Weaver
|
|
* Copyright (c) 2001 Matt Jensen.
|
|
* This program is free software distributed under the terms of the GPL.
|
|
*****************************************************************************/
|
|
|
|
/*= Kernel Includes =========================================================*/
|
|
|
|
#include <linux/init.h>
|
|
#include <linux/module.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/blkdev.h>
|
|
|
|
/*= DOS 3.3 Includes =======================================================*/
|
|
|
|
#include "dos33.h"
|
|
|
|
/*= Forward Declarations ====================================================*/
|
|
|
|
/* For the Linux module interface. */
|
|
int __init init_dos33_fs(void);
|
|
void __exit exit_dos33_fs(void);
|
|
|
|
/* For dos33_fs_type. */
|
|
struct super_block *dos33_read_super(struct super_block *,void *,int);
|
|
|
|
/* For dos33_super_operations. */
|
|
void dos33_put_super(struct super_block *);
|
|
void dos33_write_super(struct super_block *);
|
|
int dos33_statfs(struct super_block *,struct statfs *);
|
|
|
|
/*= VFS Interface Structures ================================================*/
|
|
|
|
/* Linux module operations. */
|
|
EXPORT_NO_SYMBOLS;
|
|
module_init(init_dos33_fs)
|
|
module_exit(exit_dos33_fs)
|
|
|
|
/* DOS 3.3 driver metainformation. */
|
|
DECLARE_FSTYPE_DEV(dos33_fs_type,"dos33",dos33_read_super);
|
|
|
|
/* DOS 3.3 superblock operations. */
|
|
struct super_operations dos33_super_operations = {
|
|
read_inode: dos33_read_inode,
|
|
write_inode: dos33_write_inode,
|
|
put_inode: dos33_put_inode,
|
|
put_super: dos33_put_super,
|
|
// write_super: dos33_write_super,
|
|
statfs: dos33_statfs
|
|
};
|
|
|
|
/* Number of one bits in a given 4-bit value. */
|
|
static int onebits[] = { 0,1,1,2,1,2,2,3,1,2,2,3,2,3,3,4 };
|
|
|
|
/*= Private Functions =======================================================*/
|
|
|
|
/*****************************************************************************
|
|
* dos33_count_ones()
|
|
* Counts the one (set) bits in a block of @data, @size bytes long. This is
|
|
* basically the nifty algorithm that is also employed by (and was stolen
|
|
* from) the minix fs driver.
|
|
*****************************************************************************/
|
|
int dos33_count_ones(void *data,size_t size) {
|
|
|
|
const u32 *bytes = (u32*)data;
|
|
u32 ones = 0;
|
|
u32 tmp = 0;
|
|
|
|
DOS33_DEBUG("Count Ones\n");
|
|
|
|
/* Count one bits, one nibble at a time. */
|
|
for(size /= 4;size > 0;size--) {
|
|
tmp = *bytes++;
|
|
while (tmp) {
|
|
ones += onebits[tmp & 0x0f];
|
|
tmp >>= 4;
|
|
}
|
|
}
|
|
|
|
/* Return the number of ones found. */
|
|
return ones;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* dos33_find_first_one()
|
|
* Return the position of the first one bit in a block of @data which is @size
|
|
* bytes in length. Returns -1 if no one bits were found.
|
|
*****************************************************************************/
|
|
int dos33_find_first_one(void *data,size_t size) {
|
|
// const u32 *bytes = (u32*)data;
|
|
// int pos = 0;
|
|
// u32 tmp = 0;
|
|
|
|
DOS33_DEBUG("find_first_one\n");
|
|
#if 0
|
|
/* Search for a one bit; if found, return its position. */
|
|
for (size /= 4;size > 0;size--,bytes++,pos += 32) {
|
|
if (*bytes) {
|
|
for (tmp = be32_to_cpu(*bytes);!(tmp & 0x80000000);tmp <<= 1,pos++);
|
|
return pos;
|
|
}
|
|
}
|
|
|
|
/* No one bits were found. */
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_mark_block()
|
|
* Modify an entry in the volume bitmap. Note that the bitmap must be locked
|
|
* before calling this function.
|
|
*****************************************************************************/
|
|
void dos33_mark_block(struct super_block *sb,int block,int free) {
|
|
// SHORTCUT struct prodos_sb_info *psb = PRODOS_SB(sb);
|
|
// SHORTCUT struct buffer_head *bh = psb->s_bm_bh[block / 4096];
|
|
// u32 * const data = ((u32*)bh->b_data) + block % 4096 / 32;
|
|
// const int bit = cpu_to_be32(0x80000000 >> block % 4096 % 32);
|
|
DOS33_DEBUG("mark_block\n");
|
|
#if 0
|
|
/* Check whether we are to mark the block used or free. */
|
|
if (free) {
|
|
/* We should only be freeing used blocks. */
|
|
if (*data & bit) {
|
|
PRODOS_ERROR_1(sb,"block %i is already marked free",block);
|
|
return;
|
|
}
|
|
|
|
/* Mark it free. */
|
|
*data |= bit;
|
|
if (psb->s_bm_free >= 0) psb->s_bm_free++;
|
|
}
|
|
else {
|
|
/* We should only be allocating free blocks. */
|
|
if (!(*data & bit)) {
|
|
PRODOS_ERROR_1(sb,"block %i is already marked used",block);
|
|
return;
|
|
}
|
|
|
|
/* Mark it clear. */
|
|
*data &= ~bit;
|
|
if (psb->s_bm_free >= 0) psb->s_bm_free--;
|
|
}
|
|
#endif
|
|
/* The bitmap block is now dirty. */
|
|
// mark_buffer_dirty(bh);
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_parse_options()
|
|
* Parse the mount options string and fill prodos_sb_info fields accordingly.
|
|
*****************************************************************************/
|
|
int dos33_parse_options(struct super_block *sb,char *opts) {
|
|
// SHORTCUT struct prodos_sb_info * const psb = PRODOS_SB(sb);
|
|
// char *arg_key = NULL;
|
|
// char *arg_value = NULL;
|
|
DOS33_DEBUG("parse_options\n");
|
|
#if 0
|
|
/* Step through "key[=value]" options, modifying members of psb */
|
|
if (!opts) return 1;
|
|
for (arg_key = strtok(opts,",");arg_key;arg_key = strtok(NULL,",")) {
|
|
if ((arg_value = strchr(arg_key,'='))) *arg_value++ = 0;
|
|
if (!strcmp(arg_key,"verbose")) psb->s_flags |= PRODOS_FLAG_VERBOSE;
|
|
else if (!strcmp(arg_key,"crconv")) psb->s_flags |= PRODOS_FLAG_CONVERT_CR;
|
|
else if (!strcmp(arg_key,"case")) {
|
|
if (arg_value) {
|
|
if (!strcmp(arg_value,"lower"))
|
|
psb->s_flags |= PRODOS_FLAG_LOWERCASE;
|
|
else if (strcmp(arg_value,"asis")) goto out_invalid_value;
|
|
}
|
|
else goto out_omitted_value;
|
|
}
|
|
else if (!strcmp(arg_key,"forks")) {
|
|
if (arg_value) {
|
|
if (!strcmp(arg_value,"show"))
|
|
psb->s_flags |= PRODOS_FLAG_SHOW_FORKS;
|
|
else if (strcmp(arg_value,"hide")) goto out_invalid_value;
|
|
}
|
|
else goto out_omitted_value;
|
|
}
|
|
else if (!strcmp(arg_key,"partition") || !strcmp(arg_key,"part")) {
|
|
if (arg_value) strncpy(psb->s_part,arg_value,32);
|
|
else goto out_omitted_value;
|
|
}
|
|
else goto out_invalid_key;
|
|
}
|
|
|
|
/* Success. */
|
|
return ~0;
|
|
|
|
out_omitted_value:
|
|
PRODOS_ERROR_1(sb,"option \"%s\" requires a value",arg_key);
|
|
goto out_error;
|
|
out_invalid_value:
|
|
PRODOS_ERROR_2(sb,"option \"%s\" given unrecognized value \"%s\"",arg_key,arg_value);
|
|
goto out_error;
|
|
out_invalid_key:
|
|
PRODOS_ERROR_1(sb,"unrecognized option \"%s\"",arg_key);
|
|
goto out_error;
|
|
|
|
out_error:
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/*= Exported Functions ======================================================*/
|
|
|
|
/*****************************************************************************
|
|
* dos33_count_free_blocks()
|
|
* Count free blocks on the volume by counting the one bits in the volume
|
|
* bitmap.
|
|
*****************************************************************************/
|
|
int dos33_count_free_blocks(struct super_block *sb) {
|
|
|
|
SHORTCUT struct dos33_sb_info * const psb = DOS33_SB(sb);
|
|
|
|
DOS33_DEBUG("Count_free_blocks\n");
|
|
|
|
/* Grab the bitmap lock. */
|
|
down(&psb->s_bm_lock);
|
|
|
|
/* Count free blocks if they have not already been counted. */
|
|
if (psb->s_bm_free < 0) {
|
|
psb->s_bm_free = 0;
|
|
psb->s_bm_free += (dos33_count_ones(&(psb->bitmaps[0]),0x23))/2;
|
|
}
|
|
|
|
/* Release the bitmap lock and return free block count. */
|
|
up(&psb->s_bm_lock);
|
|
return psb->s_bm_free;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_alloc_block()
|
|
* Find the first free block on the volume, mark it used, and return its
|
|
* index.
|
|
*****************************************************************************/
|
|
int dos33_alloc_block(struct super_block *sb) {
|
|
// SHORTCUT struct prodos_sb_info * const psb = PRODOS_SB(sb);
|
|
int result = -ENOSPC;
|
|
// int pos = -1;
|
|
// int i = 0;
|
|
DOS33_DEBUG("alloc_block\n");
|
|
#if 0
|
|
/* Grab the bitmap lock. */
|
|
down(&psb->s_bm_lock);
|
|
|
|
/* Fail early if we already know that all blocks are full. */
|
|
if (!psb->s_bm_free) goto out_cleanup;
|
|
|
|
/* Find first one bit in the bitmap, which signifies a free block. */
|
|
for (i = 0;psb->s_bm_bh[i];i++) {
|
|
pos = prodos_find_first_one(psb->s_bm_bh[i]->b_data,PRODOS_BLOCK_SIZE);
|
|
if (pos != -1) {
|
|
pos += (i * 4096);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* XXX: should probably cache first free block index. */
|
|
/* If a bit was found, clear it. */
|
|
if (pos >= 0 && pos < psb->s_part_size) {
|
|
prodos_mark_block(sb,pos,0);
|
|
result = pos;
|
|
}
|
|
|
|
out_cleanup:
|
|
up(&psb->s_bm_lock);
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_free_block()
|
|
* Mark a block as free by setting its bit in the volume bitmap.
|
|
*****************************************************************************/
|
|
int dos33_free_block(struct super_block *sb,u16 block) {
|
|
DOS33_DEBUG("free_blocks\n");
|
|
#if 0
|
|
/* Grab the bitmap lock. */
|
|
down(&PRODOS_SB(sb)->s_bm_lock);
|
|
|
|
/* Set the bit. */
|
|
prodos_mark_block(sb,block,1);
|
|
|
|
/* Release the lock and return. */
|
|
up(&PRODOS_SB(sb)->s_bm_lock);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_free_tree_blocks()
|
|
* Mark all blocks in a tree (file or fork; seedling, sapling, or full tree)
|
|
* as free.
|
|
*****************************************************************************/
|
|
int prodos_free_tree_blocks(struct super_block *sb,u8 stype,u16 key_block,u16 blocks_used) {
|
|
int result = 0;
|
|
// struct buffer_head *dind_bh = NULL;
|
|
// u16 ind_block = 0;
|
|
// struct buffer_head *ind_bh = NULL;
|
|
DOS33_DEBUG("free_tree_blocks\n");
|
|
#if 0
|
|
/* Grab the bitmap lock. */
|
|
down(&PRODOS_SB(sb)->s_bm_lock);
|
|
|
|
/* Free blocks in sapling and tree files. */
|
|
if (stype != PRODOS_STYPE_SEEDLING) {
|
|
int i = 0;
|
|
u16 block = 0;
|
|
|
|
/* Load the double indirect block for tree files. */
|
|
if (stype == PRODOS_STYPE_TREE) {
|
|
dind_bh = prodos_bread(sb,key_block);
|
|
if (!dind_bh) {
|
|
result = -EIO;
|
|
goto out_cleanup;
|
|
}
|
|
|
|
/* Allow for double indirect to be freed outside the loop. */
|
|
blocks_used--;
|
|
}
|
|
|
|
/* Release all data blocks and indirect blocks. */
|
|
while (blocks_used) {
|
|
/* Get next indirect block when necessary. */
|
|
if (!(i % 256)) {
|
|
ind_block = dind_bh ?
|
|
PRODOS_GET_INDIRECT_ENTRY(dind_bh->b_data,i / 256) :
|
|
key_block;
|
|
ind_bh = prodos_bread(sb,ind_block);
|
|
if (!ind_bh) {
|
|
result = -EIO;
|
|
goto out_cleanup;
|
|
}
|
|
}
|
|
|
|
/* Free next data block (if it isn't sparse.) */
|
|
block = PRODOS_GET_INDIRECT_ENTRY(ind_bh->b_data,i % 256);
|
|
if (block) {
|
|
prodos_mark_block(sb,block,1);
|
|
blocks_used--;
|
|
}
|
|
|
|
/* Free indirect block when necessary. */
|
|
if (!(++i % 256) || (blocks_used == 1)) {
|
|
prodos_brelse(ind_bh);
|
|
ind_bh = NULL;
|
|
prodos_mark_block(sb,ind_block,1);
|
|
blocks_used--;
|
|
}
|
|
}
|
|
|
|
/* Free the double indirect block for tree files. */
|
|
if (stype == PRODOS_STYPE_TREE) {
|
|
prodos_brelse(dind_bh);
|
|
dind_bh = NULL;
|
|
prodos_mark_block(sb,key_block,1);
|
|
}
|
|
}
|
|
|
|
/* Free blocks in seedling files. */
|
|
else prodos_mark_block(sb,key_block,1);
|
|
|
|
out_cleanup:
|
|
if (ind_bh) prodos_brelse(ind_bh);
|
|
if (dind_bh) prodos_brelse(dind_bh);
|
|
up(&PRODOS_SB(sb)->s_bm_lock);
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
/*= Interface Functions =====================================================*/
|
|
|
|
/*****************************************************************************
|
|
* dos33_read_super()
|
|
*****************************************************************************/
|
|
struct super_block *dos33_read_super(struct super_block *sb,void *opts,int silent) {
|
|
|
|
SHORTCUT kdev_t const dev = sb->s_dev;
|
|
struct dos33_sb_info *psb = NULL;
|
|
struct buffer_head *bh = NULL;
|
|
struct dos33_vtol *pdbf = NULL;
|
|
struct inode *root_inode = NULL;
|
|
u16 i = 0;
|
|
|
|
DOS33_DEBUG("read_super\n");
|
|
|
|
/* Initialize the device and the super_block struct. */
|
|
set_blocksize(dev,DOS33_BLOCK_SIZE);
|
|
sb->s_magic = DOS33_SUPER_MAGIC;
|
|
sb->s_op = &dos33_super_operations;
|
|
sb->s_blocksize = DOS33_BLOCK_SIZE;
|
|
sb->s_blocksize_bits = DOS33_BLOCK_SIZE_BITS;
|
|
sb->s_maxbytes = DOS33_MAX_FILE_SIZE;
|
|
sb->s_flags = MS_RDONLY | MS_NOSUID;
|
|
|
|
/* Allocate the DOS 3.3 superblock metainformation struct. */
|
|
psb = kmalloc(sizeof(struct dos33_sb_info),GFP_KERNEL);
|
|
if (!psb) {
|
|
DOS33_ERROR("failed to allocate memory for prodos_sb_info");
|
|
goto out_error;
|
|
}
|
|
DOS33_SB(sb) = psb;
|
|
|
|
/* Fill the superblock metainformation struct with default values. */
|
|
memset(DOS33_SB(sb),0,sizeof(struct dos33_sb_info));
|
|
psb->s_part_size = blk_size[MAJOR(dev)] ? blk_size[MAJOR(dev)][MINOR(dev)] * 2 : 0;
|
|
init_MUTEX(&psb->s_bm_lock);
|
|
init_MUTEX(&psb->s_dir_lock);
|
|
|
|
/* Parse mount options and, if necessary, find desired partition. */
|
|
// if (!prodos_parse_options(sb,(char*)opts)) goto out_error;
|
|
|
|
/* Load DOS 3.3 volume directory block. */
|
|
bh = dos33_bread(sb,DOS33_VOLUME_DIR_BLOCK);
|
|
if (!bh) {
|
|
DOS33_ERROR("failed to read volume directory block");
|
|
goto out_error;
|
|
}
|
|
pdbf = (struct dos33_vtol*)bh->b_data;
|
|
|
|
/* Check for DOS 3.3 footprint in the block. */
|
|
/* this is a possibly-bogus check, replace */
|
|
/* if (pdbf->four != 0x4) {
|
|
DOS33_ERROR("failed to find a DOS 3.3 filesystem");
|
|
goto out_error;
|
|
}
|
|
*/
|
|
/* Use size from the volume directory as partition size. */
|
|
psb->s_part_size = ((pdbf->tracks_per_disk*pdbf->sectors_per_track*le16_to_cpu(pdbf->bytes_per_sector))/512);
|
|
|
|
/* Initialize the volume bitmap stuff. */
|
|
psb->s_bm_free = -1;
|
|
|
|
for(i=0;i<pdbf->tracks_per_disk;i++) {
|
|
psb->bitmaps[i]=cpu_to_le32(pdbf->bitmaps[i]);
|
|
}
|
|
|
|
printk("dos33.o: found %ik DOS 3.%i filesystem, Volume %i\n",
|
|
psb->s_part_size/2,pdbf->dos_version,pdbf->volume);
|
|
|
|
/* Get root inode. */
|
|
root_inode = iget(sb,DOS33_MAKE_INO(1,0,DOS33_VOLUME_DIR_TRACK,0xf));
|
|
sb->s_root = d_alloc_root(root_inode);
|
|
if (!sb->s_root) {
|
|
DOS33_ERROR("failed to get root inode");
|
|
goto out_error;
|
|
}
|
|
sb->s_root->d_op = &dos33_dentry_operations;
|
|
|
|
/* Return a copy of the 'sb' pointer on success. */
|
|
return sb;
|
|
|
|
out_error:
|
|
if (sb->s_root) {
|
|
iput(root_inode);
|
|
root_inode = NULL;
|
|
}
|
|
if (bh) {
|
|
dos33_brelse(bh);
|
|
bh = NULL;
|
|
}
|
|
if (psb) {
|
|
kfree(DOS33_SB(sb));
|
|
DOS33_SB(sb) = NULL;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_put_super()
|
|
*****************************************************************************/
|
|
void dos33_put_super(struct super_block *sb) {
|
|
SHORTCUT struct dos33_sb_info * const psb = DOS33_SB(sb);
|
|
// int i = 0;
|
|
|
|
DOS33_DEBUG("put_super\n");
|
|
|
|
#if 0 /* Release volume bitmap blocks. */
|
|
for (i = 0;psb->s_bm_bh[i];i++) {
|
|
prodos_brelse(psb->s_bm_bh[i]);
|
|
psb->s_bm_bh[i] = NULL;
|
|
}
|
|
#endif
|
|
/* Release the ProDOS super block metainformation structure. */
|
|
kfree(psb);
|
|
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* prodos_write_super()
|
|
*****************************************************************************
|
|
void prodos_write_super(struct super_block *sb) {
|
|
|
|
}*/
|
|
|
|
/*****************************************************************************
|
|
* prodos_statfs()
|
|
*****************************************************************************/
|
|
int dos33_statfs(struct super_block *sb,struct statfs *statfs) {
|
|
SHORTCUT struct dos33_sb_info * const psb = DOS33_SB(sb);
|
|
DOS33_DEBUG("statfs\n");
|
|
|
|
/* Copy applicable information. */
|
|
statfs->f_type = DOS33_SUPER_MAGIC;
|
|
statfs->f_bsize = sb->s_blocksize;
|
|
statfs->f_blocks = psb->s_part_size;
|
|
statfs->f_bfree = dos33_count_free_blocks(sb);
|
|
statfs->f_bavail = statfs->f_bfree;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*= Module Functions ========================================================*/
|
|
|
|
/*****************************************************************************
|
|
* init_dos33_filesystem()
|
|
*****************************************************************************/
|
|
int __init init_dos33_fs(void) {
|
|
/* Since we are overlaying our prodos_inode_info structure over a
|
|
minix_inode_info structure, we should verify that the former is no larger
|
|
than the latter. */
|
|
|
|
DOS33_DEBUG("Insmodding\n");
|
|
|
|
if (sizeof(struct dos33_inode_info) > sizeof(struct minix_inode_info)) {
|
|
DOS33_ERROR("aborting; struct prodos_inode_info too big!");
|
|
return -EFAULT;
|
|
}
|
|
|
|
/* Register the filesystem. */
|
|
return register_filesystem(&dos33_fs_type);
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* exit_dos33_filesystem()
|
|
*****************************************************************************/
|
|
void __exit exit_dos33_fs(void) {
|
|
/* Unregister the filesystem. */
|
|
|
|
DOS33_DEBUG("Rmmodding\n");
|
|
unregister_filesystem(&dos33_fs_type);
|
|
}
|