diff options
Diffstat (limited to 'src/kernel/fs/ext2.c')
-rw-r--r-- | src/kernel/fs/ext2.c | 884 |
1 files changed, 160 insertions, 724 deletions
diff --git a/src/kernel/fs/ext2.c b/src/kernel/fs/ext2.c index 41c2a3d..7ce0601 100644 --- a/src/kernel/fs/ext2.c +++ b/src/kernel/fs/ext2.c @@ -1,799 +1,235 @@ #include <stddef.h> #include <stdint.h> +#include <stdbool.h> +#include <kernel/fs/ata.h> +#include <kernel/fs/ext2.h> #include <kernel/system.h> -#include <kernel/lib/stdlib.h> -#include <kernel/lib/stdio.h> -#include <kernel/lib/lib.h> #include <kernel/memory/alloc.h> -#include <kernel/fs/ext2.h> - -uint32_t ext2_file_size(vfs_node_t *node) -{ - ext2_fs_t *ext2fs = node->device; - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, node->inode_num); - uint32_t ret = inode->size; - kfree(inode); - return ret; -} +#include <kernel/lib/lib.h> +#include <kernel/lib/stdlib.h> -void ext2_mkdir(vfs_node_t *parent, char *name, uint16_t permission) -{ - ext2_fs_t *ext2fs = parent->device; - uint32_t inode_idx = alloc_inode(ext2fs); - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, inode_idx); - inode->permission = EXT2_S_IFDIR; - inode->permission |= 0xFFF & permission; - inode->atime = 0; - inode->ctime = 0; - inode->dtime = 0; - inode->gid = 0; - inode->userid = 0; - inode->f_block_addr = 0; - inode->num_sectors = 0; - inode->size = ext2fs->block_size; - inode->hard_links = 2; - inode->flags = 0; - inode->file_acl = 0; - inode->dir_acl = 0; - inode->generation = 0; - inode->os_specific1 = 0; - memset(inode->blocks, 0, sizeof(inode->blocks)); - memset(inode->os_specific2, 0, 12); - alloc_inode_block(ext2fs, inode, inode_idx, 0); - write_inode_metadata(ext2fs, inode, inode_idx); - ext2_create_entry(parent, name, inode_idx); - - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - p_inode->hard_links++; - write_inode_metadata(ext2fs, p_inode, parent->inode_num); - rewrite_bgds(ext2fs); -} +static struct ext2_superblock superblock; +static struct bgd *bgdt; +static size_t block_size; +static size_t num_groups; +static void read_block(uint32_t block_num, void *buf); +static void load_superblock(); +static void load_bgdt(); +static void read_inode(struct ext2_inode *inode, uint32_t inode_num); -void ext2_mkfile(vfs_node_t *parent, char *name, uint16_t permission) +void ext2_init_fs() { - ext2_fs_t *ext2fs = parent->device; - uint32_t inode_idx = alloc_inode(ext2fs); - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, inode_idx); - inode->permission = EXT2_S_IFREG; - inode->permission |= 0xFFF & permission; - inode->atime = 0; - inode->ctime = 0; - inode->dtime = 0; - inode->gid = 0; - inode->userid = 0; - inode->f_block_addr = 0; - inode->num_sectors = 0; - inode->size = ext2fs->block_size; - inode->hard_links = 2; - inode->flags = 0; - inode->file_acl = 0; - inode->dir_acl = 0; - inode->generation = 0; - inode->os_specific1 = 0; - memset(inode->blocks, 0, sizeof(inode->blocks)); - memset(inode->os_specific2, 0, 12); - alloc_inode_block(ext2fs, inode, inode_idx, 0); - write_inode_metadata(ext2fs, inode, inode_idx); - ext2_create_entry(parent, name, inode_idx); - - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - p_inode->hard_links++; - write_inode_metadata(ext2fs, p_inode, parent->inode_num); - rewrite_bgds(ext2fs); -} + load_superblock(); + load_bgdt(); -void ext2_unlink(vfs_node_t *parent, char *name) -{ - ext2_fs_t *ext2fs = parent->device; - ext2_remove_entry(parent, name); - - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - p_inode->hard_links--; - write_inode_metadata(ext2fs, p_inode, parent->inode_num); - rewrite_bgds(ext2fs); -} + struct ext2_inode root_inode; + read_inode(&root_inode, ROOT_INODE); + log("Creation time = %d", root_inode.creation_time); + log("UID = %d", root_inode.uid); + log("Type & perms = 0x%x", root_inode.type_and_permissions); + log("Size = %d", root_inode.size); -char **ext2_listdir(vfs_node_t *parent) -{ - ext2_fs_t *ext2fs = parent->device; - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - uint32_t curr_offset = 0; - uint32_t block_offset = 0; - uint32_t in_block_offset = 0; - int size = 0, cap = 10; - char **ret = kmalloc(sizeof(char *) * cap); - char *block_buf = read_inode_block(ext2fs, p_inode, block_offset); - while (curr_offset < p_inode->size) { - if (in_block_offset >= ext2fs->block_size) { - block_offset++; - in_block_offset = 0; - block_buf = read_inode_block(ext2fs, p_inode, block_offset); - } - if (size + 1 == cap) { - ret = krealloc(ret, sizeof(char *) * cap * 2); - cap = cap * 2; - } + log("Files:"); - direntry_t *curr_dir = (direntry_t *)(block_buf + in_block_offset); - if (curr_dir->inode != 0) { - char *temp = kcalloc(curr_dir->name_len + 1, 1); - memcpy(temp, curr_dir->name, curr_dir->name_len); - ret[size++] = temp; - } - uint32_t expected_size = - ((sizeof(direntry_t) + curr_dir->name_len) & 0xfffffffc) + 0x4; - uint32_t real_size = curr_dir->size; - if (real_size != expected_size) { - break; - } - in_block_offset += curr_dir->size; - curr_offset += curr_dir->size; - } - ret[size] = NULL; - return ret; -} + struct ext2_file file; + ext2_open_inode(ROOT_INODE, &file); + struct ext2_dirent dirent; -vfs_node_t *vfsnode_from_direntry(ext2_fs_t *ext2fs, direntry_t *dir, inode_t *inode) -{ - vfs_node_t *ret = kcalloc(sizeof(vfs_node_t), 1); - - ret->device = (void *)ext2fs; - ret->inode_num = dir->inode; - memcpy(ret->name, dir->name, dir->name_len); - - ret->uid = inode->userid; - ret->uid = inode->gid; - ret->size = inode->size; - ret->mask = inode->permission & 0xFFF; - ret->nlink = inode->hard_links; - - ret->flags = 0; - if ((inode->permission & EXT2_S_IFREG) == EXT2_S_IFREG) { - ret->flags |= FS_FILE; - ret->read = ext2_read; - ret->write = ext2_write; - ret->unlink = ext2_unlink; - ret->get_file_size = ext2_file_size; - } - if ((inode->permission & EXT2_S_IFDIR) == EXT2_S_IFDIR) { - ret->flags |= FS_DIRECTORY; - ret->mkdir = ext2_mkdir; - ret->finddir = ext2_finddir; - ret->unlink = ext2_unlink; - ret->create = ext2_mkfile; - ret->listdir = ext2_listdir; - ret->read = ext2_read; - ret->write = ext2_write; - } - if ((inode->permission & EXT2_S_IFBLK) == EXT2_S_IFBLK) { - ret->flags |= FS_BLOCKDEVICE; - } - if ((inode->permission & EXT2_S_IFCHR) == EXT2_S_IFCHR) { - ret->flags |= FS_CHARDEVICE; - } - if ((inode->permission & EXT2_S_IFIFO) == EXT2_S_IFIFO) { - ret->flags |= FS_PIPE; - } - if ((inode->permission & EXT2_S_IFLNK) == EXT2_S_IFLNK) { - ret->flags |= FS_SYMLINK; - } + while (ext2_next_dirent(&file, &dirent)) + log("Inode %d, name `%s'", dirent.inode_num, dirent.name); - ret->access_time = inode->atime; - ret->modified_time = inode->mtime; - ret->create_time = inode->ctime; + kfree(file.buf); - ret->chmod = ext2_chmod; - ret->open = ext2_open; - ret->close = ext2_close; - return ret; + log("Looking for file '/test'..."); + uint32_t inode = ext2_look_up_path("/test"); + if (inode == 0) + log("File not found"); + else + log("Found: inode = %d", inode); + panic("nice"); } -vfs_node_t *ext2_finddir(vfs_node_t *parent, char *name) +static void read_block(uint32_t block_num, void *buf) { - ext2_fs_t *ext2fs = parent->device; - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - uint32_t expected_size; - uint32_t real_size; - uint32_t curr_offset = 0; - uint32_t block_offset = 0; - uint32_t in_block_offset = 0; - char *block_buf = read_inode_block(ext2fs, p_inode, block_offset); - while (curr_offset < p_inode->size) { - if (in_block_offset >= ext2fs->block_size) { - block_offset++; - in_block_offset = 0; - block_buf = read_inode_block(ext2fs, p_inode, block_offset); - } - - direntry_t *curr_dir = (direntry_t *)(block_buf + in_block_offset); - char *temp = kcalloc(curr_dir->name_len + 1, 1); - memcpy(temp, curr_dir->name, curr_dir->name_len); - if (curr_dir->inode != 0 && !strcmp(temp, name)) { - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, curr_dir->inode); - return vfsnode_from_direntry(ext2fs, curr_dir, inode); - } - if (((sizeof(direntry_t) + curr_dir->name_len) & 0x00000003) != 0) - expected_size = - ((sizeof(direntry_t) + curr_dir->name_len) & 0xfffffffc) + 0x4; - else - expected_size = ((sizeof(direntry_t) + curr_dir->name_len) & 0xfffffffc); - real_size = curr_dir->size; - if (real_size != expected_size) { - break; - } - in_block_offset += curr_dir->size; - curr_offset += curr_dir->size; - } - return NULL; -} + uint32_t lba = block_num * block_size / SECTOR_SIZE; + size_t sectors = block_size / SECTOR_SIZE; -void ext2_create_entry(vfs_node_t *parent, char *entry_name, uint32_t entry_inode) -{ - ext2_fs_t *ext2fs = parent->device; - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - uint32_t curr_offset = 0; - uint32_t block_offset = 0; - uint32_t in_block_offset = 0; - uint32_t found = 0; - uint32_t entry_name_len = strlen(entry_name); - char *check = kcalloc(entry_name_len + 1, 1); - char *block_buf = read_inode_block(ext2fs, p_inode, block_offset); - - while (curr_offset < p_inode->size) { - if (in_block_offset >= ext2fs->block_size) { - block_offset++; - in_block_offset = 0; - block_buf = read_inode_block(ext2fs, p_inode, block_offset); - } - direntry_t *curr_dir = (direntry_t *)(block_buf + in_block_offset); - if (curr_dir->name_len == entry_name_len) { - memcpy(check, curr_dir->name, entry_name_len); - if (curr_dir->inode != 0 && !strcmp(entry_name, check)) { - log("Entry by the same name %s already exist", check); - return; - } - } - if (found) { - curr_dir->inode = entry_inode; - curr_dir->size = - (uint32_t)block_buf + ext2fs->block_size - (uint32_t)curr_dir; - curr_dir->name_len = strlen(entry_name); - curr_dir->type = 0; - memcpy(curr_dir->name, entry_name, strlen(entry_name)); - write_inode_block(ext2fs, p_inode, block_offset, block_buf); - in_block_offset += curr_dir->size; - if (in_block_offset >= ext2fs->block_size) { - block_offset++; - in_block_offset = 0; - block_buf = read_inode_block(ext2fs, p_inode, block_offset); - } - curr_dir = (direntry_t *)(block_buf + in_block_offset); - memset(curr_dir, 0, sizeof(direntry_t)); - write_inode_block(ext2fs, p_inode, block_offset, block_buf); - return; - } - uint32_t expected_size = - ((sizeof(direntry_t) + curr_dir->name_len) & 0xfffffffc) + 0x4; - uint32_t real_size = curr_dir->size; - if (real_size != expected_size) { - found = 1; - curr_dir->size = expected_size; - in_block_offset += expected_size; - curr_offset += expected_size; - continue; - } - in_block_offset += curr_dir->size; - curr_offset += curr_dir->size; - } + read_abs_sectors(lba, sectors, buf); } -void ext2_remove_entry(vfs_node_t *parent, char *entry_name) +static void load_superblock() { - ext2_fs_t *ext2fs = parent->device; - inode_t *p_inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, p_inode, parent->inode_num); - uint32_t curr_offset = 0; - uint32_t block_offset = 0; - uint32_t in_block_offset = 0; - uint32_t entry_name_len = strlen(entry_name); - char *check = kcalloc(entry_name_len + 1, 1); - char *block_buf = read_inode_block(ext2fs, p_inode, block_offset); - while (curr_offset < p_inode->size) { - if (in_block_offset >= ext2fs->block_size) { - block_offset++; - in_block_offset = 0; - block_buf = read_inode_block(ext2fs, p_inode, block_offset); - } - direntry_t *curr_dir = (direntry_t *)(block_buf + in_block_offset); - if (curr_dir->name_len == entry_name_len) { - memcpy(check, curr_dir->name, entry_name_len); - if (curr_dir->inode != 0 && !strcmp(entry_name, check)) { - curr_dir->inode = 0; - write_inode_block(ext2fs, p_inode, block_offset, block_buf); - return; - } - } - uint32_t expected_size = - ((sizeof(direntry_t) + curr_dir->name_len) & 0xfffffffc) + 0x4; - uint32_t real_size = curr_dir->size; - if (real_size != expected_size) - return; - in_block_offset += curr_dir->size; - curr_offset += curr_dir->size; - } -} + uint16_t buf[SUPERBLOCK_LENGTH / 2]; -void ext2_chmod(vfs_node_t *file, uint32_t mode) -{ - ext2_fs_t *ext2fs = file->device; - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, file->inode_num); - inode->permission = (inode->permission & 0xFFFFF000) | mode; - write_inode_metadata(ext2fs, inode, file->inode_num); -} + read_abs_sectors(SUPERBLOCK_LBA, SUPERBLOCK_SECTORS, buf); + memcpy(&superblock, buf, sizeof(struct ext2_superblock)); -uint32_t ext2_read(vfs_node_t *file, uint32_t offset, uint32_t size, char *buf) -{ - ext2_fs_t *ext2fs = file->device; - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, file->inode_num); - read_inode_filedata(ext2fs, inode, offset, size, buf); - return size; -} + block_size = 1024 << superblock.log2_block_size; + num_groups = superblock.total_blocks / superblock.blocks_per_group; -uint32_t ext2_write(vfs_node_t *file, uint32_t offset, uint32_t size, char *buf) -{ - ext2_fs_t *ext2fs = file->device; - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, file->inode_num); - write_inode_filedata(ext2fs, inode, file->inode_num, offset, size, buf); - return size; + assert(superblock.signature == EXT2_SIGNATURE); + log("Total inodes = 0x%x", superblock.total_inodes); + log("Total blocks = 0x%x", superblock.total_blocks); + log("Block size = %b", block_size); + log("Num blocks = %d", superblock.total_blocks); + log("Blocks/group = %d", superblock.blocks_per_group); + log("Inodes/group = %d", superblock.inodes_per_group); + log("Num groups = %d", num_groups); } -void ext2_open(vfs_node_t *file, uint32_t flags) +static void load_bgdt() { - ext2_fs_t *ext2fs = file->device; - if (flags & O_TRUNC) { - inode_t *inode = kmalloc(sizeof(inode_t)); - read_inode_metadata(ext2fs, inode, file->inode_num); - inode->size = 0; - write_inode_metadata(ext2fs, inode, file->inode_num); - } -} + size_t bgdt_sectors = (sizeof(struct bgd) * num_groups) / SECTOR_SIZE + 1; + size_t bgdt_block = (SUPERBLOCK_OFFSET + SUPERBLOCK_LENGTH) / block_size + 1; + uint32_t bgdt_lba = bgdt_block * block_size / SECTOR_SIZE; -void ext2_close() -{ - return; -} + uint16_t buf[bgdt_sectors * SECTOR_SIZE / 2]; + read_abs_sectors(bgdt_lba, bgdt_sectors, buf); -void read_inode_metadata(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx) -{ - uint32_t group = inode_idx / ext2fs->inodes_per_group; - uint32_t inode_table_block = ext2fs->bgds[group].inode_table; - uint32_t idx_in_group = inode_idx - group * ext2fs->inodes_per_group; - uint32_t block_offset = (idx_in_group - 1) * ext2fs->sb->inode_size / ext2fs->block_size; - uint32_t offset_in_block = - (idx_in_group - 1) - block_offset * (ext2fs->block_size / ext2fs->sb->inode_size); - char *block_buf = kmalloc(ext2fs->block_size); - read_disk_block(ext2fs, inode_table_block + block_offset, block_buf); - memcpy(inode, block_buf + offset_in_block * ext2fs->sb->inode_size, ext2fs->sb->inode_size); - kfree(block_buf); + size_t bgdt_size = sizeof(struct bgd) * num_groups; + bgdt = kmalloc(bgdt_size); + memcpy(bgdt, buf, bgdt_size); } -void write_inode_metadata(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx) +static void read_inode(struct ext2_inode *inode, uint32_t inode_num) { - uint32_t group = inode_idx / ext2fs->inodes_per_group; - uint32_t inode_table_block = ext2fs->bgds[group].inode_table; - uint32_t block_offset = (inode_idx - 1) * ext2fs->sb->inode_size / ext2fs->block_size; - uint32_t offset_in_block = - (inode_idx - 1) - block_offset * (ext2fs->block_size / ext2fs->sb->inode_size); - char *block_buf = kmalloc(ext2fs->block_size); - read_disk_block(ext2fs, inode_table_block + block_offset, block_buf); - memcpy(block_buf + offset_in_block * ext2fs->sb->inode_size, inode, ext2fs->sb->inode_size); - write_disk_block(ext2fs, inode_table_block + block_offset, block_buf); - kfree(block_buf); -} + inode_num--; + size_t block_group = inode_num / superblock.inodes_per_group; -uint32_t read_inode_filedata(ext2_fs_t *ext2fs, inode_t *inode, uint32_t offset, uint32_t size, - char *buf) -{ - uint32_t end_offset = (inode->size >= offset + size) ? (offset + size) : (inode->size); - uint32_t start_block = offset / ext2fs->block_size; - uint32_t end_block = end_offset / ext2fs->block_size; - uint32_t start_off = offset % ext2fs->block_size; - uint32_t end_size = end_offset - end_block * ext2fs->block_size; - - uint32_t i = start_block; - uint32_t curr_off = 0; - while (i <= end_block) { - uint32_t left = 0, right = ext2fs->block_size - 1; - char *block_buf = read_inode_block(ext2fs, inode, i); - if (i == start_block) - left = start_off; - if (i == end_block) - right = end_size - 1; - memcpy(buf + curr_off, block_buf + left, (right - left + 1)); - curr_off = curr_off + (right - left + 1); - kfree(block_buf); - i++; - } - return end_offset - offset; -} + struct bgd *bgd = &bgdt[block_group]; + uint32_t i_table_block = bgd->inode_table_addr; -void write_inode_filedata(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx, uint32_t offset, - uint32_t size, char *buf) -{ - if (offset + size > inode->size) { - inode->size = offset + size; - write_inode_metadata(ext2fs, inode, inode_idx); - } - uint32_t end_offset = (inode->size >= offset + size) ? (offset + size) : (inode->size); - uint32_t start_block = offset / ext2fs->block_size; - uint32_t end_block = end_offset / ext2fs->block_size; - uint32_t start_off = offset % ext2fs->block_size; - uint32_t end_size = end_offset - end_block * ext2fs->block_size; - - uint32_t i = start_block; - uint32_t curr_off = 0; - while (i <= end_block) { - uint32_t left = 0, right = ext2fs->block_size; - char *block_buf = read_inode_block(ext2fs, inode, i); - - if (i == start_block) - left = start_off; - if (i == end_block) - right = end_size - 1; - memcpy(block_buf + left, buf + curr_off, (right - left + 1)); - curr_off = curr_off + (right - left + 1); - write_inode_block(ext2fs, inode, i, block_buf); - kfree(block_buf); - i++; - } -} + size_t index = inode_num % superblock.inodes_per_group; + size_t block_offset = (index * INODE_SIZE) / block_size; + size_t offset_in_block = (index * INODE_SIZE) % block_size; + size_t block = i_table_block + block_offset; -char *read_inode_block(ext2_fs_t *ext2fs, inode_t *inode, uint32_t iblock) -{ - char *buf = kmalloc(ext2fs->block_size); - uint32_t disk_block = get_disk_block_number(ext2fs, inode, iblock); - read_disk_block(ext2fs, disk_block, buf); - return buf; + size_t num_sectors = sizeof(struct ext2_inode) / SECTOR_SIZE + 1; + uint16_t buf[num_sectors * SECTOR_SIZE / 2]; + read_abs_sectors(block * block_size / SECTOR_SIZE, num_sectors, buf); + memcpy(inode, &buf[offset_in_block / 2], sizeof(struct ext2_inode)); } -void write_inode_block(ext2_fs_t *ext2fs, inode_t *inode, uint32_t iblock, char *buf) +void ext2_open_inode(uint32_t inode_num, struct ext2_file *file) { - uint32_t disk_block = get_disk_block_number(ext2fs, inode, iblock); - write_disk_block(ext2fs, disk_block, buf); -} + read_inode(&file->inode, inode_num); + file->pos = 0; + file->block_index = 0; + file->buf = kmalloc(block_size); + file->curr_block_pos = 0; -void read_disk_block(ext2_fs_t *ext2fs, uint32_t block, char *buf) -{ - vfs_read(ext2fs->disk_device, ext2fs->block_size * block, ext2fs->block_size, buf); + read_block(file->inode.dbp[0], file->buf); } -void write_disk_block(ext2_fs_t *ext2fs, uint32_t block, char *buf) +size_t ext2_read(struct ext2_file *file, uint8_t *buf, size_t count) { - vfs_write(ext2fs->disk_device, ext2fs->block_size * block, ext2fs->block_size, buf); -} + if (file->pos + count > file->inode.size) + count = file->inode.size - file->pos; -void rewrite_bgds(ext2_fs_t *ext2fs) -{ - for (uint32_t i = 0; i < ext2fs->bgd_blocks; i++) - write_disk_block(ext2fs, 2, (void *)ext2fs->bgds + i * ext2fs->block_size); -} + size_t bytes_left = count; -void rewrite_superblock(ext2_fs_t *ext2fs) -{ - write_disk_block(ext2fs, 1, (void *)ext2fs->sb); -} + while (bytes_left > 0) { + size_t to_copy = bytes_left; -int alloc_inode_metadata_block(uint32_t *block_ptr, ext2_fs_t *ext2fs, inode_t *inode, - uint32_t inode_idx, char *buffer, unsigned int block_overwrite) -{ - if (!(*block_ptr)) { - unsigned int block_no = ext2_alloc_block(ext2fs); - if (!block_no) - return 0; - *block_ptr = block_no; - if (buffer) - write_disk_block(ext2fs, block_overwrite, (void *)buffer); - else - write_inode_metadata(ext2fs, inode, inode_idx); - return 1; - } - return 0; -} + bool new_block = file->curr_block_pos + to_copy >= block_size; + if (new_block) + to_copy = block_size - file->curr_block_pos; -uint32_t get_disk_block_number(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_block) -{ - unsigned int p = ext2fs->block_size / 4; - int a, b, c, d, e, f, g; - uint32_t *tmp = kmalloc(ext2fs->block_size); - uint32_t ret = -1; - a = inode_block - EXT2_DIRECT_BLOCKS; - if (a < 0) { - ret = inode->blocks[inode_block]; - goto done; - } - b = a - p; - if (b < 0) { - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS], (void *)tmp); - ret = tmp[a]; - goto done; - } - c = b - p * p; - if (c < 0) { - c = b / p; - d = b - c * p; - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS + 1], (void *)tmp); - read_disk_block(ext2fs, tmp[c], (void *)tmp); - ret = tmp[d]; - goto done; - } - d = c - p * p * p; - if (d < 0) { - e = c / (p * p); - f = (c - e * p * p) / p; - g = (c - e * p * p - f * p); - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS + 2], (void *)tmp); - read_disk_block(ext2fs, tmp[e], (void *)tmp); - read_disk_block(ext2fs, tmp[f], (void *)tmp); - ret = tmp[g]; - goto done; - } -done: - kfree(tmp); - return ret; -} + memcpy(buf + (count - bytes_left), file->buf + file->curr_block_pos, to_copy); + file->curr_block_pos += to_copy; + file->pos += to_copy; + bytes_left -= to_copy; -void set_disk_block_number(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx, - uint32_t inode_block, uint32_t disk_block) -{ - unsigned int p = ext2fs->block_size / 4; - int a, b, c, d, e, f, g; - int iblock = inode_block; - uint32_t *tmp = kmalloc(ext2fs->block_size); - - a = iblock - EXT2_DIRECT_BLOCKS; - if (a <= 0) { - inode->blocks[inode_block] = disk_block; - goto done; - } - b = a - p; - if (b <= 0) { - if (!alloc_inode_metadata_block(&(inode->blocks[EXT2_DIRECT_BLOCKS]), ext2fs, inode, - inode_idx, NULL, 0)) - ; - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS], (void *)tmp); - ((unsigned int *)tmp)[a] = disk_block; - write_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS], (void *)tmp); - tmp[a] = disk_block; - goto done; - } - c = b - p * p; - if (c <= 0) { - c = b / p; - d = b - c * p; - if (!alloc_inode_metadata_block(&(inode->blocks[EXT2_DIRECT_BLOCKS + 1]), ext2fs, - inode, inode_idx, NULL, 0)) - ; - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS + 1], (void *)tmp); - if (!alloc_inode_metadata_block(&(tmp[c]), ext2fs, inode, inode_idx, (void *)tmp, - inode->blocks[EXT2_DIRECT_BLOCKS + 1])) - ; - unsigned int temp = tmp[c]; - read_disk_block(ext2fs, temp, (void *)tmp); - tmp[d] = disk_block; - write_disk_block(ext2fs, temp, (void *)tmp); - goto done; - } - d = c - p * p * p; - if (d <= 0) { - e = c / (p * p); - f = (c - e * p * p) / p; - g = (c - e * p * p - f * p); - if (!alloc_inode_metadata_block(&(inode->blocks[EXT2_DIRECT_BLOCKS + 2]), ext2fs, - inode, inode_idx, NULL, 0)) - ; - read_disk_block(ext2fs, inode->blocks[EXT2_DIRECT_BLOCKS + 2], (void *)tmp); - if (!alloc_inode_metadata_block(&(tmp[e]), ext2fs, inode, inode_idx, (void *)tmp, - inode->blocks[EXT2_DIRECT_BLOCKS + 2])) - ; - unsigned int temp = tmp[e]; - read_disk_block(ext2fs, tmp[e], (void *)tmp); - if (!alloc_inode_metadata_block(&(tmp[f]), ext2fs, inode, inode_idx, (void *)tmp, - temp)) - ; - temp = tmp[f]; - read_disk_block(ext2fs, tmp[f], (void *)tmp); - tmp[g] = disk_block; - write_disk_block(ext2fs, temp, (void *)tmp); - goto done; - } -done: - kfree(tmp); -} + if (new_block) { + file->curr_block_pos = 0; + file->block_index++; + if (file->block_index >= 12) + panic("Indirect block pointers are currently unsupported"); -uint32_t ext2_alloc_block(ext2_fs_t *ext2fs) -{ - uint32_t *buf = kcalloc(ext2fs->block_size, 1); - for (uint32_t i = 0; i < ext2fs->total_groups; i++) { - if (!ext2fs->bgds[i].free_blocks) - continue; - - uint32_t bitmap_block = ext2fs->bgds[i].block_bitmap; - read_disk_block(ext2fs, bitmap_block, (void *)buf); - for (uint32_t j = 0; j < ext2fs->block_size / 4; j++) { - uint32_t sub_bitmap = buf[j]; - if (sub_bitmap == 0xFFFFFFFF) - continue; - for (uint32_t k = 0; k < 32; k++) { - uint32_t free = !((sub_bitmap >> k) & 0x1); - if (free) { - uint32_t mask = (0x1 << k); - buf[j] = buf[j] | mask; - write_disk_block(ext2fs, bitmap_block, (void *)buf); - ext2fs->bgds[i].free_blocks--; - rewrite_bgds(ext2fs); - return i * ext2fs->blocks_per_group + j * 32 + k; - } - } + read_block(file->inode.dbp[file->block_index], file->buf); } } - panic("We're out of blocks!"); - return (uint32_t)-1; + + return count; } -void ext2_free_block(ext2_fs_t *ext2fs, uint32_t block) +#define READ_SIZE (sizeof(struct ext2_dirent) - sizeof(uint8_t *)) + +bool ext2_next_dirent(struct ext2_file *file, struct ext2_dirent *dir) { - uint32_t *buf = kcalloc(ext2fs->block_size, 1); - uint32_t group_idx = block / ext2fs->blocks_per_group; - uint32_t sub_bitmap_idx = (block - (ext2fs->blocks_per_group * group_idx)) / 4; - uint32_t idx = (block - (ext2fs->blocks_per_group * group_idx)) % 4; + uint8_t buf[READ_SIZE]; + if (ext2_read(file, buf, READ_SIZE) != READ_SIZE) + return false; - uint32_t bitmap_block = ext2fs->bgds[group_idx].block_bitmap; - read_disk_block(ext2fs, bitmap_block, (void *)buf); + memcpy(dir, buf, READ_SIZE); - uint32_t mask = ~(0x1 << idx); - buf[sub_bitmap_idx] = buf[sub_bitmap_idx] & mask; + size_t size = dir->name_len + 1; + uint8_t *name = kmalloc(size); + if (ext2_read(file, name, size - 1) != size - 1) + return false; - write_disk_block(ext2fs, bitmap_block, (void *)buf); + dir->name = name; + dir->name[size - 1] = '\0'; - ext2fs->bgds[group_idx].free_blocks++; - rewrite_bgds(ext2fs); -} + size_t bytes_left = dir->total_len - (READ_SIZE + size - 1); + if (bytes_left > 0) { + uint8_t dummy[bytes_left]; + ext2_read(file, dummy, bytes_left); + } -void alloc_inode_block(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx, uint32_t block) -{ - uint32_t ret = ext2_alloc_block(ext2fs); - set_disk_block_number(ext2fs, inode, inode_idx, block, ret); - inode->num_sectors = (block + 1) * (ext2fs->block_size / 512); - write_inode_metadata(ext2fs, inode, inode_idx); + return true; } -void free_inode_block(ext2_fs_t *ext2fs, inode_t *inode, uint32_t inode_idx, uint32_t block) +uint32_t ext2_find_in_dir(uint32_t dir_inode, const char *name) { - uint32_t ret = get_disk_block_number(ext2fs, inode, block); - ext2_free_block(ext2fs, ret); - set_disk_block_number(ext2fs, inode, inode_idx, ret, 0); - write_inode_metadata(ext2fs, inode, inode_idx); -} + uint32_t inode; + struct ext2_file dir; + struct ext2_dirent dirent; -uint32_t alloc_inode(ext2_fs_t *ext2fs) -{ - uint32_t *buf = kcalloc(ext2fs->block_size, 1); - for (uint32_t i = 0; i < ext2fs->total_groups; i++) { - if (!ext2fs->bgds[i].free_inodes) - continue; - - uint32_t bitmap_block = ext2fs->bgds[i].inode_bitmap; - read_disk_block(ext2fs, bitmap_block, (void *)buf); - for (uint32_t j = 0; j < ext2fs->block_size / 4; j++) { - uint32_t sub_bitmap = buf[j]; - if (sub_bitmap == 0xFFFFFFFF) - continue; - for (uint32_t k = 0; k < 32; k++) { - uint32_t free = !((sub_bitmap >> k) & 0x1); - if (free) { - uint32_t mask = (0x1 << k); - buf[j] = buf[j] | mask; - write_disk_block(ext2fs, bitmap_block, (void *)buf); - ext2fs->bgds[i].free_inodes--; - rewrite_bgds(ext2fs); - return i * ext2fs->inodes_per_group + j * 32 + k; - } - } + ext2_open_inode(dir_inode, &dir); + while (ext2_next_dirent(&dir, &dirent)) { + if (strcmp((char *)dirent.name, name) == 0) { + inode = dirent.inode_num; + goto cleanup; } } - panic("We're out of inodes!"); - return (uint32_t)-1; + + inode = 0; + +cleanup: + kfree(dir.buf); + return inode; } -void free_inode(ext2_fs_t *ext2fs, uint32_t inode) +uint32_t ext2_look_up_path(char *path) { - uint32_t *buf = kcalloc(ext2fs->block_size, 1); - uint32_t group_idx = inode / ext2fs->inodes_per_group; - uint32_t sub_bitmap_idx = (inode - (ext2fs->inodes_per_group * group_idx)) / 4; - uint32_t idx = (inode - (ext2fs->inodes_per_group * group_idx)) % 4; + if (path[0] != '/') + return 0; - uint32_t bitmap_block = ext2fs->bgds[group_idx].inode_bitmap; - read_disk_block(ext2fs, bitmap_block, (void *)buf); + path++; + uint32_t curr_dir_inode = ROOT_INODE; - uint32_t mask = ~(0x1 << idx); - buf[sub_bitmap_idx] = buf[sub_bitmap_idx] & mask; + for (;;) { + size_t j; + for (j = 0; path[j] != '/' && path[j] != '\0'; j++) + ; - write_disk_block(ext2fs, bitmap_block, (void *)buf); + if (path[j] == '\0') + break; - ext2fs->bgds[group_idx].free_inodes++; - rewrite_bgds(ext2fs); -} + path[j] = '\0'; + curr_dir_inode = ext2_find_in_dir(curr_dir_inode, path); + path[j] = '/'; -vfs_node_t *get_ext2_root(ext2_fs_t *ext2fs, inode_t *inode) -{ - vfs_node_t *ext2root = kcalloc(sizeof(vfs_node_t), 1); - strcpy(ext2root->name, "/"); - ext2root->device = ext2fs; - ext2root->mask = inode->permission; - ext2root->inode_num = ROOT_INODE_NUMBER; - - ext2root->access_time = inode->atime; - ext2root->modified_time = inode->mtime; - ext2root->create_time = inode->ctime; - - ext2root->flags |= FS_DIRECTORY; - ext2root->read = NULL; - ext2root->write = NULL; - ext2root->chmod = ext2_chmod; - ext2root->open = ext2_open; - ext2root->close = ext2_close; - ext2root->read = ext2_read; - ext2root->write = ext2_write; - ext2root->mkdir = ext2_mkdir; - ext2root->create = ext2_mkfile; - ext2root->listdir = ext2_listdir; - ext2root->finddir = ext2_finddir; - ext2root->unlink = ext2_unlink; - return ext2root; -} + if (curr_dir_inode == 0) + return 0; -void ext2_init(char *device_path, char *mountpoint) -{ - ext2_fs_t *ext2fs = kcalloc(sizeof(ext2_fs_t), 1); - ext2fs->disk_device = file_open(device_path, 0); - log("%s", ext2fs->disk_device->name); - ext2fs->sb = kmalloc(SUPERBLOCK_SIZE); - ext2fs->block_size = 1024; - read_disk_block(ext2fs, 1, (void *)ext2fs->sb); //! - log("%x", ext2fs->sb->ext2_magic); - log("%x", ext2fs->sb->total_inodes); - ext2fs->block_size = (1024 << ext2fs->sb->log2block_size); - ext2fs->blocks_per_group = ext2fs->sb->blocks_per_group; - ext2fs->inodes_per_group = ext2fs->sb->inodes_per_group; - - ext2fs->total_groups = ext2fs->sb->total_blocks / - ext2fs->blocks_per_group; // REMEMBER: Zero Division Exception - if (ext2fs->blocks_per_group * ext2fs->total_groups < ext2fs->total_groups) - ext2fs->total_groups++; - - ext2fs->bgd_blocks = (ext2fs->total_groups * sizeof(bgd_t)) / ext2fs->block_size; - if (ext2fs->bgd_blocks * ext2fs->block_size < ext2fs->total_groups * sizeof(bgd_t)) - ext2fs->bgd_blocks++; - - ext2fs->bgds = kcalloc(sizeof(bgd_t), ext2fs->bgd_blocks * ext2fs->block_size); - for (uint32_t i = 0; i < ext2fs->bgd_blocks; i++) { - read_disk_block(ext2fs, 2, (void *)ext2fs->bgds + i * ext2fs->block_size); + path += j + 1; } - inode_t *root_inode = kcalloc(sizeof(inode_t), 1); - read_inode_metadata(ext2fs, root_inode, ROOT_INODE_NUMBER); - vfs_mount(mountpoint, get_ext2_root(ext2fs, root_inode)); + uint32_t inode = ext2_find_in_dir(curr_dir_inode, path); + if (inode == 0) + return 0; + + return inode; }
\ No newline at end of file |