2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-17 02:20:36 +04:00
/*
* linux / fs / adfs / inode . c
*
* Copyright ( C ) 1997 - 1999 Russell King
*/
# include <linux/buffer_head.h>
2010-03-05 11:21:37 +03:00
# include <linux/writeback.h>
2005-04-17 02:20:36 +04:00
# include "adfs.h"
/*
* Lookup / Create a block at offset ' block ' into ' inode ' . We currently do
* not support creation of new blocks , so we return - EIO for this case .
*/
static int
adfs_get_block ( struct inode * inode , sector_t block , struct buffer_head * bh ,
int create )
{
if ( ! create ) {
if ( block > = inode - > i_blocks )
goto abort_toobig ;
2019-12-09 14:11:38 +03:00
block = __adfs_block_map ( inode - > i_sb , ADFS_I ( inode ) - > indaddr ,
block ) ;
2005-04-17 02:20:36 +04:00
if ( block )
map_bh ( bh , inode - > i_sb , block ) ;
return 0 ;
}
/* don't support allocation of blocks yet */
return - EIO ;
abort_toobig :
return 0 ;
}
static int adfs_writepage ( struct page * page , struct writeback_control * wbc )
{
return block_write_full_page ( page , adfs_get_block , wbc ) ;
}
2022-04-29 17:40:40 +03:00
static int adfs_read_folio ( struct file * file , struct folio * folio )
2005-04-17 02:20:36 +04:00
{
2022-04-29 17:40:40 +03:00
return block_read_full_folio ( folio , adfs_get_block ) ;
2005-04-17 02:20:36 +04:00
}
2012-12-15 14:51:11 +04:00
static void adfs_write_failed ( struct address_space * mapping , loff_t to )
{
struct inode * inode = mapping - > host ;
if ( to > inode - > i_size )
2013-09-13 02:13:56 +04:00
truncate_pagecache ( inode , inode - > i_size ) ;
2012-12-15 14:51:11 +04:00
}
2007-10-16 12:25:23 +04:00
static int adfs_write_begin ( struct file * file , struct address_space * mapping ,
2022-02-22 22:31:43 +03:00
loff_t pos , unsigned len ,
2007-10-16 12:25:23 +04:00
struct page * * pagep , void * * fsdata )
2005-04-17 02:20:36 +04:00
{
2010-06-04 13:29:55 +04:00
int ret ;
2007-10-16 12:25:23 +04:00
* pagep = NULL ;
2022-02-22 19:25:12 +03:00
ret = cont_write_begin ( file , mapping , pos , len , pagep , fsdata ,
2007-10-16 12:25:23 +04:00
adfs_get_block ,
& ADFS_I ( mapping - > host ) - > mmu_private ) ;
2012-12-15 14:51:11 +04:00
if ( unlikely ( ret ) )
adfs_write_failed ( mapping , pos + len ) ;
2010-06-04 13:29:55 +04:00
return ret ;
2005-04-17 02:20:36 +04:00
}
static sector_t _adfs_bmap ( struct address_space * mapping , sector_t block )
{
return generic_block_bmap ( mapping , block , adfs_get_block ) ;
}
2006-06-28 15:26:44 +04:00
static const struct address_space_operations adfs_aops = {
2022-02-09 23:22:12 +03:00
. dirty_folio = block_dirty_folio ,
2022-02-09 23:21:34 +03:00
. invalidate_folio = block_invalidate_folio ,
2022-04-29 17:40:40 +03:00
. read_folio = adfs_read_folio ,
2005-04-17 02:20:36 +04:00
. writepage = adfs_writepage ,
2007-10-16 12:25:23 +04:00
. write_begin = adfs_write_begin ,
. write_end = generic_write_end ,
2005-04-17 02:20:36 +04:00
. bmap = _adfs_bmap
} ;
/*
* Convert ADFS attributes and filetype to Linux permission .
*/
static umode_t
adfs_atts2mode ( struct super_block * sb , struct inode * inode )
{
2011-03-23 02:35:06 +03:00
unsigned int attr = ADFS_I ( inode ) - > attr ;
2005-04-17 02:20:36 +04:00
umode_t mode , rmask ;
struct adfs_sb_info * asb = ADFS_SB ( sb ) ;
if ( attr & ADFS_NDA_DIRECTORY ) {
mode = S_IRUGO & asb - > s_owner_mask ;
return S_IFDIR | S_IXUGO | mode ;
}
2019-06-04 16:50:24 +03:00
switch ( adfs_filetype ( ADFS_I ( inode ) - > loadaddr ) ) {
2005-04-17 02:20:36 +04:00
case 0xfc0 : /* LinkFS */
return S_IFLNK | S_IRWXUGO ;
case 0xfe6 : /* UnixExec */
rmask = S_IRUGO | S_IXUGO ;
break ;
default :
rmask = S_IRUGO ;
}
mode = S_IFREG ;
if ( attr & ADFS_NDA_OWNER_READ )
mode | = rmask & asb - > s_owner_mask ;
if ( attr & ADFS_NDA_OWNER_WRITE )
mode | = S_IWUGO & asb - > s_owner_mask ;
if ( attr & ADFS_NDA_PUBLIC_READ )
mode | = rmask & asb - > s_other_mask ;
if ( attr & ADFS_NDA_PUBLIC_WRITE )
mode | = S_IWUGO & asb - > s_other_mask ;
return mode ;
}
/*
* Convert Linux permission to ADFS attribute . We try to do the reverse
* of atts2mode , but there is not a 1 : 1 translation .
*/
2019-12-09 14:08:23 +03:00
static int adfs_mode2atts ( struct super_block * sb , struct inode * inode ,
umode_t ia_mode )
2005-04-17 02:20:36 +04:00
{
2019-12-09 14:08:23 +03:00
struct adfs_sb_info * asb = ADFS_SB ( sb ) ;
2005-04-17 02:20:36 +04:00
umode_t mode ;
int attr ;
/* FIXME: should we be able to alter a link? */
if ( S_ISLNK ( inode - > i_mode ) )
return ADFS_I ( inode ) - > attr ;
2019-12-09 14:08:23 +03:00
/* Directories do not have read/write permissions on the media */
2005-04-17 02:20:36 +04:00
if ( S_ISDIR ( inode - > i_mode ) )
2019-12-09 14:08:23 +03:00
return ADFS_NDA_DIRECTORY ;
2005-04-17 02:20:36 +04:00
2019-12-09 14:08:23 +03:00
attr = 0 ;
mode = ia_mode & asb - > s_owner_mask ;
2005-04-17 02:20:36 +04:00
if ( mode & S_IRUGO )
attr | = ADFS_NDA_OWNER_READ ;
if ( mode & S_IWUGO )
attr | = ADFS_NDA_OWNER_WRITE ;
2019-12-09 14:08:23 +03:00
mode = ia_mode & asb - > s_other_mask ;
2005-04-17 02:20:36 +04:00
mode & = ~ asb - > s_owner_mask ;
if ( mode & S_IRUGO )
attr | = ADFS_NDA_PUBLIC_READ ;
if ( mode & S_IWUGO )
attr | = ADFS_NDA_PUBLIC_WRITE ;
return attr ;
}
2019-12-09 14:08:18 +03:00
static const s64 nsec_unix_epoch_diff_risc_os_epoch = 2208988800000000000LL ;
2005-04-17 02:20:36 +04:00
/*
* Convert an ADFS time to Unix time . ADFS has a 40 - bit centi - second time
2011-03-23 02:35:05 +03:00
* referenced to 1 Jan 1900 ( til 2248 ) so we need to discard 2208988800 seconds
* of time to convert from RISC OS epoch to Unix epoch .
2005-04-17 02:20:36 +04:00
*/
static void
2018-08-22 08:01:09 +03:00
adfs_adfs2unix_time ( struct timespec64 * tv , struct inode * inode )
2005-04-17 02:20:36 +04:00
{
unsigned int high , low ;
2011-03-23 02:35:05 +03:00
/* 01 Jan 1970 00:00:00 (Unix epoch) as nanoseconds since
* 01 Jan 1900 00 : 00 : 00 ( RISC OS epoch )
*/
s64 nsec ;
2005-04-17 02:20:36 +04:00
2019-06-04 16:50:24 +03:00
if ( ! adfs_inode_is_stamped ( inode ) )
2005-04-17 02:20:36 +04:00
goto cur_time ;
2011-03-23 02:35:05 +03:00
high = ADFS_I ( inode ) - > loadaddr & 0xFF ; /* top 8 bits of timestamp */
low = ADFS_I ( inode ) - > execaddr ; /* bottom 32 bits of timestamp */
2005-04-17 02:20:36 +04:00
2011-03-23 02:35:05 +03:00
/* convert 40-bit centi-seconds to 32-bit seconds
* going via nanoseconds to retain precision
*/
nsec = ( ( ( s64 ) high < < 32 ) | ( s64 ) low ) * 10000000 ; /* cs to ns */
2005-04-17 02:20:36 +04:00
/* Files dated pre 01 Jan 1970 00:00:00. */
2011-03-23 02:35:05 +03:00
if ( nsec < nsec_unix_epoch_diff_risc_os_epoch )
2005-04-17 02:20:36 +04:00
goto too_early ;
2011-03-23 02:35:05 +03:00
/* convert from RISC OS to Unix epoch */
nsec - = nsec_unix_epoch_diff_risc_os_epoch ;
2005-04-17 02:20:36 +04:00
2018-08-22 08:01:09 +03:00
* tv = ns_to_timespec64 ( nsec ) ;
2005-04-17 02:20:36 +04:00
return ;
cur_time :
2018-08-22 08:01:09 +03:00
* tv = current_time ( inode ) ;
2005-04-17 02:20:36 +04:00
return ;
too_early :
tv - > tv_sec = tv - > tv_nsec = 0 ;
return ;
}
2019-12-09 14:08:18 +03:00
/* Convert an Unix time to ADFS time for an entry that is already stamped. */
static void adfs_unix2adfs_time ( struct inode * inode ,
const struct timespec64 * ts )
2005-04-17 02:20:36 +04:00
{
2019-12-09 14:08:18 +03:00
s64 cs , nsec = timespec64_to_ns ( ts ) ;
2005-04-17 02:20:36 +04:00
2019-12-09 14:08:18 +03:00
/* convert from Unix to RISC OS epoch */
nsec + = nsec_unix_epoch_diff_risc_os_epoch ;
2005-04-17 02:20:36 +04:00
2019-12-09 14:08:18 +03:00
/* convert from nanoseconds to centiseconds */
cs = div_s64 ( nsec , 10000000 ) ;
cs = clamp_t ( s64 , cs , 0 , 0xffffffffff ) ;
ADFS_I ( inode ) - > loadaddr & = ~ 0xff ;
ADFS_I ( inode ) - > loadaddr | = ( cs > > 32 ) & 0xff ;
ADFS_I ( inode ) - > execaddr = cs ;
2005-04-17 02:20:36 +04:00
}
/*
* Fill in the inode information from the object information .
*
* Note that this is an inode - less filesystem , so we can ' t use the inode
* number to reference the metadata on the media . Instead , we use the
* inode number to hold the object ID , which in turn will tell us where
* the data is held . We also save the parent object ID , and with these
* two , we can locate the metadata .
*
* This does mean that we rely on an objects parent remaining the same at
* all times - we cannot cope with a cross - directory rename ( yet ) .
*/
struct inode *
adfs_iget ( struct super_block * sb , struct object_info * obj )
{
struct inode * inode ;
2023-10-04 21:52:01 +03:00
struct timespec64 ts ;
2005-04-17 02:20:36 +04:00
inode = new_inode ( sb ) ;
if ( ! inode )
goto out ;
inode - > i_uid = ADFS_SB ( sb ) - > s_uid ;
inode - > i_gid = ADFS_SB ( sb ) - > s_gid ;
2019-06-04 16:49:57 +03:00
inode - > i_ino = obj - > indaddr ;
2005-04-17 02:20:36 +04:00
inode - > i_size = obj - > size ;
2011-10-28 16:13:29 +04:00
set_nlink ( inode , 2 ) ;
2005-04-17 02:20:36 +04:00
inode - > i_blocks = ( inode - > i_size + sb - > s_blocksize - 1 ) > >
sb - > s_blocksize_bits ;
/*
* we need to save the parent directory ID so that
* write_inode can update the directory information
* for this file . This will need special handling
* for cross - directory renames .
*/
ADFS_I ( inode ) - > parent_id = obj - > parent_id ;
2019-12-09 14:11:38 +03:00
ADFS_I ( inode ) - > indaddr = obj - > indaddr ;
2005-04-17 02:20:36 +04:00
ADFS_I ( inode ) - > loadaddr = obj - > loadaddr ;
ADFS_I ( inode ) - > execaddr = obj - > execaddr ;
ADFS_I ( inode ) - > attr = obj - > attr ;
inode - > i_mode = adfs_atts2mode ( sb , inode ) ;
2023-10-04 21:52:01 +03:00
adfs_adfs2unix_time ( & ts , inode ) ;
inode_set_atime_to_ts ( inode , ts ) ;
inode_set_mtime_to_ts ( inode , ts ) ;
inode_set_ctime_to_ts ( inode , ts ) ;
2005-04-17 02:20:36 +04:00
if ( S_ISDIR ( inode - > i_mode ) ) {
inode - > i_op = & adfs_dir_inode_operations ;
inode - > i_fop = & adfs_dir_operations ;
} else if ( S_ISREG ( inode - > i_mode ) ) {
inode - > i_op = & adfs_file_inode_operations ;
inode - > i_fop = & adfs_file_operations ;
inode - > i_mapping - > a_ops = & adfs_aops ;
ADFS_I ( inode ) - > mmu_private = inode - > i_size ;
}
2018-06-30 10:15:49 +03:00
inode_fake_hash ( inode ) ;
2005-04-17 02:20:36 +04:00
out :
return inode ;
}
/*
* Validate and convert a changed access mode / time to their ADFS equivalents .
* adfs_write_inode will actually write the information back to the directory
* later .
*/
int
2023-01-13 14:49:11 +03:00
adfs_notify_change ( struct mnt_idmap * idmap , struct dentry * dentry ,
2021-01-21 16:19:43 +03:00
struct iattr * attr )
2005-04-17 02:20:36 +04:00
{
2015-03-18 01:25:59 +03:00
struct inode * inode = d_inode ( dentry ) ;
2005-04-17 02:20:36 +04:00
struct super_block * sb = inode - > i_sb ;
unsigned int ia_valid = attr - > ia_valid ;
int error ;
2023-01-13 14:49:11 +03:00
error = setattr_prepare ( & nop_mnt_idmap , dentry , attr ) ;
2005-04-17 02:20:36 +04:00
/*
* we can ' t change the UID or GID of any file -
* we have a global UID / GID in the superblock
*/
2012-02-08 03:58:38 +04:00
if ( ( ia_valid & ATTR_UID & & ! uid_eq ( attr - > ia_uid , ADFS_SB ( sb ) - > s_uid ) ) | |
( ia_valid & ATTR_GID & & ! gid_eq ( attr - > ia_gid , ADFS_SB ( sb ) - > s_gid ) ) )
2005-04-17 02:20:36 +04:00
error = - EPERM ;
if ( error )
goto out ;
2010-05-26 19:05:34 +04:00
/* XXX: this is missing some actual on-disk truncation.. */
2005-04-17 02:20:36 +04:00
if ( ia_valid & ATTR_SIZE )
2010-06-04 13:30:04 +04:00
truncate_setsize ( inode , attr - > ia_size ) ;
2005-04-17 02:20:36 +04:00
2019-12-09 14:08:18 +03:00
if ( ia_valid & ATTR_MTIME & & adfs_inode_is_stamped ( inode ) ) {
adfs_unix2adfs_time ( inode , & attr - > ia_mtime ) ;
2023-10-04 21:52:01 +03:00
adfs_adfs2unix_time ( & attr - > ia_mtime , inode ) ;
inode_set_mtime_to_ts ( inode , attr - > ia_mtime ) ;
2005-04-17 02:20:36 +04:00
}
2019-12-09 14:08:18 +03:00
2005-04-17 02:20:36 +04:00
/*
* FIXME : should we make these = = to i_mtime since we don ' t
* have the ability to represent them in our filesystem ?
*/
if ( ia_valid & ATTR_ATIME )
2023-10-04 21:52:01 +03:00
inode_set_atime_to_ts ( inode , attr - > ia_atime ) ;
2005-04-17 02:20:36 +04:00
if ( ia_valid & ATTR_CTIME )
2023-07-05 22:00:47 +03:00
inode_set_ctime_to_ts ( inode , attr - > ia_ctime ) ;
2005-04-17 02:20:36 +04:00
if ( ia_valid & ATTR_MODE ) {
2019-12-09 14:08:23 +03:00
ADFS_I ( inode ) - > attr = adfs_mode2atts ( sb , inode , attr - > ia_mode ) ;
2005-04-17 02:20:36 +04:00
inode - > i_mode = adfs_atts2mode ( sb , inode ) ;
}
/*
* FIXME : should we be marking this inode dirty even if
* we don ' t have any metadata to write back ?
*/
if ( ia_valid & ( ATTR_SIZE | ATTR_MTIME | ATTR_MODE ) )
mark_inode_dirty ( inode ) ;
out :
return error ;
}
/*
* write an existing inode back to the directory , and therefore the disk .
* The adfs - specific inode data has already been updated by
* adfs_notify_change ( )
*/
2010-03-05 11:21:37 +03:00
int adfs_write_inode ( struct inode * inode , struct writeback_control * wbc )
2005-04-17 02:20:36 +04:00
{
struct super_block * sb = inode - > i_sb ;
struct object_info obj ;
2019-12-09 14:11:38 +03:00
obj . indaddr = ADFS_I ( inode ) - > indaddr ;
2005-04-17 02:20:36 +04:00
obj . name_len = 0 ;
obj . parent_id = ADFS_I ( inode ) - > parent_id ;
obj . loadaddr = ADFS_I ( inode ) - > loadaddr ;
obj . execaddr = ADFS_I ( inode ) - > execaddr ;
obj . attr = ADFS_I ( inode ) - > attr ;
obj . size = inode - > i_size ;
2022-01-20 05:09:53 +03:00
return adfs_dir_update ( sb , & obj , wbc - > sync_mode = = WB_SYNC_ALL ) ;
2005-04-17 02:20:36 +04:00
}