2018-10-10 11:53:07 +02:00
# include <AK/Bitmap.h>
2018-10-24 12:43:52 +02:00
# include <AK/BufferStream.h>
2019-06-07 11:43:58 +02:00
# include <AK/StdLibExtras.h>
2019-06-07 19:29:34 +02:00
# include <Kernel/FileSystem/Ext2FileSystem.h>
# include <Kernel/FileSystem/ext2_fs.h>
2019-02-21 15:45:31 +01:00
# include <Kernel/Process.h>
2019-06-07 19:29:34 +02:00
# include <Kernel/RTC.h>
# include <Kernel/UnixTypes.h>
2019-06-07 11:43:58 +02:00
# include <LibC/errno_numbers.h>
2018-10-10 11:53:07 +02:00
2018-10-17 11:47:14 +02:00
//#define EXT2_DEBUG
2018-10-10 11:53:07 +02:00
2019-09-30 11:20:51 +02:00
static const size_t max_block_size = 4096 ;
2019-03-02 01:50:34 +01:00
static const ssize_t max_inline_symlink_length = 60 ;
2019-07-03 21:17:35 +02:00
static u8 to_ext2_file_type ( mode_t mode )
2019-05-31 17:41:33 +02:00
{
if ( is_regular_file ( mode ) )
return EXT2_FT_REG_FILE ;
if ( is_directory ( mode ) )
return EXT2_FT_DIR ;
if ( is_character_device ( mode ) )
return EXT2_FT_CHRDEV ;
if ( is_block_device ( mode ) )
return EXT2_FT_BLKDEV ;
if ( is_fifo ( mode ) )
return EXT2_FT_FIFO ;
if ( is_socket ( mode ) )
return EXT2_FT_SOCK ;
if ( is_symlink ( mode ) )
return EXT2_FT_SYMLINK ;
return EXT2_FT_UNKNOWN ;
}
2019-07-11 15:38:47 +02:00
NonnullRefPtr < Ext2FS > Ext2FS : : create ( NonnullRefPtr < DiskDevice > device )
2018-10-10 11:53:07 +02:00
{
2018-11-15 17:13:10 +01:00
return adopt ( * new Ext2FS ( move ( device ) ) ) ;
2018-10-10 11:53:07 +02:00
}
2019-06-21 18:37:47 +02:00
Ext2FS : : Ext2FS ( NonnullRefPtr < DiskDevice > & & device )
2018-11-15 17:13:10 +01:00
: DiskBackedFS ( move ( device ) )
2018-10-10 11:53:07 +02:00
{
}
2018-11-15 17:13:10 +01:00
Ext2FS : : ~ Ext2FS ( )
2018-10-10 11:53:07 +02:00
{
}
2019-11-02 11:49:11 +01:00
bool Ext2FS : : flush_super_block ( )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-11-02 11:49:11 +01:00
bool success = device ( ) . write_blocks ( 2 , 1 , ( const u8 * ) & m_super_block ) ;
2019-02-11 11:38:14 +01:00
ASSERT ( success ) ;
2018-10-10 11:53:07 +02:00
return true ;
}
2019-05-25 17:23:17 +02:00
unsigned Ext2FS : : first_block_of_group ( GroupIndex group_index ) const
2018-10-10 11:53:07 +02:00
{
2019-05-25 17:23:17 +02:00
return super_block ( ) . s_first_data_block + ( group_index * super_block ( ) . s_blocks_per_group ) ;
2018-10-10 11:53:07 +02:00
}
2019-05-25 17:23:17 +02:00
const ext2_group_desc & Ext2FS : : group_descriptor ( GroupIndex group_index ) const
2018-10-10 11:53:07 +02:00
{
// FIXME: Should this fail gracefully somehow?
2019-05-25 17:23:17 +02:00
ASSERT ( group_index < = m_block_group_count ) ;
2018-10-10 11:53:07 +02:00
2018-12-03 00:20:00 +01:00
if ( ! m_cached_group_descriptor_table ) {
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-31 17:31:23 +01:00
unsigned blocks_to_read = ceil_div ( m_block_group_count * ( unsigned ) sizeof ( ext2_group_desc ) , block_size ( ) ) ;
unsigned first_block_of_bgdt = block_size ( ) = = 1024 ? 2 : 1 ;
2018-10-24 13:38:53 +02:00
# ifdef EXT2_DEBUG
2019-01-31 17:31:23 +01:00
kprintf ( " ext2fs: block group count: %u, blocks-to-read: %u \n " , m_block_group_count , blocks_to_read ) ;
kprintf ( " ext2fs: first block of BGDT: %u \n " , first_block_of_bgdt ) ;
2018-10-24 13:38:53 +02:00
# endif
2019-09-30 11:04:30 +02:00
m_cached_group_descriptor_table = ByteBuffer : : create_uninitialized ( block_size ( ) * blocks_to_read ) ;
read_blocks ( first_block_of_bgdt , blocks_to_read , m_cached_group_descriptor_table . data ( ) ) ;
2018-10-10 11:53:07 +02:00
}
2019-09-30 08:57:01 +02:00
return reinterpret_cast < ext2_group_desc * > ( m_cached_group_descriptor_table . data ( ) ) [ group_index - 1 ] ;
2018-10-10 11:53:07 +02:00
}
2018-11-15 17:13:10 +01:00
bool Ext2FS : : initialize ( )
2018-10-10 11:53:07 +02:00
{
2019-11-02 11:49:11 +01:00
bool success = const_cast < DiskDevice & > ( device ( ) ) . read_blocks ( 2 , 1 , ( u8 * ) & m_super_block ) ;
ASSERT ( success ) ;
2019-01-31 17:31:23 +01:00
auto & super_block = this - > super_block ( ) ;
2018-10-24 13:38:53 +02:00
# ifdef EXT2_DEBUG
2019-01-31 17:31:23 +01:00
kprintf ( " ext2fs: super block magic: %x (super block size: %u) \n " , super_block . s_magic , sizeof ( ext2_super_block ) ) ;
2018-10-24 13:38:53 +02:00
# endif
2019-01-31 17:31:23 +01:00
if ( super_block . s_magic ! = EXT2_SUPER_MAGIC )
2018-10-10 11:53:07 +02:00
return false ;
2018-10-24 13:38:53 +02:00
# ifdef EXT2_DEBUG
2019-01-31 17:31:23 +01:00
kprintf ( " ext2fs: %u inodes, %u blocks \n " , super_block . s_inodes_count , super_block . s_blocks_count ) ;
kprintf ( " ext2fs: block size = %u \n " , EXT2_BLOCK_SIZE ( & super_block ) ) ;
kprintf ( " ext2fs: first data block = %u \n " , super_block . s_first_data_block ) ;
2018-12-29 03:36:22 +01:00
kprintf ( " ext2fs: inodes per block = %u \n " , inodes_per_block ( ) ) ;
kprintf ( " ext2fs: inodes per group = %u \n " , inodes_per_group ( ) ) ;
2019-01-31 17:31:23 +01:00
kprintf ( " ext2fs: free inodes = %u \n " , super_block . s_free_inodes_count ) ;
kprintf ( " ext2fs: desc per block = %u \n " , EXT2_DESC_PER_BLOCK ( & super_block ) ) ;
kprintf ( " ext2fs: desc size = %u \n " , EXT2_DESC_SIZE ( & super_block ) ) ;
2018-10-24 13:38:53 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-09-30 11:20:51 +02:00
2019-01-31 17:31:23 +01:00
set_block_size ( EXT2_BLOCK_SIZE ( & super_block ) ) ;
2018-10-10 11:53:07 +02:00
2019-09-30 11:20:51 +02:00
ASSERT ( block_size ( ) < = ( int ) max_block_size ) ;
2019-01-31 17:31:23 +01:00
m_block_group_count = ceil_div ( super_block . s_blocks_count , super_block . s_blocks_per_group ) ;
2018-10-10 11:53:07 +02:00
2019-01-31 17:31:23 +01:00
if ( m_block_group_count = = 0 ) {
2018-10-31 20:10:39 +01:00
kprintf ( " ext2fs: no block groups :( \n " ) ;
2018-10-10 11:53:07 +02:00
return false ;
}
2018-10-31 21:31:56 +01:00
// Preheat the BGD cache.
2018-12-03 00:20:00 +01:00
group_descriptor ( 0 ) ;
2018-10-31 21:31:56 +01:00
2018-10-24 13:38:53 +02:00
# ifdef EXT2_DEBUG
2019-01-31 17:31:23 +01:00
for ( unsigned i = 1 ; i < = m_block_group_count ; + + i ) {
2018-12-29 03:36:22 +01:00
auto & group = group_descriptor ( i ) ;
2018-10-31 20:10:39 +01:00
kprintf ( " ext2fs: group[%u] { block_bitmap: %u, inode_bitmap: %u, inode_table: %u } \n " ,
2018-10-10 11:53:07 +02:00
i ,
group . bg_block_bitmap ,
group . bg_inode_bitmap ,
group . bg_inode_table ) ;
}
2018-10-24 13:38:53 +02:00
# endif
2018-10-10 11:53:07 +02:00
return true ;
}
2018-11-15 17:13:10 +01:00
const char * Ext2FS : : class_name ( ) const
2018-10-10 11:53:07 +02:00
{
2019-01-31 06:13:55 +01:00
return " Ext2FS " ;
2018-10-10 11:53:07 +02:00
}
2018-12-03 00:20:00 +01:00
InodeIdentifier Ext2FS : : root_inode ( ) const
2018-10-10 11:53:07 +02:00
{
2019-01-23 05:38:54 +01:00
return { fsid ( ) , EXT2_ROOT_INO } ;
2018-10-10 11:53:07 +02:00
}
2019-09-30 11:20:51 +02:00
bool Ext2FS : : read_block_containing_inode ( unsigned inode , unsigned & block_index , unsigned & offset , u8 * buffer ) const
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-31 17:31:23 +01:00
auto & super_block = this - > super_block ( ) ;
2018-10-10 11:53:07 +02:00
2019-01-31 17:31:23 +01:00
if ( inode ! = EXT2_ROOT_INO & & inode < EXT2_FIRST_INO ( & super_block ) )
2019-09-30 11:20:51 +02:00
return false ;
2018-10-10 11:53:07 +02:00
2019-01-31 17:31:23 +01:00
if ( inode > super_block . s_inodes_count )
2019-09-30 11:20:51 +02:00
return false ;
2018-10-10 11:53:07 +02:00
2018-12-03 00:20:00 +01:00
auto & bgd = group_descriptor ( group_index_from_inode ( inode ) ) ;
2018-10-10 11:53:07 +02:00
2018-12-03 00:20:00 +01:00
offset = ( ( inode - 1 ) % inodes_per_group ( ) ) * inode_size ( ) ;
2019-01-31 17:31:23 +01:00
block_index = bgd . bg_inode_table + ( offset > > EXT2_BLOCK_SIZE_BITS ( & super_block ) ) ;
offset & = block_size ( ) - 1 ;
2018-10-10 11:53:07 +02:00
2019-09-30 11:20:51 +02:00
return read_block ( block_index , buffer ) ;
2018-10-10 11:53:07 +02:00
}
2019-01-23 02:45:25 +01:00
Ext2FS : : BlockListShape Ext2FS : : compute_block_list_shape ( unsigned blocks )
{
BlockListShape shape ;
const unsigned entries_per_block = EXT2_ADDR_PER_BLOCK ( & super_block ( ) ) ;
unsigned blocks_remaining = blocks ;
2019-09-28 14:31:40 +10:00
2019-01-23 02:45:25 +01:00
shape . direct_blocks = min ( ( unsigned ) EXT2_NDIR_BLOCKS , blocks_remaining ) ;
blocks_remaining - = shape . direct_blocks ;
if ( ! blocks_remaining )
return shape ;
2019-09-28 14:31:40 +10:00
2019-01-23 02:45:25 +01:00
shape . indirect_blocks = min ( blocks_remaining , entries_per_block ) ;
blocks_remaining - = shape . indirect_blocks ;
shape . meta_blocks + = 1 ;
if ( ! blocks_remaining )
return shape ;
2019-09-28 14:31:40 +10:00
2019-01-23 02:45:25 +01:00
shape . doubly_indirect_blocks = min ( blocks_remaining , entries_per_block * entries_per_block ) ;
blocks_remaining - = shape . doubly_indirect_blocks ;
2019-09-28 14:31:40 +10:00
shape . meta_blocks + = 1 ;
shape . meta_blocks + = shape . doubly_indirect_blocks / entries_per_block ;
if ( ( shape . doubly_indirect_blocks % entries_per_block ) ! = 0 )
shape . meta_blocks + = 1 ;
2019-01-23 02:45:25 +01:00
if ( ! blocks_remaining )
return shape ;
2019-09-28 14:31:40 +10:00
dbg ( ) < < " we don't know how to compute tind ext2fs blocks yet! " ;
ASSERT_NOT_REACHED ( ) ;
2019-01-23 02:45:25 +01:00
shape . triply_indirect_blocks = min ( blocks_remaining , entries_per_block * entries_per_block * entries_per_block ) ;
blocks_remaining - = shape . triply_indirect_blocks ;
2019-09-28 14:31:40 +10:00
if ( ! blocks_remaining )
return shape ;
ASSERT_NOT_REACHED ( ) ;
return { } ;
2019-01-23 02:45:25 +01:00
}
2019-01-23 04:29:56 +01:00
bool Ext2FS : : write_block_list_for_inode ( InodeIndex inode_index , ext2_inode & e2inode , const Vector < BlockIndex > & blocks )
2019-01-23 02:45:25 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-03-27 14:24:37 +01:00
// NOTE: There is a mismatch between i_blocks and blocks.size() since i_blocks includes meta blocks and blocks.size() does not.
auto old_block_count = ceil_div ( e2inode . i_size , block_size ( ) ) ;
2019-01-23 02:45:25 +01:00
2019-03-27 14:24:37 +01:00
auto old_shape = compute_block_list_shape ( old_block_count ) ;
2019-01-23 02:45:25 +01:00
auto new_shape = compute_block_list_shape ( blocks . size ( ) ) ;
Vector < BlockIndex > new_meta_blocks ;
if ( new_shape . meta_blocks > old_shape . meta_blocks ) {
new_meta_blocks = allocate_blocks ( group_index_from_inode ( inode_index ) , new_shape . meta_blocks - old_shape . meta_blocks ) ;
}
2019-01-31 17:31:23 +01:00
e2inode . i_blocks = ( blocks . size ( ) + new_shape . meta_blocks ) * ( block_size ( ) / 512 ) ;
2019-01-23 15:43:29 +01:00
2019-03-27 14:24:37 +01:00
bool inode_dirty = false ;
2019-01-23 02:45:25 +01:00
unsigned output_block_index = 0 ;
unsigned remaining_blocks = blocks . size ( ) ;
for ( unsigned i = 0 ; i < new_shape . direct_blocks ; + + i ) {
2019-03-27 14:24:37 +01:00
if ( e2inode . i_block [ i ] ! = blocks [ output_block_index ] )
inode_dirty = true ;
e2inode . i_block [ i ] = blocks [ output_block_index ] ;
+ + output_block_index ;
2019-01-23 02:45:25 +01:00
- - remaining_blocks ;
}
2019-03-27 14:24:37 +01:00
if ( inode_dirty ) {
2019-04-28 22:07:25 +02:00
# ifdef EXT2_DEBUG
2019-09-28 14:31:40 +10:00
dbgprintf ( " Ext2FS: Writing %u direct block(s) to i_block array of inode %u \n " , min ( EXT2_NDIR_BLOCKS , blocks . size ( ) ) , inode_index ) ;
2019-04-28 22:07:25 +02:00
for ( int i = 0 ; i < min ( EXT2_NDIR_BLOCKS , blocks . size ( ) ) ; + + i )
dbgprintf ( " + %u \n " , blocks [ i ] ) ;
# endif
2019-03-27 14:24:37 +01:00
write_ext2_inode ( inode_index , e2inode ) ;
inode_dirty = false ;
}
2019-01-23 02:45:25 +01:00
if ( ! remaining_blocks )
return true ;
2019-09-28 14:31:40 +10:00
const unsigned entries_per_block = EXT2_ADDR_PER_BLOCK ( & super_block ( ) ) ;
bool ind_block_new = ! e2inode . i_block [ EXT2_IND_BLOCK ] ;
if ( ind_block_new ) {
2019-03-27 14:24:37 +01:00
BlockIndex new_indirect_block = new_meta_blocks . take_last ( ) ;
if ( e2inode . i_block [ EXT2_IND_BLOCK ] ! = new_indirect_block )
inode_dirty = true ;
e2inode . i_block [ EXT2_IND_BLOCK ] = new_indirect_block ;
if ( inode_dirty ) {
2019-09-28 14:31:40 +10:00
# ifdef EXT2_DEBUG
2019-03-27 14:24:37 +01:00
dbgprintf ( " Ext2FS: Adding the indirect block to i_block array of inode %u \n " , inode_index ) ;
2019-09-28 14:31:40 +10:00
# endif
2019-03-27 14:24:37 +01:00
write_ext2_inode ( inode_index , e2inode ) ;
inode_dirty = false ;
}
2019-01-23 02:45:25 +01:00
}
2019-03-27 14:24:37 +01:00
if ( old_shape . indirect_blocks = = new_shape . indirect_blocks ) {
// No need to update the singly indirect block array.
remaining_blocks - = new_shape . indirect_blocks ;
2019-09-28 14:31:40 +10:00
output_block_index + = new_shape . indirect_blocks ;
2019-03-27 14:24:37 +01:00
} else {
2019-01-31 17:31:23 +01:00
auto block_contents = ByteBuffer : : create_uninitialized ( block_size ( ) ) ;
2019-01-23 02:45:25 +01:00
BufferStream stream ( block_contents ) ;
2019-09-28 14:31:40 +10:00
ASSERT ( new_shape . indirect_blocks < = entries_per_block ) ;
2019-01-23 02:45:25 +01:00
for ( unsigned i = 0 ; i < new_shape . indirect_blocks ; + + i ) {
stream < < blocks [ output_block_index + + ] ;
- - remaining_blocks ;
}
stream . fill_to_end ( 0 ) ;
2019-09-30 11:20:51 +02:00
bool success = write_block ( e2inode . i_block [ EXT2_IND_BLOCK ] , block_contents . data ( ) ) ;
2019-02-11 11:38:14 +01:00
ASSERT ( success ) ;
2019-01-23 02:45:25 +01:00
}
2019-09-28 14:31:40 +10:00
if ( ! remaining_blocks )
return true ;
bool dind_block_dirty = false ;
bool dind_block_new = ! e2inode . i_block [ EXT2_DIND_BLOCK ] ;
if ( dind_block_new ) {
BlockIndex new_dindirect_block = new_meta_blocks . take_last ( ) ;
if ( e2inode . i_block [ EXT2_DIND_BLOCK ] ! = new_dindirect_block )
inode_dirty = true ;
e2inode . i_block [ EXT2_DIND_BLOCK ] = new_dindirect_block ;
if ( inode_dirty ) {
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FS: Adding the doubly-indirect block to i_block array of inode %u \n " , inode_index ) ;
# endif
write_ext2_inode ( inode_index , e2inode ) ;
inode_dirty = false ;
}
}
if ( old_shape . doubly_indirect_blocks = = new_shape . doubly_indirect_blocks ) {
// No need to update the doubly indirect block data.
remaining_blocks - = new_shape . doubly_indirect_blocks ;
output_block_index + = new_shape . doubly_indirect_blocks ;
} else {
unsigned indirect_block_count = new_shape . doubly_indirect_blocks / entries_per_block ;
if ( ( new_shape . doubly_indirect_blocks % entries_per_block ) ! = 0 )
indirect_block_count + + ;
2019-09-30 11:04:30 +02:00
auto dind_block_contents = ByteBuffer : : create_uninitialized ( block_size ( ) ) ;
read_block ( e2inode . i_block [ EXT2_DIND_BLOCK ] , dind_block_contents . data ( ) ) ;
2019-09-28 14:31:40 +10:00
if ( dind_block_new ) {
2019-09-30 08:57:01 +02:00
memset ( dind_block_contents . data ( ) , 0 , dind_block_contents . size ( ) ) ;
2019-09-28 14:31:40 +10:00
dind_block_dirty = true ;
}
2019-09-30 08:57:01 +02:00
auto * dind_block_as_pointers = ( unsigned * ) dind_block_contents . data ( ) ;
2019-09-28 14:31:40 +10:00
ASSERT ( indirect_block_count < = entries_per_block ) ;
for ( unsigned i = 0 ; i < indirect_block_count ; + + i ) {
bool ind_block_dirty = false ;
BlockIndex indirect_block_index = dind_block_as_pointers [ i ] ;
bool ind_block_new = ! indirect_block_index ;
if ( ind_block_new ) {
indirect_block_index = new_meta_blocks . take_last ( ) ;
dind_block_as_pointers [ i ] = indirect_block_index ;
dind_block_dirty = true ;
}
2019-09-30 11:04:30 +02:00
auto ind_block_contents = ByteBuffer : : create_uninitialized ( block_size ( ) ) ;
read_block ( indirect_block_index , ind_block_contents . data ( ) ) ;
2019-09-28 14:31:40 +10:00
if ( ind_block_new ) {
2019-09-30 08:57:01 +02:00
memset ( ind_block_contents . data ( ) , 0 , dind_block_contents . size ( ) ) ;
2019-09-28 14:31:40 +10:00
ind_block_dirty = true ;
}
2019-09-30 08:57:01 +02:00
auto * ind_block_as_pointers = ( unsigned * ) ind_block_contents . data ( ) ;
2019-09-28 14:31:40 +10:00
unsigned entries_to_write = new_shape . doubly_indirect_blocks - ( i * entries_per_block ) ;
if ( entries_to_write > entries_per_block )
entries_to_write = entries_per_block ;
ASSERT ( entries_to_write < = entries_per_block ) ;
for ( unsigned j = 0 ; j < entries_to_write ; + + j ) {
BlockIndex output_block = blocks [ output_block_index + + ] ;
if ( ind_block_as_pointers [ j ] ! = output_block ) {
ind_block_as_pointers [ j ] = output_block ;
ind_block_dirty = true ;
}
- - remaining_blocks ;
}
for ( unsigned j = entries_to_write ; j < entries_per_block ; + + j ) {
if ( ind_block_as_pointers [ j ] ! = 0 ) {
ind_block_as_pointers [ j ] = 0 ;
ind_block_dirty = true ;
}
}
if ( ind_block_dirty ) {
2019-09-30 11:20:51 +02:00
bool success = write_block ( indirect_block_index , ind_block_contents . data ( ) ) ;
2019-09-28 14:31:40 +10:00
ASSERT ( success ) ;
}
}
for ( unsigned i = indirect_block_count ; i < entries_per_block ; + + i ) {
if ( dind_block_as_pointers [ i ] ! = 0 ) {
dind_block_as_pointers [ i ] = 0 ;
dind_block_dirty = true ;
}
}
if ( dind_block_dirty ) {
2019-09-30 11:20:51 +02:00
bool success = write_block ( e2inode . i_block [ EXT2_DIND_BLOCK ] , dind_block_contents . data ( ) ) ;
2019-09-28 14:31:40 +10:00
ASSERT ( success ) ;
}
}
2019-01-23 02:45:25 +01:00
if ( ! remaining_blocks )
return true ;
// FIXME: Implement!
2019-09-28 14:31:40 +10:00
dbg ( ) < < " we don't know how to write tind ext2fs blocks yet! " ;
2019-01-23 02:45:25 +01:00
ASSERT_NOT_REACHED ( ) ;
}
2019-05-25 17:23:17 +02:00
Vector < Ext2FS : : BlockIndex > Ext2FS : : block_list_for_inode ( const ext2_inode & e2inode , bool include_block_list_blocks ) const
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-31 17:31:23 +01:00
unsigned entries_per_block = EXT2_ADDR_PER_BLOCK ( & super_block ( ) ) ;
2018-10-10 11:53:07 +02:00
// NOTE: i_blocks is number of 512-byte blocks, not number of fs-blocks.
2019-01-31 17:31:23 +01:00
unsigned block_count = e2inode . i_blocks / ( block_size ( ) / 512 ) ;
2019-05-25 19:19:43 +02:00
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FS::block_list_for_inode(): i_size=%u, i_blocks=%u, block_count=%u \n " , e2inode . i_size , block_count ) ;
# endif
2019-05-25 17:23:17 +02:00
unsigned blocks_remaining = block_count ;
Vector < BlockIndex > list ;
2019-01-22 16:34:24 +01:00
if ( include_block_list_blocks ) {
// This seems like an excessive over-estimate but w/e.
2019-05-25 17:23:17 +02:00
list . ensure_capacity ( blocks_remaining * 2 ) ;
2019-01-22 16:34:24 +01:00
} else {
2019-05-25 17:23:17 +02:00
list . ensure_capacity ( blocks_remaining ) ;
2019-01-22 16:34:24 +01:00
}
2018-10-10 11:53:07 +02:00
2019-01-31 17:31:23 +01:00
unsigned direct_count = min ( block_count , ( unsigned ) EXT2_NDIR_BLOCKS ) ;
for ( unsigned i = 0 ; i < direct_count ; + + i ) {
2019-05-25 19:19:43 +02:00
auto block_index = e2inode . i_block [ i ] ;
if ( ! block_index )
return list ;
list . unchecked_append ( block_index ) ;
2019-05-25 17:23:17 +02:00
- - blocks_remaining ;
2018-10-10 11:53:07 +02:00
}
2019-05-25 17:23:17 +02:00
if ( ! blocks_remaining )
2018-10-10 11:53:07 +02:00
return list ;
2019-06-07 11:43:58 +02:00
auto process_block_array = [ & ] ( unsigned array_block_index , auto & & callback ) {
2019-01-22 16:34:24 +01:00
if ( include_block_list_blocks )
2019-01-31 17:31:23 +01:00
callback ( array_block_index ) ;
2019-09-30 11:04:30 +02:00
auto array_block = ByteBuffer : : create_uninitialized ( block_size ( ) ) ;
read_block ( array_block_index , array_block . data ( ) ) ;
2019-01-31 17:31:23 +01:00
ASSERT ( array_block ) ;
2019-09-30 08:57:01 +02:00
auto * array = reinterpret_cast < const __u32 * > ( array_block . data ( ) ) ;
2019-05-25 17:23:17 +02:00
unsigned count = min ( blocks_remaining , entries_per_block ) ;
2018-10-10 11:53:07 +02:00
for ( unsigned i = 0 ; i < count ; + + i ) {
if ( ! array [ i ] ) {
2019-05-25 17:23:17 +02:00
blocks_remaining = 0 ;
2018-10-10 11:53:07 +02:00
return ;
}
callback ( array [ i ] ) ;
2019-05-25 17:23:17 +02:00
- - blocks_remaining ;
2018-10-10 11:53:07 +02:00
}
} ;
2019-06-07 11:43:58 +02:00
process_block_array ( e2inode . i_block [ EXT2_IND_BLOCK ] , [ & ] ( unsigned entry ) {
2018-11-13 00:17:30 +01:00
list . unchecked_append ( entry ) ;
2018-10-10 11:53:07 +02:00
} ) ;
2019-05-25 17:23:17 +02:00
if ( ! blocks_remaining )
2018-10-10 11:53:07 +02:00
return list ;
2019-06-07 11:43:58 +02:00
process_block_array ( e2inode . i_block [ EXT2_DIND_BLOCK ] , [ & ] ( unsigned entry ) {
process_block_array ( entry , [ & ] ( unsigned entry ) {
2018-11-13 00:17:30 +01:00
list . unchecked_append ( entry ) ;
2018-10-10 11:53:07 +02:00
} ) ;
} ) ;
2019-05-25 17:23:17 +02:00
if ( ! blocks_remaining )
2018-10-10 11:53:07 +02:00
return list ;
2019-06-07 11:43:58 +02:00
process_block_array ( e2inode . i_block [ EXT2_TIND_BLOCK ] , [ & ] ( unsigned entry ) {
process_block_array ( entry , [ & ] ( unsigned entry ) {
process_block_array ( entry , [ & ] ( unsigned entry ) {
2018-11-13 00:17:30 +01:00
list . unchecked_append ( entry ) ;
2018-10-10 11:53:07 +02:00
} ) ;
} ) ;
} ) ;
return list ;
}
2019-01-22 20:48:48 +01:00
void Ext2FS : : free_inode ( Ext2FSInode & inode )
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-22 20:48:48 +01:00
ASSERT ( inode . m_raw_inode . i_links_count = = 0 ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-01-22 20:48:48 +01:00
dbgprintf ( " Ext2FS: inode %u has no more links, time to delete! \n " , inode . index ( ) ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-01-22 20:48:48 +01:00
2019-03-25 02:06:57 +01:00
struct timeval now ;
kgettimeofday ( now ) ;
inode . m_raw_inode . i_dtime = now . tv_sec ;
2019-01-22 20:48:48 +01:00
write_ext2_inode ( inode . index ( ) , inode . m_raw_inode ) ;
auto block_list = block_list_for_inode ( inode . m_raw_inode , true ) ;
for ( auto block_index : block_list )
2019-02-15 23:24:01 +01:00
set_block_allocation_state ( block_index , false ) ;
2019-01-22 20:48:48 +01:00
set_inode_allocation_state ( inode . index ( ) , false ) ;
2019-01-28 04:16:01 +01:00
if ( inode . is_directory ( ) ) {
auto & bgd = const_cast < ext2_group_desc & > ( group_descriptor ( group_index_from_inode ( inode . index ( ) ) ) ) ;
- - bgd . bg_used_dirs_count ;
dbgprintf ( " Ext2FS: decremented bg_used_dirs_count %u -> %u \n " , bgd . bg_used_dirs_count - 1 , bgd . bg_used_dirs_count ) ;
2019-11-02 11:36:56 +01:00
m_block_group_descriptors_dirty = true ;
2019-01-28 04:16:01 +01:00
}
}
void Ext2FS : : flush_block_group_descriptor_table ( )
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-31 17:31:23 +01:00
unsigned blocks_to_write = ceil_div ( m_block_group_count * ( unsigned ) sizeof ( ext2_group_desc ) , block_size ( ) ) ;
unsigned first_block_of_bgdt = block_size ( ) = = 1024 ? 2 : 1 ;
2019-09-30 11:20:51 +02:00
write_blocks ( first_block_of_bgdt , blocks_to_write , m_cached_group_descriptor_table . data ( ) ) ;
2019-01-22 20:48:48 +01:00
}
2019-11-02 11:36:56 +01:00
void Ext2FS : : flush_writes ( )
{
if ( m_super_block_dirty ) {
2019-11-02 11:49:11 +01:00
flush_super_block ( ) ;
2019-11-02 11:36:56 +01:00
m_super_block_dirty = false ;
}
if ( m_block_group_descriptors_dirty ) {
flush_block_group_descriptor_table ( ) ;
m_block_group_descriptors_dirty = false ;
}
2019-11-02 12:18:43 +01:00
for ( auto & cached_bitmap : m_cached_bitmaps ) {
if ( cached_bitmap - > dirty ) {
write_block ( cached_bitmap - > bitmap_block_index , cached_bitmap - > buffer . data ( ) ) ;
cached_bitmap - > dirty = false ;
# ifdef EXT2_DEBUG
dbg ( ) < < " Flushed bitmap block " < < cached_bitmap - > bitmap_block_index ;
# endif
}
}
2019-11-02 11:36:56 +01:00
DiskBackedFS : : flush_writes ( ) ;
}
2019-02-03 04:05:30 +01:00
Ext2FSInode : : Ext2FSInode ( Ext2FS & fs , unsigned index )
2018-12-19 21:18:28 +01:00
: Inode ( fs , index )
2018-11-13 13:02:39 +01:00
{
}
2018-11-15 17:13:10 +01:00
Ext2FSInode : : ~ Ext2FSInode ( )
2018-11-13 13:02:39 +01:00
{
2019-01-22 20:48:48 +01:00
if ( m_raw_inode . i_links_count = = 0 )
fs ( ) . free_inode ( * this ) ;
2018-11-13 13:02:39 +01:00
}
2019-01-01 03:16:36 +01:00
InodeMetadata Ext2FSInode : : metadata ( ) const
{
2019-02-20 13:09:59 +01:00
// FIXME: This should probably take the inode lock, no?
2019-01-01 03:16:36 +01:00
InodeMetadata metadata ;
metadata . inode = identifier ( ) ;
metadata . size = m_raw_inode . i_size ;
metadata . mode = m_raw_inode . i_mode ;
metadata . uid = m_raw_inode . i_uid ;
metadata . gid = m_raw_inode . i_gid ;
2019-01-31 17:31:23 +01:00
metadata . link_count = m_raw_inode . i_links_count ;
2019-01-01 03:16:36 +01:00
metadata . atime = m_raw_inode . i_atime ;
metadata . ctime = m_raw_inode . i_ctime ;
metadata . mtime = m_raw_inode . i_mtime ;
metadata . dtime = m_raw_inode . i_dtime ;
2019-01-31 17:31:23 +01:00
metadata . block_size = fs ( ) . block_size ( ) ;
metadata . block_count = m_raw_inode . i_blocks ;
2018-11-13 13:32:16 +01:00
2019-07-31 17:24:54 +02:00
if ( : : is_character_device ( m_raw_inode . i_mode ) | | : : is_block_device ( m_raw_inode . i_mode ) ) {
2018-11-13 13:32:16 +01:00
unsigned dev = m_raw_inode . i_block [ 0 ] ;
2019-07-31 17:24:54 +02:00
if ( ! dev )
dev = m_raw_inode . i_block [ 1 ] ;
2019-02-16 09:57:42 +01:00
metadata . major_device = ( dev & 0xfff00 ) > > 8 ;
metadata . minor_device = ( dev & 0xff ) | ( ( dev > > 12 ) & 0xfff00 ) ;
2018-11-13 13:32:16 +01:00
}
2019-01-01 03:16:36 +01:00
return metadata ;
2018-11-13 13:32:16 +01:00
}
2018-12-19 21:56:45 +01:00
void Ext2FSInode : : flush_metadata ( )
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2018-12-20 00:39:29 +01:00
dbgprintf ( " Ext2FSInode: flush_metadata for inode %u \n " , index ( ) ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-12-19 21:56:45 +01:00
fs ( ) . write_ext2_inode ( index ( ) , m_raw_inode ) ;
2019-01-01 03:55:13 +01:00
if ( is_directory ( ) ) {
2019-01-28 04:16:01 +01:00
// Unless we're about to go away permanently, invalidate the lookup cache.
if ( m_raw_inode . i_links_count ! = 0 ) {
// FIXME: This invalidation is way too hardcore. It's sad to throw away the whole cache.
m_lookup_cache . clear ( ) ;
}
2019-01-01 03:55:13 +01:00
}
2018-12-19 22:28:09 +01:00
set_metadata_dirty ( false ) ;
2018-12-19 21:56:45 +01:00
}
2019-06-21 18:37:47 +02:00
RefPtr < Inode > Ext2FS : : get_inode ( InodeIdentifier inode ) const
2018-11-13 13:02:39 +01:00
{
2019-02-20 21:41:53 +01:00
LOCKER ( m_lock ) ;
2019-01-23 05:38:54 +01:00
ASSERT ( inode . fsid ( ) = = fsid ( ) ) ;
2019-02-20 21:41:53 +01:00
2018-11-13 13:02:39 +01:00
{
auto it = m_inode_cache . find ( inode . index ( ) ) ;
if ( it ! = m_inode_cache . end ( ) )
return ( * it ) . value ;
}
2018-12-25 00:32:57 +01:00
2019-01-01 03:35:33 +01:00
if ( ! get_inode_allocation_state ( inode . index ( ) ) ) {
m_inode_cache . set ( inode . index ( ) , nullptr ) ;
return nullptr ;
}
2018-12-25 00:32:57 +01:00
unsigned block_index ;
unsigned offset ;
2019-09-30 11:20:51 +02:00
u8 block [ max_block_size ] ;
if ( ! read_block_containing_inode ( inode . index ( ) , block_index , offset , block ) )
2019-06-07 11:43:58 +02:00
return { } ;
2018-12-25 00:32:57 +01:00
2018-11-13 13:02:39 +01:00
auto it = m_inode_cache . find ( inode . index ( ) ) ;
if ( it ! = m_inode_cache . end ( ) )
return ( * it ) . value ;
2019-02-03 04:05:30 +01:00
auto new_inode = adopt ( * new Ext2FSInode ( const_cast < Ext2FS & > ( * this ) , inode . index ( ) ) ) ;
2019-09-30 11:20:51 +02:00
memcpy ( & new_inode - > m_raw_inode , reinterpret_cast < ext2_inode * > ( block + offset ) , sizeof ( ext2_inode ) ) ;
2019-07-11 15:38:47 +02:00
m_inode_cache . set ( inode . index ( ) , new_inode ) ;
2018-11-13 13:02:39 +01:00
return new_inode ;
}
2019-07-03 21:17:35 +02:00
ssize_t Ext2FSInode : : read_bytes ( off_t offset , ssize_t count , u8 * buffer , FileDescription * ) const
2018-11-13 13:02:39 +01:00
{
2019-02-20 13:09:59 +01:00
Locker inode_locker ( m_lock ) ;
2018-11-13 13:02:39 +01:00
ASSERT ( offset > = 0 ) ;
if ( m_raw_inode . i_size = = 0 )
return 0 ;
// Symbolic links shorter than 60 characters are store inline inside the i_block array.
// This avoids wasting an entire block on short links. (Most links are short.)
if ( is_symlink ( ) & & size ( ) < max_inline_symlink_length ) {
2019-01-23 06:53:01 +01:00
ssize_t nread = min ( ( off_t ) size ( ) - offset , static_cast < off_t > ( count ) ) ;
2019-07-03 21:17:35 +02:00
memcpy ( buffer , ( ( const u8 * ) m_raw_inode . i_block ) + offset , ( size_t ) nread ) ;
2018-11-13 13:02:39 +01:00
return nread ;
}
2019-02-20 13:09:59 +01:00
Locker fs_locker ( fs ( ) . m_lock ) ;
2018-12-21 02:10:45 +01:00
if ( m_block_list . is_empty ( ) ) {
2018-12-03 00:20:00 +01:00
auto block_list = fs ( ) . block_list_for_inode ( m_raw_inode ) ;
2018-11-13 13:02:39 +01:00
if ( m_block_list . size ( ) ! = block_list . size ( ) )
m_block_list = move ( block_list ) ;
}
2018-12-21 02:10:45 +01:00
if ( m_block_list . is_empty ( ) ) {
2018-11-13 13:02:39 +01:00
kprintf ( " ext2fs: read_bytes: empty block list for inode %u \n " , index ( ) ) ;
return - EIO ;
}
2019-04-23 13:00:53 +02:00
const int block_size = fs ( ) . block_size ( ) ;
2018-11-13 13:02:39 +01:00
2019-04-23 13:00:53 +02:00
int first_block_logical_index = offset / block_size ;
int last_block_logical_index = ( offset + count ) / block_size ;
2018-11-13 13:02:39 +01:00
if ( last_block_logical_index > = m_block_list . size ( ) )
last_block_logical_index = m_block_list . size ( ) - 1 ;
2019-04-23 13:00:53 +02:00
int offset_into_first_block = offset % block_size ;
2018-11-13 13:02:39 +01:00
2018-12-02 23:34:50 +01:00
ssize_t nread = 0 ;
2019-04-23 13:00:53 +02:00
int remaining_count = min ( ( off_t ) count , ( off_t ) size ( ) - offset ) ;
2019-07-03 21:17:35 +02:00
u8 * out = buffer ;
2018-11-13 13:02:39 +01:00
# ifdef EXT2_DEBUG
2019-01-23 04:59:47 +01:00
kprintf ( " Ext2FS: Reading up to %u bytes %d bytes into inode %u:%u to %p \n " , count , offset , identifier ( ) . fsid ( ) , identifier ( ) . index ( ) , buffer ) ;
2018-12-29 03:36:22 +01:00
//kprintf("ok let's do it, read(%u, %u) -> blocks %u thru %u, oifb: %u\n", offset, count, first_block_logical_index, last_block_logical_index, offset_into_first_block);
2018-11-13 13:02:39 +01:00
# endif
2019-09-30 11:20:51 +02:00
u8 block [ max_block_size ] ;
2019-09-30 11:04:30 +02:00
2019-04-23 13:00:53 +02:00
for ( int bi = first_block_logical_index ; remaining_count & & bi < = last_block_logical_index ; + + bi ) {
2019-09-30 11:04:30 +02:00
bool success = fs ( ) . read_block ( m_block_list [ bi ] , block ) ;
if ( ! success ) {
2019-01-31 17:31:23 +01:00
kprintf ( " ext2fs: read_bytes: read_block(%u) failed (lbi: %u) \n " , m_block_list [ bi ] , bi ) ;
2018-11-13 13:02:39 +01:00
return - EIO ;
}
2019-04-23 13:00:53 +02:00
int offset_into_block = ( bi = = first_block_logical_index ) ? offset_into_first_block : 0 ;
int num_bytes_to_copy = min ( block_size - offset_into_block , remaining_count ) ;
2019-09-30 11:04:30 +02:00
memcpy ( out , block + offset_into_block , num_bytes_to_copy ) ;
2018-11-13 13:02:39 +01:00
remaining_count - = num_bytes_to_copy ;
nread + = num_bytes_to_copy ;
out + = num_bytes_to_copy ;
}
return nread ;
}
2019-11-02 12:53:31 +01:00
KResult Ext2FSInode : : resize ( u64 new_size )
2019-04-28 22:07:25 +02:00
{
2019-07-03 21:17:35 +02:00
u64 block_size = fs ( ) . block_size ( ) ;
u64 old_size = size ( ) ;
2019-04-28 22:07:25 +02:00
int blocks_needed_before = ceil_div ( old_size , block_size ) ;
int blocks_needed_after = ceil_div ( new_size , block_size ) ;
2019-04-28 22:14:37 +02:00
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FSInode::resize(): blocks needed before (size was %Q): %d \n " , old_size , blocks_needed_before ) ;
dbgprintf ( " Ext2FSInode::resize(): blocks needed after (size is %Q): %d \n " , new_size , blocks_needed_after ) ;
# endif
2019-04-28 22:07:25 +02:00
2019-11-02 12:53:31 +01:00
if ( blocks_needed_after > blocks_needed_before ) {
u32 additional_blocks_needed = blocks_needed_after - blocks_needed_before ;
if ( additional_blocks_needed > fs ( ) . super_block ( ) . s_free_blocks_count )
return KResult ( - ENOSPC ) ;
}
2019-04-28 22:07:25 +02:00
auto block_list = fs ( ) . block_list_for_inode ( m_raw_inode ) ;
if ( blocks_needed_after > blocks_needed_before ) {
auto new_blocks = fs ( ) . allocate_blocks ( fs ( ) . group_index_from_inode ( index ( ) ) , blocks_needed_after - blocks_needed_before ) ;
block_list . append ( move ( new_blocks ) ) ;
} else if ( blocks_needed_after < blocks_needed_before ) {
2019-05-25 17:23:17 +02:00
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FSInode::resize(): Shrinking. Old block list is %d entries: \n " , block_list . size ( ) ) ;
for ( auto block_index : block_list ) {
dbgprintf ( " # %u \n " , block_index ) ;
}
# endif
2019-04-28 22:07:25 +02:00
while ( block_list . size ( ) ! = blocks_needed_after ) {
auto block_index = block_list . take_last ( ) ;
fs ( ) . set_block_allocation_state ( block_index , false ) ;
}
}
bool success = fs ( ) . write_block_list_for_inode ( index ( ) , m_raw_inode , block_list ) ;
if ( ! success )
2019-11-02 12:53:31 +01:00
return KResult ( - EIO ) ;
2019-04-28 22:07:25 +02:00
m_raw_inode . i_size = new_size ;
set_metadata_dirty ( true ) ;
m_block_list = move ( block_list ) ;
2019-11-02 12:53:31 +01:00
return KSuccess ;
2019-04-28 22:07:25 +02:00
}
2019-07-03 21:17:35 +02:00
ssize_t Ext2FSInode : : write_bytes ( off_t offset , ssize_t count , const u8 * data , FileDescription * )
2019-01-23 04:29:56 +01:00
{
2019-03-02 01:50:34 +01:00
ASSERT ( offset > = 0 ) ;
ASSERT ( count > = 0 ) ;
2019-02-20 13:09:59 +01:00
Locker inode_locker ( m_lock ) ;
Locker fs_locker ( fs ( ) . m_lock ) ;
2019-01-23 04:29:56 +01:00
2019-03-02 01:50:34 +01:00
if ( is_symlink ( ) ) {
if ( ( offset + count ) < max_inline_symlink_length ) {
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FSInode: write_bytes poking into i_block array for inline symlink '%s' (%u bytes) \n " , String ( ( const char * ) data , count ) . characters ( ) , count ) ;
# endif
2019-07-03 21:17:35 +02:00
memcpy ( ( ( u8 * ) m_raw_inode . i_block ) + offset , data , ( size_t ) count ) ;
2019-04-23 13:00:53 +02:00
if ( ( offset + count ) > ( off_t ) m_raw_inode . i_size )
2019-03-02 01:50:34 +01:00
m_raw_inode . i_size = offset + count ;
set_metadata_dirty ( true ) ;
return count ;
}
}
2019-01-23 04:29:56 +01:00
2019-02-25 22:06:55 +01:00
const ssize_t block_size = fs ( ) . block_size ( ) ;
2019-07-03 21:17:35 +02:00
u64 old_size = size ( ) ;
u64 new_size = max ( static_cast < u64 > ( offset ) + count , ( u64 ) size ( ) ) ;
2019-01-23 04:29:56 +01:00
2019-11-02 12:53:31 +01:00
auto resize_result = resize ( new_size ) ;
if ( resize_result . is_error ( ) )
return resize_result ;
2019-01-23 04:29:56 +01:00
2019-04-23 13:00:53 +02:00
int first_block_logical_index = offset / block_size ;
int last_block_logical_index = ( offset + count ) / block_size ;
2019-04-28 22:07:25 +02:00
if ( last_block_logical_index > = m_block_list . size ( ) )
last_block_logical_index = m_block_list . size ( ) - 1 ;
2019-01-23 04:29:56 +01:00
2019-04-23 13:00:53 +02:00
int offset_into_first_block = offset % block_size ;
2019-01-23 04:29:56 +01:00
2019-04-27 17:14:27 +02:00
int last_logical_block_index_in_file = new_size / block_size ;
2019-03-02 01:50:34 +01:00
2019-01-23 04:29:56 +01:00
ssize_t nwritten = 0 ;
2019-04-23 13:00:53 +02:00
int remaining_count = min ( ( off_t ) count , ( off_t ) new_size - offset ) ;
2019-07-03 21:17:35 +02:00
const u8 * in = data ;
2019-01-23 04:29:56 +01:00
# ifdef EXT2_DEBUG
2019-01-23 04:59:47 +01:00
dbgprintf ( " Ext2FSInode::write_bytes: Writing %u bytes %d bytes into inode %u:%u from %p \n " , count , offset , fsid ( ) , index ( ) , data ) ;
2019-01-23 04:29:56 +01:00
# endif
auto buffer_block = ByteBuffer : : create_uninitialized ( block_size ) ;
2019-04-23 13:00:53 +02:00
for ( int bi = first_block_logical_index ; remaining_count & & bi < = last_block_logical_index ; + + bi ) {
int offset_into_block = ( bi = = first_block_logical_index ) ? offset_into_first_block : 0 ;
int num_bytes_to_copy = min ( block_size - offset_into_block , remaining_count ) ;
2019-01-23 04:29:56 +01:00
ByteBuffer block ;
2019-03-02 01:50:34 +01:00
if ( offset_into_block ! = 0 | | num_bytes_to_copy ! = block_size ) {
2019-09-30 11:04:30 +02:00
block = ByteBuffer : : create_uninitialized ( block_size ) ;
bool success = fs ( ) . read_block ( m_block_list [ bi ] , block . data ( ) ) ;
if ( ! success ) {
2019-04-28 22:07:25 +02:00
kprintf ( " Ext2FSInode::write_bytes: read_block(%u) failed (lbi: %u) \n " , m_block_list [ bi ] , bi ) ;
2019-01-23 04:29:56 +01:00
return - EIO ;
}
} else
block = buffer_block ;
2019-09-30 08:57:01 +02:00
memcpy ( block . data ( ) + offset_into_block , in , num_bytes_to_copy ) ;
2019-03-02 01:50:34 +01:00
if ( bi = = last_logical_block_index_in_file & & num_bytes_to_copy < block_size ) {
2019-04-23 13:00:53 +02:00
int padding_start = new_size % block_size ;
int padding_bytes = block_size - padding_start ;
2019-03-02 01:50:34 +01:00
# ifdef EXT2_DEBUG
dbgprintf ( " Ext2FSInode::write_bytes padding last block of file with zero x %u (new_size=%u, offset_into_block=%u, num_bytes_to_copy=%u) \n " , padding_bytes , new_size , offset_into_block , num_bytes_to_copy ) ;
# endif
2019-09-30 08:57:01 +02:00
memset ( block . data ( ) + padding_start , 0 , padding_bytes ) ;
2019-03-02 01:50:34 +01:00
}
2019-01-23 04:29:56 +01:00
# ifdef EXT2_DEBUG
2019-05-25 17:23:17 +02:00
dbgprintf ( " Ext2FSInode::write_bytes: writing block %u (offset_into_block: %u) \n " , m_block_list [ bi ] , offset_into_block ) ;
2019-01-23 04:29:56 +01:00
# endif
2019-09-30 11:20:51 +02:00
bool success = fs ( ) . write_block ( m_block_list [ bi ] , block . data ( ) ) ;
2019-01-23 04:29:56 +01:00
if ( ! success ) {
2019-04-28 22:07:25 +02:00
kprintf ( " Ext2FSInode::write_bytes: write_block(%u) failed (lbi: %u) \n " , m_block_list [ bi ] , bi ) ;
2019-02-11 11:38:14 +01:00
ASSERT_NOT_REACHED ( ) ;
2019-01-23 04:29:56 +01:00
return - EIO ;
}
remaining_count - = num_bytes_to_copy ;
nwritten + = num_bytes_to_copy ;
in + = num_bytes_to_copy ;
}
# ifdef EXT2_DEBUG
2019-05-25 17:23:17 +02:00
dbgprintf ( " Ext2FSInode::write_bytes: after write, i_size=%u, i_blocks=%u (%u blocks in list) \n " , m_raw_inode . i_size , m_raw_inode . i_blocks , m_block_list . size ( ) ) ;
2019-01-23 04:29:56 +01:00
# endif
2019-02-05 08:17:46 +01:00
if ( old_size ! = new_size )
inode_size_changed ( old_size , new_size ) ;
inode_contents_changed ( offset , count , data ) ;
2019-01-23 04:29:56 +01:00
return nwritten ;
}
2019-01-28 04:16:01 +01:00
bool Ext2FSInode : : traverse_as_directory ( Function < bool ( const FS : : DirectoryEntry & ) > callback ) const
2018-11-13 23:44:54 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-10-02 13:37:44 +02:00
ASSERT ( is_directory ( ) ) ;
2018-11-13 23:44:54 +01:00
# ifdef EXT2_DEBUG
kprintf ( " Ext2Inode::traverse_as_directory: inode=%u: \n " , index ( ) ) ;
# endif
auto buffer = read_entire ( ) ;
ASSERT ( buffer ) ;
2019-09-30 08:57:01 +02:00
auto * entry = reinterpret_cast < ext2_dir_entry_2 * > ( buffer . data ( ) ) ;
2018-11-13 23:44:54 +01:00
2018-12-21 02:10:45 +01:00
while ( entry < buffer . end_pointer ( ) ) {
2018-11-13 23:44:54 +01:00
if ( entry - > inode ! = 0 ) {
# ifdef EXT2_DEBUG
2019-01-23 04:59:47 +01:00
kprintf ( " Ext2Inode::traverse_as_directory: %u, name_len: %u, rec_len: %u, file_type: %u, name: %s \n " , entry - > inode , entry - > name_len , entry - > rec_len , entry - > file_type , String ( entry - > name , entry - > name_len ) . characters ( ) ) ;
2018-11-13 23:44:54 +01:00
# endif
if ( ! callback ( { entry - > name , entry - > name_len , { fsid ( ) , entry - > inode } , entry - > file_type } ) )
break ;
}
entry = ( ext2_dir_entry_2 * ) ( ( char * ) entry + entry - > rec_len ) ;
}
return true ;
}
2019-06-09 12:46:23 +02:00
bool Ext2FSInode : : write_directory ( const Vector < FS : : DirectoryEntry > & entries )
{
LOCKER ( m_lock ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-06-09 12:46:23 +02:00
dbgprintf ( " Ext2FS: New directory inode %u contents to write: \n " , index ( ) ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-06-09 12:46:23 +02:00
int directory_size = 0 ;
for ( auto & entry : entries ) {
//kprintf(" - %08u %s\n", entry.inode.index(), entry.name);
directory_size + = EXT2_DIR_REC_LEN ( entry . name_length ) ;
}
auto block_size = fs ( ) . block_size ( ) ;
int blocks_needed = ceil_div ( directory_size , block_size ) ;
int occupied_size = blocks_needed * block_size ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-06-09 12:46:23 +02:00
dbgprintf ( " Ext2FS: directory size: %u (occupied: %u) \n " , directory_size , occupied_size ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-06-09 12:46:23 +02:00
auto directory_data = ByteBuffer : : create_uninitialized ( occupied_size ) ;
BufferStream stream ( directory_data ) ;
for ( int i = 0 ; i < entries . size ( ) ; + + i ) {
auto & entry = entries [ i ] ;
int record_length = EXT2_DIR_REC_LEN ( entry . name_length ) ;
if ( i = = entries . size ( ) - 1 )
record_length + = occupied_size - directory_size ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-06-09 12:46:23 +02:00
dbgprintf ( " * inode: %u " , entry . inode . index ( ) ) ;
2019-07-03 21:17:35 +02:00
dbgprintf ( " , name_len: %u " , u16 ( entry . name_length ) ) ;
dbgprintf ( " , rec_len: %u " , u16 ( record_length ) ) ;
dbgprintf ( " , file_type: %u " , u8 ( entry . file_type ) ) ;
2019-06-09 12:46:23 +02:00
dbgprintf ( " , name: %s \n " , entry . name ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-06-09 12:46:23 +02:00
2019-07-03 21:17:35 +02:00
stream < < u32 ( entry . inode . index ( ) ) ;
stream < < u16 ( record_length ) ;
stream < < u8 ( entry . name_length ) ;
stream < < u8 ( entry . file_type ) ;
2019-06-09 12:46:23 +02:00
stream < < entry . name ;
int padding = record_length - entry . name_length - 8 ;
for ( int j = 0 ; j < padding ; + + j )
2019-07-03 21:17:35 +02:00
stream < < u8 ( 0 ) ;
2019-06-09 12:46:23 +02:00
}
stream . fill_to_end ( 0 ) ;
2019-09-30 08:57:01 +02:00
ssize_t nwritten = write_bytes ( 0 , directory_data . size ( ) , directory_data . data ( ) , nullptr ) ;
2019-06-09 12:46:23 +02:00
return nwritten = = directory_data . size ( ) ;
}
2019-06-09 10:25:19 +02:00
KResult Ext2FSInode : : add_child ( InodeIdentifier child_id , const StringView & name , mode_t mode )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2018-12-25 00:27:39 +01:00
ASSERT ( is_directory ( ) ) ;
2018-10-10 11:53:07 +02:00
2019-09-10 21:04:27 +02:00
if ( name . length ( ) > EXT2_NAME_LEN )
return KResult ( - ENAMETOOLONG ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-07-08 15:38:44 +02:00
dbg ( ) < < " Ext2FSInode::add_child(): Adding inode " < < child_id . index ( ) < < " with name ' " < < name < < " and mode " < < mode < < " to directory " < < index ( ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2018-12-25 00:27:39 +01:00
Vector < FS : : DirectoryEntry > entries ;
bool name_already_exists = false ;
2019-06-07 11:43:58 +02:00
traverse_as_directory ( [ & ] ( auto & entry ) {
2019-06-12 16:25:28 +03:00
if ( name = = entry . name ) {
2018-12-25 00:27:39 +01:00
name_already_exists = true ;
2018-10-10 11:53:07 +02:00
return false ;
}
entries . append ( entry ) ;
return true ;
} ) ;
2018-12-25 00:27:39 +01:00
if ( name_already_exists ) {
2019-07-08 15:38:44 +02:00
dbg ( ) < < " Ext2FSInode::add_child(): Name ' " < < name < < " ' already exists in inode " < < index ( ) ;
2019-02-27 15:31:26 +01:00
return KResult ( - EEXIST ) ;
2018-10-10 11:53:07 +02:00
}
2019-02-21 13:26:40 +01:00
auto child_inode = fs ( ) . get_inode ( child_id ) ;
if ( child_inode )
child_inode - > increment_link_count ( ) ;
2019-08-01 16:31:05 +02:00
entries . empend ( name . characters_without_null_termination ( ) , name . length ( ) , child_id , to_ext2_file_type ( mode ) ) ;
2019-06-09 12:46:23 +02:00
bool success = write_directory ( entries ) ;
2019-02-21 13:26:40 +01:00
if ( success )
2019-01-22 00:58:13 +01:00
m_lookup_cache . set ( name , child_id . index ( ) ) ;
2019-02-27 15:31:26 +01:00
return KSuccess ;
2018-10-10 11:53:07 +02:00
}
2019-06-09 10:25:19 +02:00
KResult Ext2FSInode : : remove_child ( const StringView & name )
2019-01-22 07:03:44 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-28 04:16:01 +01:00
# ifdef EXT2_DEBUG
2019-07-08 15:38:44 +02:00
dbg ( ) < < " Ext2FSInode::remove_child( " < < name < < " ) in inode " < < index ( ) ;
2019-01-28 04:16:01 +01:00
# endif
2019-01-22 07:03:44 +01:00
ASSERT ( is_directory ( ) ) ;
unsigned child_inode_index ;
2019-02-27 14:11:25 +01:00
auto it = m_lookup_cache . find ( name ) ;
if ( it = = m_lookup_cache . end ( ) )
return KResult ( - ENOENT ) ;
child_inode_index = ( * it ) . value ;
2019-01-22 07:03:44 +01:00
InodeIdentifier child_id { fsid ( ) , child_inode_index } ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-07-08 15:38:44 +02:00
dbg ( ) < < " Ext2FSInode::remove_child(): Removing ' " < < name < < " ' in directory " < < index ( ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-01-22 07:03:44 +01:00
Vector < FS : : DirectoryEntry > entries ;
2019-06-07 11:43:58 +02:00
traverse_as_directory ( [ & ] ( auto & entry ) {
2019-06-12 16:25:28 +03:00
if ( name ! = entry . name )
2019-02-21 13:26:40 +01:00
entries . append ( entry ) ;
2019-01-22 07:03:44 +01:00
return true ;
} ) ;
2019-06-09 12:46:23 +02:00
bool success = write_directory ( entries ) ;
2019-01-22 07:03:44 +01:00
if ( ! success ) {
2019-06-09 12:46:23 +02:00
// FIXME: Plumb error from write_directory().
2019-02-27 14:11:25 +01:00
return KResult ( - EIO ) ;
2019-01-22 07:03:44 +01:00
}
2019-02-27 14:11:25 +01:00
m_lookup_cache . remove ( name ) ;
2019-01-22 07:03:44 +01:00
auto child_inode = fs ( ) . get_inode ( child_id ) ;
child_inode - > decrement_link_count ( ) ;
2019-02-27 14:11:25 +01:00
return KSuccess ;
2019-01-22 07:03:44 +01:00
}
2018-12-03 00:20:00 +01:00
unsigned Ext2FS : : inodes_per_block ( ) const
2018-10-10 11:53:07 +02:00
{
2018-12-03 00:20:00 +01:00
return EXT2_INODES_PER_BLOCK ( & super_block ( ) ) ;
2018-10-10 11:53:07 +02:00
}
2018-12-03 00:20:00 +01:00
unsigned Ext2FS : : inodes_per_group ( ) const
2018-10-10 11:53:07 +02:00
{
2018-12-03 00:20:00 +01:00
return EXT2_INODES_PER_GROUP ( & super_block ( ) ) ;
2018-10-10 11:53:07 +02:00
}
2018-12-03 00:20:00 +01:00
unsigned Ext2FS : : inode_size ( ) const
2018-10-10 11:53:07 +02:00
{
2018-12-03 00:20:00 +01:00
return EXT2_INODE_SIZE ( & super_block ( ) ) ;
2018-10-10 11:53:07 +02:00
}
2018-12-03 00:20:00 +01:00
unsigned Ext2FS : : blocks_per_group ( ) const
2018-10-10 11:53:07 +02:00
{
2018-12-03 00:20:00 +01:00
return EXT2_BLOCKS_PER_GROUP ( & super_block ( ) ) ;
2018-10-10 11:53:07 +02:00
}
2018-12-03 00:20:00 +01:00
bool Ext2FS : : write_ext2_inode ( unsigned inode , const ext2_inode & e2inode )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-31 17:31:23 +01:00
unsigned block_index ;
2018-10-10 11:53:07 +02:00
unsigned offset ;
2019-09-30 11:20:51 +02:00
u8 block [ max_block_size ] ;
if ( ! read_block_containing_inode ( inode , block_index , offset , block ) )
2018-10-10 11:53:07 +02:00
return false ;
2019-09-30 11:20:51 +02:00
memcpy ( reinterpret_cast < ext2_inode * > ( block + offset ) , & e2inode , inode_size ( ) ) ;
2019-02-11 11:38:14 +01:00
bool success = write_block ( block_index , block ) ;
ASSERT ( success ) ;
return success ;
2018-10-10 11:53:07 +02:00
}
2019-09-22 18:34:52 +02:00
Ext2FS : : BlockIndex Ext2FS : : allocate_block ( GroupIndex preferred_group_index )
2018-10-16 00:35:03 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-09-22 18:34:52 +02:00
dbg ( ) < < " Ext2FS: allocate_block() preferred_group_index: " < < preferred_group_index ;
2019-07-21 18:34:26 +02:00
# endif
2019-09-22 18:34:52 +02:00
bool found_a_group = false ;
GroupIndex group_index = preferred_group_index ;
2018-10-16 00:35:03 +02:00
2019-09-22 18:34:52 +02:00
if ( group_descriptor ( preferred_group_index ) . bg_free_blocks_count ) {
found_a_group = true ;
} else {
for ( group_index = 1 ; group_index < m_block_group_count ; + + group_index ) {
if ( group_descriptor ( group_index ) . bg_free_blocks_count ) {
found_a_group = true ;
break ;
}
}
2018-10-16 00:35:03 +02:00
}
2019-09-22 18:34:52 +02:00
ASSERT ( found_a_group ) ;
auto & bgd = group_descriptor ( group_index ) ;
2019-11-02 12:26:20 +01:00
auto & cached_bitmap = get_bitmap_block ( bgd . bg_block_bitmap ) ;
2018-10-16 00:35:03 +02:00
2019-04-23 14:51:47 +02:00
int blocks_in_group = min ( blocks_per_group ( ) , super_block ( ) . s_blocks_count ) ;
2019-11-02 12:18:43 +01:00
auto block_bitmap = Bitmap : : wrap ( cached_bitmap . buffer . data ( ) , blocks_in_group ) ;
2019-09-22 18:34:52 +02:00
BlockIndex first_block_in_group = ( group_index - 1 ) * blocks_per_group ( ) + first_block_index ( ) ;
int first_unset_bit_index = block_bitmap . find_first_unset ( ) ;
ASSERT ( first_unset_bit_index ! = - 1 ) ;
BlockIndex block_index = ( unsigned ) first_unset_bit_index + first_block_in_group ;
set_block_allocation_state ( block_index , true ) ;
return block_index ;
}
2019-04-23 14:51:47 +02:00
2019-09-22 18:34:52 +02:00
Vector < Ext2FS : : BlockIndex > Ext2FS : : allocate_blocks ( GroupIndex preferred_group_index , int count )
{
LOCKER ( m_lock ) ;
# ifdef EXT2_DEBUG
2019-09-28 14:31:40 +10:00
dbgprintf ( " Ext2FS: allocate_blocks(preferred group: %u, count: %u) \n " , preferred_group_index , count ) ;
2019-09-22 18:34:52 +02:00
# endif
if ( count = = 0 )
return { } ;
Vector < BlockIndex > blocks ;
2019-09-28 14:31:40 +10:00
# ifdef EXT2_DEBUG
2019-09-22 18:34:52 +02:00
dbg ( ) < < " Ext2FS: allocate_blocks: " ;
2019-09-28 14:31:40 +10:00
# endif
2019-09-22 18:34:52 +02:00
blocks . ensure_capacity ( count ) ;
for ( int i = 0 ; i < count ; + + i ) {
auto block_index = allocate_block ( preferred_group_index ) ;
blocks . unchecked_append ( block_index ) ;
2019-09-28 14:31:40 +10:00
# ifdef EXT2_DEBUG
2019-09-22 18:34:52 +02:00
dbg ( ) < < " > " < < block_index ;
2019-09-28 14:31:40 +10:00
# endif
2018-10-16 00:35:03 +02:00
}
2019-09-22 18:34:52 +02:00
ASSERT ( blocks . size ( ) = = count ) ;
2018-10-16 00:35:03 +02:00
return blocks ;
}
2019-04-23 13:00:53 +02:00
unsigned Ext2FS : : allocate_inode ( GroupIndex preferred_group , off_t expected_size )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-05-25 17:23:17 +02:00
dbgprintf ( " Ext2FS: allocate_inode(preferredGroup: %u, expected_size: %u) \n " , preferred_group , expected_size ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-01-31 17:31:23 +01:00
unsigned needed_blocks = ceil_div ( expected_size , block_size ( ) ) ;
2018-10-10 11:53:07 +02:00
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-01-31 17:31:23 +01:00
dbgprintf ( " Ext2FS: minimum needed blocks: %u \n " , needed_blocks ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-04-23 15:07:07 +02:00
unsigned group_index = 0 ;
2018-10-10 11:53:07 +02:00
2019-06-07 11:43:58 +02:00
auto is_suitable_group = [ this , needed_blocks ] ( GroupIndex group_index ) {
2019-05-25 17:23:17 +02:00
auto & bgd = group_descriptor ( group_index ) ;
2019-01-31 17:31:23 +01:00
return bgd . bg_free_inodes_count & & bgd . bg_free_blocks_count > = needed_blocks ;
2018-10-10 11:53:07 +02:00
} ;
2019-01-31 17:31:23 +01:00
if ( preferred_group & & is_suitable_group ( preferred_group ) ) {
2019-04-23 15:07:07 +02:00
group_index = preferred_group ;
2018-10-10 11:53:07 +02:00
} else {
2019-01-31 17:31:23 +01:00
for ( unsigned i = 1 ; i < = m_block_group_count ; + + i ) {
if ( is_suitable_group ( i ) )
2019-04-23 15:07:07 +02:00
group_index = i ;
2018-10-10 11:53:07 +02:00
}
}
2019-04-23 15:07:07 +02:00
if ( ! group_index ) {
2019-01-31 17:31:23 +01:00
kprintf ( " Ext2FS: allocate_inode: no suitable group found for new inode with %u blocks needed :( \n " , needed_blocks ) ;
2018-10-10 11:53:07 +02:00
return 0 ;
}
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-04-23 15:07:07 +02:00
dbgprintf ( " Ext2FS: allocate_inode: found suitable group [%u] for new inode with %u blocks needed :^) \n " , group_index , needed_blocks ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-04-23 15:07:07 +02:00
auto & bgd = group_descriptor ( group_index ) ;
unsigned inodes_in_group = min ( inodes_per_group ( ) , super_block ( ) . s_inodes_count ) ;
2019-02-25 22:06:55 +01:00
unsigned first_free_inode_in_group = 0 ;
2019-04-23 15:07:07 +02:00
unsigned first_inode_in_group = ( group_index - 1 ) * inodes_per_group ( ) + 1 ;
2019-11-02 12:34:18 +01:00
auto & cached_bitmap = get_bitmap_block ( bgd . bg_inode_bitmap ) ;
auto inode_bitmap = Bitmap : : wrap ( cached_bitmap . buffer . data ( ) , inodes_in_group ) ;
2019-04-23 15:07:07 +02:00
for ( int i = 0 ; i < inode_bitmap . size ( ) ; + + i ) {
if ( inode_bitmap . get ( i ) )
continue ;
first_free_inode_in_group = first_inode_in_group + i ;
break ;
}
2018-10-10 11:53:07 +02:00
2019-02-25 22:06:55 +01:00
if ( ! first_free_inode_in_group ) {
2019-01-22 00:58:13 +01:00
kprintf ( " Ext2FS: first_free_inode_in_group returned no inode, despite bgd claiming there are inodes :( \n " ) ;
2018-10-10 11:53:07 +02:00
return 0 ;
}
2019-02-25 22:06:55 +01:00
unsigned inode = first_free_inode_in_group ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2018-11-18 14:57:41 +01:00
dbgprintf ( " Ext2FS: found suitable inode %u \n " , inode ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-02-15 23:24:01 +01:00
ASSERT ( get_inode_allocation_state ( inode ) = = false ) ;
2018-10-10 11:53:07 +02:00
// FIXME: allocate blocks if needed!
return inode ;
}
2019-02-15 23:24:01 +01:00
Ext2FS : : GroupIndex Ext2FS : : group_index_from_block_index ( BlockIndex block_index ) const
{
if ( ! block_index )
return 0 ;
return ( block_index - 1 ) / blocks_per_group ( ) + 1 ;
}
2018-12-03 00:20:00 +01:00
unsigned Ext2FS : : group_index_from_inode ( unsigned inode ) const
2018-10-10 11:53:07 +02:00
{
if ( ! inode )
return 0 ;
2018-12-03 00:20:00 +01:00
return ( inode - 1 ) / inodes_per_group ( ) + 1 ;
2018-10-10 11:53:07 +02:00
}
2019-01-01 03:35:33 +01:00
bool Ext2FS : : get_inode_allocation_state ( InodeIndex index ) const
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:35:33 +01:00
if ( index = = 0 )
return true ;
2019-02-11 12:46:56 +01:00
unsigned group_index = group_index_from_inode ( index ) ;
auto & bgd = group_descriptor ( group_index ) ;
unsigned index_in_group = index - ( ( group_index - 1 ) * inodes_per_group ( ) ) ;
2019-04-23 16:21:07 +02:00
unsigned bit_index = ( index_in_group - 1 ) % inodes_per_group ( ) ;
2019-11-02 12:34:18 +01:00
auto & cached_bitmap = const_cast < Ext2FS & > ( * this ) . get_bitmap_block ( bgd . bg_inode_bitmap ) ;
return cached_bitmap . bitmap ( inodes_per_group ( ) ) . get ( bit_index ) ;
2019-01-01 03:35:33 +01:00
}
2019-05-25 17:23:17 +02:00
bool Ext2FS : : set_inode_allocation_state ( InodeIndex inode_index , bool new_state )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-05-25 17:23:17 +02:00
unsigned group_index = group_index_from_inode ( inode_index ) ;
2019-02-11 12:46:56 +01:00
auto & bgd = group_descriptor ( group_index ) ;
2019-05-25 17:23:17 +02:00
unsigned index_in_group = inode_index - ( ( group_index - 1 ) * inodes_per_group ( ) ) ;
2019-04-23 16:21:07 +02:00
unsigned bit_index = ( index_in_group - 1 ) % inodes_per_group ( ) ;
2019-11-02 12:34:18 +01:00
auto & cached_bitmap = get_bitmap_block ( bgd . bg_inode_bitmap ) ;
bool current_state = cached_bitmap . bitmap ( inodes_per_group ( ) ) . get ( bit_index ) ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-05-25 17:23:17 +02:00
dbgprintf ( " Ext2FS: set_inode_allocation_state(%u) %u -> %u \n " , inode_index , current_state , new_state ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-05-25 17:23:17 +02:00
if ( current_state = = new_state ) {
2019-04-23 16:21:07 +02:00
ASSERT_NOT_REACHED ( ) ;
2018-10-10 11:53:07 +02:00
return true ;
2019-04-23 16:21:07 +02:00
}
2018-10-10 11:53:07 +02:00
2019-11-02 12:34:18 +01:00
cached_bitmap . bitmap ( inodes_per_group ( ) ) . set ( bit_index , new_state ) ;
cached_bitmap . dirty = true ;
2018-10-10 11:53:07 +02:00
// Update superblock
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-11-02 11:49:11 +01:00
dbgprintf ( " Ext2FS: superblock free inode count %u -> %u \n " , m_super_block . s_free_inodes_count , m_super_block . s_free_inodes_count - 1 ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-05-25 17:23:17 +02:00
if ( new_state )
2019-11-02 11:49:11 +01:00
- - m_super_block . s_free_inodes_count ;
2018-10-10 11:53:07 +02:00
else
2019-11-02 11:49:11 +01:00
+ + m_super_block . s_free_inodes_count ;
2019-11-02 11:36:56 +01:00
m_super_block_dirty = true ;
2018-10-10 11:53:07 +02:00
// Update BGD
2019-01-31 17:31:23 +01:00
auto & mutable_bgd = const_cast < ext2_group_desc & > ( bgd ) ;
2019-05-25 17:23:17 +02:00
if ( new_state )
2019-01-31 17:31:23 +01:00
- - mutable_bgd . bg_free_inodes_count ;
2018-10-10 11:53:07 +02:00
else
2019-01-31 17:31:23 +01:00
+ + mutable_bgd . bg_free_inodes_count ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2018-11-18 14:57:41 +01:00
dbgprintf ( " Ext2FS: group free inode count %u -> %u \n " , bgd . bg_free_inodes_count , bgd . bg_free_inodes_count - 1 ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-11-02 11:36:56 +01:00
m_block_group_descriptors_dirty = true ;
2018-10-16 00:35:03 +02:00
return true ;
}
2019-09-22 18:34:52 +02:00
Ext2FS : : BlockIndex Ext2FS : : first_block_index ( ) const
{
return block_size ( ) = = 1024 ? 1 : 0 ;
}
2019-11-02 12:26:20 +01:00
Ext2FS : : CachedBitmap & Ext2FS : : get_bitmap_block ( BlockIndex bitmap_block_index )
2019-11-02 12:18:43 +01:00
{
for ( auto & cached_bitmap : m_cached_bitmaps ) {
if ( cached_bitmap - > bitmap_block_index = = bitmap_block_index )
return * cached_bitmap ;
}
auto block = ByteBuffer : : create_uninitialized ( block_size ( ) ) ;
bool success = read_block ( bitmap_block_index , block . data ( ) ) ;
ASSERT ( success ) ;
m_cached_bitmaps . append ( make < CachedBitmap > ( bitmap_block_index , move ( block ) ) ) ;
return * m_cached_bitmaps . last ( ) ;
}
2019-02-15 23:24:01 +01:00
bool Ext2FS : : set_block_allocation_state ( BlockIndex block_index , bool new_state )
2018-10-16 00:35:03 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-04-25 22:05:32 +02:00
# ifdef EXT2_DEBUG
2019-02-15 23:24:01 +01:00
dbgprintf ( " Ext2FS: set_block_allocation_state(block=%u, state=%u) \n " , block_index , new_state ) ;
2019-04-25 22:05:32 +02:00
# endif
2019-11-02 12:18:43 +01:00
GroupIndex group_index = group_index_from_block_index ( block_index ) ;
2019-02-15 23:24:01 +01:00
auto & bgd = group_descriptor ( group_index ) ;
2019-09-22 18:34:52 +02:00
BlockIndex index_in_group = ( block_index - first_block_index ( ) ) - ( ( group_index - 1 ) * blocks_per_group ( ) ) ;
2019-04-23 16:38:45 +02:00
unsigned bit_index = index_in_group % blocks_per_group ( ) ;
2019-11-02 12:18:43 +01:00
2019-11-02 12:26:20 +01:00
auto & cached_bitmap = get_bitmap_block ( bgd . bg_block_bitmap ) ;
2019-11-02 12:18:43 +01:00
bool current_state = cached_bitmap . bitmap ( blocks_per_group ( ) ) . get ( bit_index ) ;
2019-04-25 22:05:32 +02:00
# ifdef EXT2_DEBUG
2019-11-02 12:18:43 +01:00
dbgprintf ( " Ext2FS: block %u state: %u -> %u (in bitmap block %u) \n " , block_index , current_state , new_state , bgd . bg_block_bitmap ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-16 00:35:03 +02:00
2019-04-23 16:21:07 +02:00
if ( current_state = = new_state ) {
ASSERT_NOT_REACHED ( ) ;
2018-10-16 00:35:03 +02:00
return true ;
2019-04-23 16:21:07 +02:00
}
2018-10-16 00:35:03 +02:00
2019-11-02 12:18:43 +01:00
cached_bitmap . bitmap ( blocks_per_group ( ) ) . set ( bit_index , new_state ) ;
cached_bitmap . dirty = true ;
2018-10-16 00:35:03 +02:00
// Update superblock
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-11-02 11:49:11 +01:00
dbgprintf ( " Ext2FS: superblock free block count %u -> %u \n " , m_super_block . s_free_blocks_count , m_super_block . s_free_blocks_count - 1 ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-01-31 17:31:23 +01:00
if ( new_state )
2019-11-02 11:49:11 +01:00
- - m_super_block . s_free_blocks_count ;
2018-10-16 00:35:03 +02:00
else
2019-11-02 11:49:11 +01:00
+ + m_super_block . s_free_blocks_count ;
2019-11-02 11:36:56 +01:00
m_super_block_dirty = true ;
2018-10-16 00:35:03 +02:00
// Update BGD
2019-01-31 17:31:23 +01:00
auto & mutable_bgd = const_cast < ext2_group_desc & > ( bgd ) ;
if ( new_state )
- - mutable_bgd . bg_free_blocks_count ;
2018-10-16 00:35:03 +02:00
else
2019-01-31 17:31:23 +01:00
+ + mutable_bgd . bg_free_blocks_count ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-04-25 22:05:32 +02:00
dbgprintf ( " Ext2FS: group %u free block count %u -> %u \n " , group_index , bgd . bg_free_blocks_count , bgd . bg_free_blocks_count - 1 ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-16 00:35:03 +02:00
2019-11-02 11:36:56 +01:00
m_block_group_descriptors_dirty = true ;
2018-10-10 11:53:07 +02:00
return true ;
}
2019-06-21 18:37:47 +02:00
RefPtr < Inode > Ext2FS : : create_directory ( InodeIdentifier parent_id , const String & name , mode_t mode , int & error )
2018-10-16 00:35:03 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-23 05:38:54 +01:00
ASSERT ( parent_id . fsid ( ) = = fsid ( ) ) ;
2018-10-16 00:35:03 +02:00
// Fix up the mode to definitely be a directory.
// FIXME: This is a bit on the hackish side.
mode & = ~ 0170000 ;
mode | = 0040000 ;
// NOTE: When creating a new directory, make the size 1 block.
// There's probably a better strategy here, but this works for now.
2019-05-03 22:59:58 +02:00
auto inode = create_inode ( parent_id , name , mode , block_size ( ) , 0 , error ) ;
2018-12-24 23:44:46 +01:00
if ( ! inode )
2018-12-24 23:58:00 +01:00
return nullptr ;
2018-10-16 00:35:03 +02:00
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2018-12-24 23:44:46 +01:00
dbgprintf ( " Ext2FS: create_directory: created new directory named '%s' with inode %u \n " , name . characters ( ) , inode - > identifier ( ) . index ( ) ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-16 00:35:03 +02:00
Vector < DirectoryEntry > entries ;
2019-08-01 16:31:05 +02:00
entries . empend ( " . " , inode - > identifier ( ) , EXT2_FT_DIR ) ;
entries . empend ( " .. " , parent_id , EXT2_FT_DIR ) ;
2018-10-16 00:35:03 +02:00
2019-06-09 12:46:23 +02:00
bool success = static_cast < Ext2FSInode & > ( * inode ) . write_directory ( entries ) ;
2018-10-16 00:35:03 +02:00
ASSERT ( success ) ;
2018-12-24 23:58:00 +01:00
auto parent_inode = get_inode ( parent_id ) ;
error = parent_inode - > increment_link_count ( ) ;
if ( error < 0 )
return nullptr ;
2018-10-16 00:35:03 +02:00
2018-12-24 23:44:46 +01:00
auto & bgd = const_cast < ext2_group_desc & > ( group_descriptor ( group_index_from_inode ( inode - > identifier ( ) . index ( ) ) ) ) ;
2018-10-16 00:35:03 +02:00
+ + bgd . bg_used_dirs_count ;
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2018-11-18 14:57:41 +01:00
dbgprintf ( " Ext2FS: incremented bg_used_dirs_count %u -> %u \n " , bgd . bg_used_dirs_count - 1 , bgd . bg_used_dirs_count ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-16 00:35:03 +02:00
2019-11-02 11:36:56 +01:00
m_block_group_descriptors_dirty = true ;
2018-10-16 00:35:03 +02:00
2018-11-18 14:57:41 +01:00
error = 0 ;
2018-10-16 00:35:03 +02:00
return inode ;
}
2019-06-21 18:37:47 +02:00
RefPtr < Inode > Ext2FS : : create_inode ( InodeIdentifier parent_id , const String & name , mode_t mode , off_t size , dev_t dev , int & error )
2018-10-10 11:53:07 +02:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-23 05:38:54 +01:00
ASSERT ( parent_id . fsid ( ) = = fsid ( ) ) ;
2018-12-25 00:27:39 +01:00
auto parent_inode = get_inode ( parent_id ) ;
2018-10-10 11:53:07 +02:00
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-02-22 09:13:05 +01:00
dbgprintf ( " Ext2FS: Adding inode '%s' (mode %o) to parent directory %u: \n " , name . characters ( ) , mode , parent_inode - > identifier ( ) . index ( ) ) ;
2019-07-21 18:34:26 +02:00
# endif
2018-10-10 11:53:07 +02:00
2019-09-22 18:34:52 +02:00
auto needed_blocks = ceil_div ( size , block_size ( ) ) ;
2019-09-22 18:44:05 +02:00
if ( ( size_t ) needed_blocks > super_block ( ) . s_free_blocks_count ) {
2019-09-22 18:34:52 +02:00
dbg ( ) < < " Ext2FS: create_inode: not enough free blocks " ;
error = - ENOSPC ;
return { } ;
}
2018-10-10 11:53:07 +02:00
// NOTE: This doesn't commit the inode allocation just yet!
2019-01-23 04:59:47 +01:00
auto inode_id = allocate_inode ( 0 , size ) ;
2018-12-24 23:44:46 +01:00
if ( ! inode_id ) {
2019-01-22 00:58:13 +01:00
kprintf ( " Ext2FS: create_inode: allocate_inode failed \n " ) ;
2018-11-18 14:57:41 +01:00
error = - ENOSPC ;
2019-06-07 11:43:58 +02:00
return { } ;
2018-10-16 00:35:03 +02:00
}
2018-10-10 11:53:07 +02:00
// Try adding it to the directory first, in case the name is already in use.
2019-06-09 12:48:34 +02:00
auto result = parent_inode - > add_child ( { fsid ( ) , inode_id } , name , mode ) ;
2019-02-27 15:31:26 +01:00
if ( result . is_error ( ) ) {
error = result ;
2019-06-07 11:43:58 +02:00
return { } ;
2019-02-27 15:31:26 +01:00
}
2018-10-10 11:53:07 +02:00
2019-09-22 18:50:24 +02:00
auto blocks = allocate_blocks ( group_index_from_inode ( inode_id ) , needed_blocks ) ;
ASSERT ( blocks . size ( ) = = needed_blocks ) ;
2018-10-10 11:53:07 +02:00
// Looks like we're good, time to update the inode bitmap and group+global inode counters.
2019-02-27 15:31:26 +01:00
bool success = set_inode_allocation_state ( inode_id , true ) ;
2018-10-10 11:53:07 +02:00
ASSERT ( success ) ;
2019-01-31 17:31:23 +01:00
unsigned initial_links_count ;
if ( is_directory ( mode ) )
initial_links_count = 2 ; // (parent directory + "." entry in self)
2018-10-16 00:35:03 +02:00
else
2019-01-31 17:31:23 +01:00
initial_links_count = 1 ;
2018-10-16 00:35:03 +02:00
2019-03-25 02:06:57 +01:00
struct timeval now ;
kgettimeofday ( now ) ;
2019-01-31 04:17:16 +01:00
ext2_inode e2inode ;
memset ( & e2inode , 0 , sizeof ( ext2_inode ) ) ;
e2inode . i_mode = mode ;
2019-03-23 22:03:17 +01:00
e2inode . i_uid = current - > process ( ) . euid ( ) ;
e2inode . i_gid = current - > process ( ) . egid ( ) ;
2019-01-31 04:17:16 +01:00
e2inode . i_size = size ;
2019-03-25 02:06:57 +01:00
e2inode . i_atime = now . tv_sec ;
e2inode . i_ctime = now . tv_sec ;
e2inode . i_mtime = now . tv_sec ;
2019-01-31 04:17:16 +01:00
e2inode . i_dtime = 0 ;
2019-01-31 17:31:23 +01:00
e2inode . i_links_count = initial_links_count ;
2019-01-31 04:17:16 +01:00
2019-05-03 22:59:58 +02:00
if ( is_character_device ( mode ) )
e2inode . i_block [ 0 ] = dev ;
else if ( is_block_device ( mode ) )
e2inode . i_block [ 1 ] = dev ;
2019-01-31 04:17:16 +01:00
success = write_block_list_for_inode ( inode_id , e2inode , blocks ) ;
2019-01-23 02:45:25 +01:00
ASSERT ( success ) ;
2018-10-16 00:35:03 +02:00
2019-07-21 18:34:26 +02:00
# ifdef EXT2_DEBUG
2019-01-01 03:55:13 +01:00
dbgprintf ( " Ext2FS: writing initial metadata for inode %u \n " , inode_id ) ;
2019-07-21 18:34:26 +02:00
# endif
2019-01-31 04:17:16 +01:00
e2inode . i_flags = 0 ;
success = write_ext2_inode ( inode_id , e2inode ) ;
2018-10-10 11:53:07 +02:00
ASSERT ( success ) ;
2019-02-20 21:41:53 +01:00
// We might have cached the fact that this inode didn't exist. Wipe the slate.
m_inode_cache . remove ( inode_id ) ;
2019-01-23 05:38:54 +01:00
return get_inode ( { fsid ( ) , inode_id } ) ;
2018-10-10 11:53:07 +02:00
}
2019-01-28 04:16:01 +01:00
void Ext2FSInode : : populate_lookup_cache ( ) const
2018-11-15 16:34:36 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
if ( ! m_lookup_cache . is_empty ( ) )
return ;
2018-11-15 17:04:55 +01:00
HashMap < String , unsigned > children ;
2019-06-07 11:43:58 +02:00
traverse_as_directory ( [ & children ] ( auto & entry ) {
2018-11-15 17:04:55 +01:00
children . set ( String ( entry . name , entry . name_length ) , entry . inode . index ( ) ) ;
return true ;
} ) ;
2018-12-21 02:10:45 +01:00
if ( ! m_lookup_cache . is_empty ( ) )
2018-11-15 17:04:55 +01:00
return ;
m_lookup_cache = move ( children ) ;
}
2018-11-15 16:34:36 +01:00
2019-06-01 18:01:28 +02:00
InodeIdentifier Ext2FSInode : : lookup ( StringView name )
2018-11-15 17:04:55 +01:00
{
ASSERT ( is_directory ( ) ) ;
populate_lookup_cache ( ) ;
2018-11-15 16:34:36 +01:00
LOCKER ( m_lock ) ;
2019-08-24 22:39:12 +02:00
auto it = m_lookup_cache . find ( name . hash ( ) , [ & ] ( auto & entry ) { return entry . key = = name ; } ) ;
2018-11-15 17:04:55 +01:00
if ( it ! = m_lookup_cache . end ( ) )
2018-11-15 16:34:36 +01:00
return { fsid ( ) , ( * it ) . value } ;
2019-06-07 11:43:58 +02:00
return { } ;
2018-11-15 17:04:55 +01:00
}
2018-11-15 16:34:36 +01:00
2019-06-21 15:29:31 +02:00
void Ext2FSInode : : one_ref_left ( )
2019-01-01 02:38:09 +01:00
{
2019-01-01 02:52:21 +01:00
// FIXME: I would like to not live forever, but uncached Ext2FS is fucking painful right now.
2019-01-01 02:38:09 +01:00
}
2019-01-01 03:16:36 +01:00
2019-01-23 06:53:01 +01:00
int Ext2FSInode : : set_atime ( time_t t )
2019-01-01 03:16:36 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:16:36 +01:00
if ( fs ( ) . is_readonly ( ) )
return - EROFS ;
m_raw_inode . i_atime = t ;
set_metadata_dirty ( true ) ;
return 0 ;
}
2019-01-23 06:53:01 +01:00
int Ext2FSInode : : set_ctime ( time_t t )
2019-01-01 03:16:36 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:16:36 +01:00
if ( fs ( ) . is_readonly ( ) )
return - EROFS ;
m_raw_inode . i_ctime = t ;
set_metadata_dirty ( true ) ;
return 0 ;
}
2019-01-23 06:53:01 +01:00
int Ext2FSInode : : set_mtime ( time_t t )
2019-01-01 03:16:36 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:16:36 +01:00
if ( fs ( ) . is_readonly ( ) )
return - EROFS ;
m_raw_inode . i_mtime = t ;
set_metadata_dirty ( true ) ;
return 0 ;
}
int Ext2FSInode : : increment_link_count ( )
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:16:36 +01:00
if ( fs ( ) . is_readonly ( ) )
return - EROFS ;
+ + m_raw_inode . i_links_count ;
set_metadata_dirty ( true ) ;
return 0 ;
}
int Ext2FSInode : : decrement_link_count ( )
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-01 03:16:36 +01:00
if ( fs ( ) . is_readonly ( ) )
return - EROFS ;
2019-01-22 07:03:44 +01:00
ASSERT ( m_raw_inode . i_links_count ) ;
2019-01-01 03:16:36 +01:00
- - m_raw_inode . i_links_count ;
2019-01-22 07:03:44 +01:00
if ( m_raw_inode . i_links_count = = 0 )
fs ( ) . uncache_inode ( index ( ) ) ;
2019-01-01 03:16:36 +01:00
set_metadata_dirty ( true ) ;
return 0 ;
}
2019-01-22 07:03:44 +01:00
void Ext2FS : : uncache_inode ( InodeIndex index )
{
2019-02-20 21:41:53 +01:00
LOCKER ( m_lock ) ;
2019-02-21 13:26:40 +01:00
m_inode_cache . remove ( index ) ;
2019-01-22 07:03:44 +01:00
}
2019-01-28 04:16:01 +01:00
size_t Ext2FSInode : : directory_entry_count ( ) const
{
ASSERT ( is_directory ( ) ) ;
LOCKER ( m_lock ) ;
2019-02-27 17:07:34 +01:00
populate_lookup_cache ( ) ;
2019-01-28 04:16:01 +01:00
return m_lookup_cache . size ( ) ;
}
2019-01-29 04:55:08 +01:00
2019-02-25 20:47:56 +01:00
KResult Ext2FSInode : : chmod ( mode_t mode )
2019-01-29 04:55:08 +01:00
{
2019-02-20 13:09:59 +01:00
LOCKER ( m_lock ) ;
2019-01-29 04:55:08 +01:00
if ( m_raw_inode . i_mode = = mode )
2019-02-25 20:47:56 +01:00
return KSuccess ;
2019-01-29 04:55:08 +01:00
m_raw_inode . i_mode = mode ;
set_metadata_dirty ( true ) ;
2019-02-25 20:47:56 +01:00
return KSuccess ;
2019-01-29 04:55:08 +01:00
}
2019-02-21 14:48:00 +01:00
2019-02-27 12:32:53 +01:00
KResult Ext2FSInode : : chown ( uid_t uid , gid_t gid )
{
LOCKER ( m_lock ) ;
if ( m_raw_inode . i_uid = = uid & & m_raw_inode . i_gid = = gid )
return KSuccess ;
m_raw_inode . i_uid = uid ;
m_raw_inode . i_gid = gid ;
set_metadata_dirty ( true ) ;
return KSuccess ;
}
2019-04-23 13:00:53 +02:00
KResult Ext2FSInode : : truncate ( off_t size )
2019-03-27 16:42:30 +01:00
{
LOCKER ( m_lock ) ;
2019-04-23 13:00:53 +02:00
if ( ( off_t ) m_raw_inode . i_size = = size )
2019-03-27 16:42:30 +01:00
return KSuccess ;
2019-11-02 12:53:31 +01:00
auto result = resize ( size ) ;
if ( result . is_error ( ) )
return result ;
2019-03-27 16:42:30 +01:00
set_metadata_dirty ( true ) ;
return KSuccess ;
}
2019-02-21 14:48:00 +01:00
unsigned Ext2FS : : total_block_count ( ) const
{
LOCKER ( m_lock ) ;
return super_block ( ) . s_blocks_count ;
}
unsigned Ext2FS : : free_block_count ( ) const
{
LOCKER ( m_lock ) ;
return super_block ( ) . s_free_blocks_count ;
}
unsigned Ext2FS : : total_inode_count ( ) const
{
LOCKER ( m_lock ) ;
return super_block ( ) . s_inodes_count ;
}
unsigned Ext2FS : : free_inode_count ( ) const
{
LOCKER ( m_lock ) ;
return super_block ( ) . s_free_inodes_count ;
}
2019-08-11 23:56:39 +10:00
KResult Ext2FS : : prepare_to_unmount ( ) const
{
2019-08-17 13:49:37 +02:00
LOCKER ( m_lock ) ;
for ( auto & it : m_inode_cache ) {
if ( it . value - > ref_count ( ) > 1 )
2019-08-11 23:56:39 +10:00
return KResult ( - EBUSY ) ;
}
m_inode_cache . clear ( ) ;
return KSuccess ;
}