2018-09-05 01:46:30 +03:00
// SPDX-License-Identifier: GPL-2.0+
2009-04-07 06:01:32 +04:00
/*
2021-11-09 05:35:01 +03:00
* NILFS segment usage file .
2009-04-07 06:01:32 +04:00
*
* Copyright ( C ) 2006 - 2008 Nippon Telegraph and Telephone Corporation .
*
2016-05-24 02:23:09 +03:00
* Written by Koji Sato .
* Revised by Ryusuke Konishi .
2009-04-07 06:01:32 +04:00
*/
# include <linux/kernel.h>
# include <linux/fs.h>
# include <linux/string.h>
# include <linux/buffer_head.h>
# include <linux/errno.h>
# include "mdt.h"
# include "sufile.h"
2015-11-07 03:32:05 +03:00
# include <trace/events/nilfs2.h>
2012-07-31 01:42:10 +04:00
/**
* struct nilfs_sufile_info - on - memory private data of sufile
* @ mi : on - memory private data of metadata file
* @ ncleansegs : number of clean segments
* @ allocmin : lower limit of allocatable segment range
* @ allocmax : upper limit of allocatable segment range
*/
2009-11-12 21:41:55 +03:00
struct nilfs_sufile_info {
struct nilfs_mdt_info mi ;
2011-05-04 20:23:57 +04:00
unsigned long ncleansegs ; /* number of clean segments */
__u64 allocmin ; /* lower limit of allocatable segment range */
__u64 allocmax ; /* upper limit of allocatable segment range */
2009-11-12 21:41:55 +03:00
} ;
static inline struct nilfs_sufile_info * NILFS_SUI ( struct inode * sufile )
{
return ( struct nilfs_sufile_info * ) NILFS_MDT ( sufile ) ;
}
2009-04-07 06:01:32 +04:00
static inline unsigned long
nilfs_sufile_segment_usages_per_block ( const struct inode * sufile )
{
return NILFS_MDT ( sufile ) - > mi_entries_per_block ;
}
static unsigned long
nilfs_sufile_get_blkoff ( const struct inode * sufile , __u64 segnum )
{
__u64 t = segnum + NILFS_MDT ( sufile ) - > mi_first_entry_offset ;
2016-05-24 02:23:25 +03:00
2024-03-06 17:25:47 +03:00
t = div64_ul ( t , nilfs_sufile_segment_usages_per_block ( sufile ) ) ;
2009-04-07 06:01:32 +04:00
return ( unsigned long ) t ;
}
static unsigned long
nilfs_sufile_get_offset ( const struct inode * sufile , __u64 segnum )
{
__u64 t = segnum + NILFS_MDT ( sufile ) - > mi_first_entry_offset ;
2016-05-24 02:23:25 +03:00
2009-04-07 06:01:32 +04:00
return do_div ( t , nilfs_sufile_segment_usages_per_block ( sufile ) ) ;
}
static unsigned long
nilfs_sufile_segment_usages_in_block ( const struct inode * sufile , __u64 curr ,
__u64 max )
{
return min_t ( unsigned long ,
nilfs_sufile_segment_usages_per_block ( sufile ) -
nilfs_sufile_get_offset ( sufile , curr ) ,
max - curr + 1 ) ;
}
static struct nilfs_segment_usage *
nilfs_sufile_block_get_segment_usage ( const struct inode * sufile , __u64 segnum ,
struct buffer_head * bh , void * kaddr )
{
return kaddr + bh_offset ( bh ) +
nilfs_sufile_get_offset ( sufile , segnum ) *
NILFS_MDT ( sufile ) - > mi_entry_size ;
}
static inline int nilfs_sufile_get_header_block ( struct inode * sufile ,
struct buffer_head * * bhp )
{
return nilfs_mdt_get_block ( sufile , 0 , 0 , NULL , bhp ) ;
}
static inline int
nilfs_sufile_get_segment_usage_block ( struct inode * sufile , __u64 segnum ,
int create , struct buffer_head * * bhp )
{
return nilfs_mdt_get_block ( sufile ,
nilfs_sufile_get_blkoff ( sufile , segnum ) ,
create , NULL , bhp ) ;
}
2011-05-04 20:23:58 +04:00
static int nilfs_sufile_delete_segment_usage_block ( struct inode * sufile ,
__u64 segnum )
{
return nilfs_mdt_delete_block ( sufile ,
nilfs_sufile_get_blkoff ( sufile , segnum ) ) ;
}
2009-04-05 13:24:11 +04:00
static void nilfs_sufile_mod_counter ( struct buffer_head * header_bh ,
u64 ncleanadd , u64 ndirtyadd )
{
struct nilfs_sufile_header * header ;
void * kaddr ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2009-04-05 13:24:11 +04:00
header = kaddr + bh_offset ( header_bh ) ;
le64_add_cpu ( & header - > sh_ncleansegs , ncleanadd ) ;
le64_add_cpu ( & header - > sh_ndirtysegs , ndirtyadd ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:24:11 +04:00
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( header_bh ) ;
2009-04-05 13:24:11 +04:00
}
2009-11-13 02:45:32 +03:00
/**
* nilfs_sufile_get_ncleansegs - return the number of clean segments
* @ sufile : inode of segment usage file
*/
unsigned long nilfs_sufile_get_ncleansegs ( struct inode * sufile )
{
return NILFS_SUI ( sufile ) - > ncleansegs ;
}
2009-05-16 16:49:10 +04:00
/**
* nilfs_sufile_updatev - modify multiple segment usages at a time
* @ sufile : inode of segment usage file
* @ segnumv : array of segment numbers
* @ nsegs : size of @ segnumv array
* @ create : creation flag
* @ ndone : place to store number of modified segments on @ segnumv
* @ dofunc : primitive operation for the update
*
* Description : nilfs_sufile_updatev ( ) repeatedly calls @ dofunc
* against the given array of segments . The @ dofunc is called with
* buffers of a header block and the sufile block in which the target
* segment usage entry is contained . If @ ndone is given , the number
* of successfully modified segments from the head is stored in the
* place @ ndone points to .
*
* Return Value : On success , zero is returned . On error , one of the
* following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*
* % - ENOENT - Given segment usage is in hole block ( may be returned if
* @ create is zero )
*
* % - EINVAL - Invalid segment usage number
*/
int nilfs_sufile_updatev ( struct inode * sufile , __u64 * segnumv , size_t nsegs ,
int create , size_t * ndone ,
void ( * dofunc ) ( struct inode * , __u64 ,
struct buffer_head * ,
struct buffer_head * ) )
{
struct buffer_head * header_bh , * bh ;
unsigned long blkoff , prev_blkoff ;
__u64 * seg ;
size_t nerr = 0 , n = 0 ;
int ret = 0 ;
if ( unlikely ( nsegs = = 0 ) )
goto out ;
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
for ( seg = segnumv ; seg < segnumv + nsegs ; seg + + ) {
if ( unlikely ( * seg > = nilfs_sufile_get_nsegments ( sufile ) ) ) {
2020-08-12 04:35:49 +03:00
nilfs_warn ( sufile - > i_sb ,
" %s: invalid segment number: %llu " ,
__func__ , ( unsigned long long ) * seg ) ;
2009-05-16 16:49:10 +04:00
nerr + + ;
}
}
if ( nerr > 0 ) {
ret = - EINVAL ;
goto out_sem ;
}
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out_sem ;
seg = segnumv ;
blkoff = nilfs_sufile_get_blkoff ( sufile , * seg ) ;
ret = nilfs_mdt_get_block ( sufile , blkoff , create , NULL , & bh ) ;
if ( ret < 0 )
goto out_header ;
for ( ; ; ) {
dofunc ( sufile , * seg , header_bh , bh ) ;
if ( + + seg > = segnumv + nsegs )
break ;
prev_blkoff = blkoff ;
blkoff = nilfs_sufile_get_blkoff ( sufile , * seg ) ;
if ( blkoff = = prev_blkoff )
continue ;
/* get different block */
brelse ( bh ) ;
ret = nilfs_mdt_get_block ( sufile , blkoff , create , NULL , & bh ) ;
if ( unlikely ( ret < 0 ) )
goto out_header ;
}
brelse ( bh ) ;
out_header :
n = seg - segnumv ;
brelse ( header_bh ) ;
out_sem :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
out :
if ( ndone )
* ndone = n ;
return ret ;
}
2009-04-05 13:24:11 +04:00
int nilfs_sufile_update ( struct inode * sufile , __u64 segnum , int create ,
void ( * dofunc ) ( struct inode * , __u64 ,
struct buffer_head * ,
struct buffer_head * ) )
{
struct buffer_head * header_bh , * bh ;
int ret ;
if ( unlikely ( segnum > = nilfs_sufile_get_nsegments ( sufile ) ) ) {
2020-08-12 04:35:49 +03:00
nilfs_warn ( sufile - > i_sb , " %s: invalid segment number: %llu " ,
__func__ , ( unsigned long long ) segnum ) ;
2009-04-05 13:24:11 +04:00
return - EINVAL ;
}
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out_sem ;
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , create , & bh ) ;
if ( ! ret ) {
dofunc ( sufile , segnum , header_bh , bh ) ;
brelse ( bh ) ;
}
brelse ( header_bh ) ;
out_sem :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2011-05-04 20:23:57 +04:00
/**
* nilfs_sufile_set_alloc_range - limit range of segment to be allocated
* @ sufile : inode of segment usage file
* @ start : minimum segment number of allocatable region ( inclusive )
* @ end : maximum segment number of allocatable region ( inclusive )
*
* Return Value : On success , 0 is returned . On error , one of the
* following negative error codes is returned .
*
* % - ERANGE - invalid segment region
*/
int nilfs_sufile_set_alloc_range ( struct inode * sufile , __u64 start , __u64 end )
{
struct nilfs_sufile_info * sui = NILFS_SUI ( sufile ) ;
__u64 nsegs ;
int ret = - ERANGE ;
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
nsegs = nilfs_sufile_get_nsegments ( sufile ) ;
if ( start < = end & & end < nsegs ) {
sui - > allocmin = start ;
sui - > allocmax = end ;
ret = 0 ;
}
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-04-07 06:01:32 +04:00
/**
* nilfs_sufile_alloc - allocate a segment
* @ sufile : inode of segment usage file
* @ segnump : pointer to segment number
*
* Description : nilfs_sufile_alloc ( ) allocates a clean segment .
*
* Return Value : On success , 0 is returned and the segment number of the
* allocated segment is stored in the place pointed by @ segnump . On error , one
* of the following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*
* % - ENOSPC - No clean segment left .
*/
int nilfs_sufile_alloc ( struct inode * sufile , __u64 * segnump )
{
struct buffer_head * header_bh , * su_bh ;
struct nilfs_sufile_header * header ;
struct nilfs_segment_usage * su ;
2011-05-04 20:23:57 +04:00
struct nilfs_sufile_info * sui = NILFS_SUI ( sufile ) ;
2009-04-07 06:01:32 +04:00
size_t susz = NILFS_MDT ( sufile ) - > mi_entry_size ;
__u64 segnum , maxsegnum , last_alloc ;
void * kaddr ;
2015-11-07 03:32:14 +03:00
unsigned long nsegments , nsus , cnt ;
2011-05-04 20:23:57 +04:00
int ret , j ;
2009-04-07 06:01:32 +04:00
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out_sem ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2009-11-12 21:10:21 +03:00
header = kaddr + bh_offset ( header_bh ) ;
2009-04-07 06:01:32 +04:00
last_alloc = le64_to_cpu ( header - > sh_last_alloc ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
nsegments = nilfs_sufile_get_nsegments ( sufile ) ;
2011-05-04 20:23:57 +04:00
maxsegnum = sui - > allocmax ;
2009-04-07 06:01:32 +04:00
segnum = last_alloc + 1 ;
2011-05-04 20:23:57 +04:00
if ( segnum < sui - > allocmin | | segnum > sui - > allocmax )
segnum = sui - > allocmin ;
for ( cnt = 0 ; cnt < nsegments ; cnt + = nsus ) {
if ( segnum > maxsegnum ) {
if ( cnt < sui - > allocmax - sui - > allocmin + 1 ) {
/*
* wrap around in the limited region .
* if allocation started from
* sui - > allocmin , this never happens .
*/
segnum = sui - > allocmin ;
maxsegnum = last_alloc ;
} else if ( segnum > sui - > allocmin & &
sui - > allocmax + 1 < nsegments ) {
segnum = sui - > allocmax + 1 ;
maxsegnum = nsegments - 1 ;
} else if ( sui - > allocmin > 0 ) {
segnum = 0 ;
maxsegnum = sui - > allocmin - 1 ;
} else {
break ; /* never happens */
}
2009-04-07 06:01:32 +04:00
}
2015-11-07 03:32:05 +03:00
trace_nilfs2_segment_usage_check ( sufile , segnum , cnt ) ;
2009-04-07 06:01:32 +04:00
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 1 ,
& su_bh ) ;
if ( ret < 0 )
goto out_header ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-07 06:01:32 +04:00
su = nilfs_sufile_block_get_segment_usage (
sufile , segnum , su_bh , kaddr ) ;
nsus = nilfs_sufile_segment_usages_in_block (
sufile , segnum , maxsegnum ) ;
for ( j = 0 ; j < nsus ; j + + , su = ( void * ) su + susz , segnum + + ) {
if ( ! nilfs_segment_usage_clean ( su ) )
continue ;
/* found a clean segment */
nilfs_segment_usage_set_dirty ( su ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2009-11-12 21:10:21 +03:00
header = kaddr + bh_offset ( header_bh ) ;
2009-04-07 06:01:32 +04:00
le64_add_cpu ( & header - > sh_ncleansegs , - 1 ) ;
le64_add_cpu ( & header - > sh_ndirtysegs , 1 ) ;
header - > sh_last_alloc = cpu_to_le64 ( segnum ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
2011-05-04 20:23:57 +04:00
sui - > ncleansegs - - ;
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( header_bh ) ;
mark_buffer_dirty ( su_bh ) ;
2009-04-07 06:01:32 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
brelse ( su_bh ) ;
* segnump = segnum ;
2015-11-07 03:32:05 +03:00
trace_nilfs2_segment_usage_allocated ( sufile , segnum ) ;
2009-04-07 06:01:32 +04:00
goto out_header ;
}
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
brelse ( su_bh ) ;
}
/* no segments left */
ret = - ENOSPC ;
out_header :
brelse ( header_bh ) ;
out_sem :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-04-05 13:24:11 +04:00
void nilfs_sufile_do_cancel_free ( struct inode * sufile , __u64 segnum ,
struct buffer_head * header_bh ,
struct buffer_head * su_bh )
2009-04-07 06:01:32 +04:00
{
struct nilfs_segment_usage * su ;
void * kaddr ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-05 13:24:11 +04:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , su_bh , kaddr ) ;
2009-04-07 06:01:55 +04:00
if ( unlikely ( ! nilfs_segment_usage_clean ( su ) ) ) {
2020-08-12 04:35:49 +03:00
nilfs_warn ( sufile - > i_sb , " %s: segment %llu must be clean " ,
__func__ , ( unsigned long long ) segnum ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:24:11 +04:00
return ;
2009-04-07 06:01:32 +04:00
}
nilfs_segment_usage_set_dirty ( su ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
2009-04-05 13:24:11 +04:00
nilfs_sufile_mod_counter ( header_bh , - 1 , 1 ) ;
2009-11-12 21:41:55 +03:00
NILFS_SUI ( sufile ) - > ncleansegs - - ;
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( su_bh ) ;
2009-04-07 06:01:32 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
}
2009-04-05 13:30:58 +04:00
void nilfs_sufile_do_scrap ( struct inode * sufile , __u64 segnum ,
struct buffer_head * header_bh ,
struct buffer_head * su_bh )
{
struct nilfs_segment_usage * su ;
void * kaddr ;
int clean , dirty ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-05 13:30:58 +04:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , su_bh , kaddr ) ;
2016-08-03 00:05:28 +03:00
if ( su - > su_flags = = cpu_to_le32 ( BIT ( NILFS_SEGMENT_USAGE_DIRTY ) ) & &
2009-04-05 13:30:58 +04:00
su - > su_nblocks = = cpu_to_le32 ( 0 ) ) {
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:30:58 +04:00
return ;
}
clean = nilfs_segment_usage_clean ( su ) ;
dirty = nilfs_segment_usage_dirty ( su ) ;
/* make the segment garbage */
su - > su_lastmod = cpu_to_le64 ( 0 ) ;
su - > su_nblocks = cpu_to_le32 ( 0 ) ;
2016-08-03 00:05:28 +03:00
su - > su_flags = cpu_to_le32 ( BIT ( NILFS_SEGMENT_USAGE_DIRTY ) ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:30:58 +04:00
nilfs_sufile_mod_counter ( header_bh , clean ? ( u64 ) - 1 : 0 , dirty ? 0 : 1 ) ;
2009-11-12 21:41:55 +03:00
NILFS_SUI ( sufile ) - > ncleansegs - = clean ;
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( su_bh ) ;
2009-04-05 13:30:58 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
}
2009-04-05 13:24:11 +04:00
void nilfs_sufile_do_free ( struct inode * sufile , __u64 segnum ,
struct buffer_head * header_bh ,
struct buffer_head * su_bh )
2009-04-07 06:01:32 +04:00
{
struct nilfs_segment_usage * su ;
void * kaddr ;
2009-04-05 13:24:11 +04:00
int sudirty ;
2009-04-07 06:01:32 +04:00
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-05 13:24:11 +04:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , su_bh , kaddr ) ;
if ( nilfs_segment_usage_clean ( su ) ) {
2020-08-12 04:35:49 +03:00
nilfs_warn ( sufile - > i_sb , " %s: segment %llu is already clean " ,
__func__ , ( unsigned long long ) segnum ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:24:11 +04:00
return ;
2009-04-07 06:01:32 +04:00
}
2023-12-07 07:57:30 +03:00
if ( unlikely ( nilfs_segment_usage_error ( su ) ) )
nilfs_warn ( sufile - > i_sb , " free segment %llu marked in error " ,
( unsigned long long ) segnum ) ;
2009-04-07 06:01:32 +04:00
2009-04-05 13:24:11 +04:00
sudirty = nilfs_segment_usage_dirty ( su ) ;
2023-12-07 07:57:30 +03:00
if ( unlikely ( ! sudirty ) )
nilfs_warn ( sufile - > i_sb , " free unallocated segment %llu " ,
( unsigned long long ) segnum ) ;
2009-04-05 13:24:11 +04:00
nilfs_segment_usage_set_clean ( su ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( su_bh ) ;
2009-04-07 06:01:32 +04:00
2009-04-05 13:24:11 +04:00
nilfs_sufile_mod_counter ( header_bh , 1 , sudirty ? ( u64 ) - 1 : 0 ) ;
2009-11-12 21:41:55 +03:00
NILFS_SUI ( sufile ) - > ncleansegs + + ;
2009-04-05 13:24:11 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
2015-11-07 03:32:05 +03:00
trace_nilfs2_segment_usage_freed ( sufile , segnum ) ;
2009-04-07 06:01:32 +04:00
}
2009-11-18 11:25:12 +03:00
/**
* nilfs_sufile_mark_dirty - mark the buffer having a segment usage dirty
* @ sufile : inode of segment usage file
* @ segnum : segment number
*/
int nilfs_sufile_mark_dirty ( struct inode * sufile , __u64 segnum )
{
struct buffer_head * bh ;
2022-11-18 09:33:04 +03:00
void * kaddr ;
struct nilfs_segment_usage * su ;
2009-11-18 11:25:12 +03:00
int ret ;
2022-11-18 09:33:04 +03:00
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
2009-11-18 11:25:12 +03:00
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 0 , & bh ) ;
2023-12-05 11:59:47 +03:00
if ( ret )
goto out_sem ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( bh - > b_page ) ;
2023-12-05 11:59:47 +03:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , bh , kaddr ) ;
if ( unlikely ( nilfs_segment_usage_error ( su ) ) ) {
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2023-12-05 11:59:47 +03:00
brelse ( bh ) ;
if ( nilfs_segment_is_active ( nilfs , segnum ) ) {
nilfs_error ( sufile - > i_sb ,
" active segment %llu is erroneous " ,
( unsigned long long ) segnum ) ;
} else {
/*
* Segments marked erroneous are never allocated by
* nilfs_sufile_alloc ( ) ; only active segments , ie ,
* the segments indexed by ns_segnum or ns_nextnum ,
* can be erroneous here .
*/
WARN_ON_ONCE ( 1 ) ;
}
ret = - EIO ;
} else {
2022-11-18 09:33:04 +03:00
nilfs_segment_usage_set_dirty ( su ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2023-12-05 11:59:47 +03:00
mark_buffer_dirty ( bh ) ;
nilfs_mdt_mark_dirty ( sufile ) ;
2009-11-18 11:25:12 +03:00
brelse ( bh ) ;
}
2023-12-05 11:59:47 +03:00
out_sem :
2022-11-18 09:33:04 +03:00
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
2009-11-18 11:25:12 +03:00
return ret ;
}
2009-11-18 12:23:34 +03:00
/**
* nilfs_sufile_set_segment_usage - set usage of a segment
* @ sufile : inode of segment usage file
* @ segnum : segment number
* @ nblocks : number of live blocks in the segment
* @ modtime : modification time ( option )
*/
int nilfs_sufile_set_segment_usage ( struct inode * sufile , __u64 segnum ,
2018-02-07 02:39:21 +03:00
unsigned long nblocks , time64_t modtime )
2009-11-18 12:23:34 +03:00
{
struct buffer_head * bh ;
struct nilfs_segment_usage * su ;
void * kaddr ;
int ret ;
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 0 , & bh ) ;
if ( ret < 0 )
goto out_sem ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( bh - > b_page ) ;
2009-11-18 12:23:34 +03:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , bh , kaddr ) ;
2023-12-05 11:59:47 +03:00
if ( modtime ) {
/*
* Check segusage error and set su_lastmod only when updating
* this entry with a valid timestamp , not for cancellation .
*/
WARN_ON_ONCE ( nilfs_segment_usage_error ( su ) ) ;
2009-11-18 12:23:34 +03:00
su - > su_lastmod = cpu_to_le64 ( modtime ) ;
2023-12-05 11:59:47 +03:00
}
2009-11-18 12:23:34 +03:00
su - > su_nblocks = cpu_to_le32 ( nblocks ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-11-18 12:23:34 +03:00
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( bh ) ;
2009-11-18 12:23:34 +03:00
nilfs_mdt_mark_dirty ( sufile ) ;
brelse ( bh ) ;
out_sem :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-04-07 06:01:32 +04:00
/**
* nilfs_sufile_get_stat - get segment usage statistics
* @ sufile : inode of segment usage file
2020-10-16 06:13:11 +03:00
* @ sustat : pointer to a structure of segment usage statistics
2009-04-07 06:01:32 +04:00
*
* Description : nilfs_sufile_get_stat ( ) returns information about segment
* usage .
*
* Return Value : On success , 0 is returned , and segment usage information is
2020-10-16 06:13:11 +03:00
* stored in the place pointed by @ sustat . On error , one of the following
2009-04-07 06:01:32 +04:00
* negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*/
int nilfs_sufile_get_stat ( struct inode * sufile , struct nilfs_sustat * sustat )
{
struct buffer_head * header_bh ;
struct nilfs_sufile_header * header ;
2011-05-05 07:56:51 +04:00
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
2009-04-07 06:01:32 +04:00
void * kaddr ;
int ret ;
down_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out_sem ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2009-11-12 21:10:21 +03:00
header = kaddr + bh_offset ( header_bh ) ;
2009-04-07 06:01:32 +04:00
sustat - > ss_nsegs = nilfs_sufile_get_nsegments ( sufile ) ;
sustat - > ss_ncleansegs = le64_to_cpu ( header - > sh_ncleansegs ) ;
sustat - > ss_ndirtysegs = le64_to_cpu ( header - > sh_ndirtysegs ) ;
2009-04-07 06:01:54 +04:00
sustat - > ss_ctime = nilfs - > ns_ctime ;
sustat - > ss_nongc_ctime = nilfs - > ns_nongc_ctime ;
spin_lock ( & nilfs - > ns_last_segment_lock ) ;
sustat - > ss_prot_seq = nilfs - > ns_prot_seq ;
spin_unlock ( & nilfs - > ns_last_segment_lock ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
brelse ( header_bh ) ;
out_sem :
up_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-04-05 13:24:11 +04:00
void nilfs_sufile_do_set_error ( struct inode * sufile , __u64 segnum ,
struct buffer_head * header_bh ,
struct buffer_head * su_bh )
2009-04-07 06:01:32 +04:00
{
struct nilfs_segment_usage * su ;
void * kaddr ;
2009-04-05 13:24:11 +04:00
int suclean ;
2009-04-07 06:01:32 +04:00
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-07 06:01:32 +04:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum , su_bh , kaddr ) ;
if ( nilfs_segment_usage_error ( su ) ) {
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-05 13:24:11 +04:00
return ;
2009-04-07 06:01:32 +04:00
}
2009-04-05 10:03:16 +04:00
suclean = nilfs_segment_usage_clean ( su ) ;
2009-04-07 06:01:32 +04:00
nilfs_segment_usage_set_error ( su ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
2009-11-12 21:41:55 +03:00
if ( suclean ) {
2009-04-05 13:24:11 +04:00
nilfs_sufile_mod_counter ( header_bh , - 1 , 0 ) ;
2009-11-12 21:41:55 +03:00
NILFS_SUI ( sufile ) - > ncleansegs - - ;
}
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( su_bh ) ;
2009-04-07 06:01:32 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
}
2011-05-04 20:23:58 +04:00
/**
2017-11-18 02:29:43 +03:00
* nilfs_sufile_truncate_range - truncate range of segment array
* @ sufile : inode of segment usage file
* @ start : start segment number ( inclusive )
* @ end : end segment number ( inclusive )
*
* Return Value : On success , 0 is returned . On error , one of the
* following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*
* % - EINVAL - Invalid number of segments specified
*
* % - EBUSY - Dirty or active segments are present in the range
*/
2011-05-04 20:23:58 +04:00
static int nilfs_sufile_truncate_range ( struct inode * sufile ,
__u64 start , __u64 end )
{
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
struct buffer_head * header_bh ;
struct buffer_head * su_bh ;
struct nilfs_segment_usage * su , * su2 ;
size_t susz = NILFS_MDT ( sufile ) - > mi_entry_size ;
unsigned long segusages_per_block ;
unsigned long nsegs , ncleaned ;
__u64 segnum ;
void * kaddr ;
ssize_t n , nc ;
int ret ;
int j ;
nsegs = nilfs_sufile_get_nsegments ( sufile ) ;
ret = - EINVAL ;
if ( start > end | | start > = nsegs )
goto out ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out ;
segusages_per_block = nilfs_sufile_segment_usages_per_block ( sufile ) ;
ncleaned = 0 ;
for ( segnum = start ; segnum < = end ; segnum + = n ) {
n = min_t ( unsigned long ,
segusages_per_block -
nilfs_sufile_get_offset ( sufile , segnum ) ,
end - segnum + 1 ) ;
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 0 ,
& su_bh ) ;
if ( ret < 0 ) {
if ( ret ! = - ENOENT )
goto out_header ;
/* hole */
continue ;
}
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2011-05-04 20:23:58 +04:00
su = nilfs_sufile_block_get_segment_usage (
sufile , segnum , su_bh , kaddr ) ;
su2 = su ;
for ( j = 0 ; j < n ; j + + , su = ( void * ) su + susz ) {
if ( ( le32_to_cpu ( su - > su_flags ) &
2016-08-03 00:05:28 +03:00
~ BIT ( NILFS_SEGMENT_USAGE_ERROR ) ) | |
2011-05-04 20:23:58 +04:00
nilfs_segment_is_active ( nilfs , segnum + j ) ) {
ret = - EBUSY ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2011-05-04 20:23:58 +04:00
brelse ( su_bh ) ;
goto out_header ;
}
}
nc = 0 ;
for ( su = su2 , j = 0 ; j < n ; j + + , su = ( void * ) su + susz ) {
if ( nilfs_segment_usage_error ( su ) ) {
nilfs_segment_usage_set_clean ( su ) ;
nc + + ;
}
}
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2011-05-04 20:23:58 +04:00
if ( nc > 0 ) {
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( su_bh ) ;
2011-05-04 20:23:58 +04:00
ncleaned + = nc ;
}
brelse ( su_bh ) ;
if ( n = = segusages_per_block ) {
/* make hole */
nilfs_sufile_delete_segment_usage_block ( sufile , segnum ) ;
}
}
ret = 0 ;
out_header :
if ( ncleaned > 0 ) {
NILFS_SUI ( sufile ) - > ncleansegs + = ncleaned ;
nilfs_sufile_mod_counter ( header_bh , ncleaned , 0 ) ;
nilfs_mdt_mark_dirty ( sufile ) ;
}
brelse ( header_bh ) ;
out :
return ret ;
}
2011-05-04 20:23:58 +04:00
/**
* nilfs_sufile_resize - resize segment array
* @ sufile : inode of segment usage file
* @ newnsegs : new number of segments
*
* Return Value : On success , 0 is returned . On error , one of the
* following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*
* % - ENOSPC - Enough free space is not left for shrinking
*
* % - EBUSY - Dirty or active segments exist in the region to be truncated
*/
int nilfs_sufile_resize ( struct inode * sufile , __u64 newnsegs )
{
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
struct buffer_head * header_bh ;
struct nilfs_sufile_header * header ;
struct nilfs_sufile_info * sui = NILFS_SUI ( sufile ) ;
void * kaddr ;
unsigned long nsegs , nrsvsegs ;
int ret = 0 ;
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
nsegs = nilfs_sufile_get_nsegments ( sufile ) ;
if ( nsegs = = newnsegs )
goto out ;
ret = - ENOSPC ;
nrsvsegs = nilfs_nrsvsegs ( nilfs , newnsegs ) ;
if ( newnsegs < nsegs & & nsegs - newnsegs + nrsvsegs > sui - > ncleansegs )
goto out ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out ;
if ( newnsegs > nsegs ) {
sui - > ncleansegs + = newnsegs - nsegs ;
} else /* newnsegs < nsegs */ {
ret = nilfs_sufile_truncate_range ( sufile , newnsegs , nsegs - 1 ) ;
if ( ret < 0 )
goto out_header ;
sui - > ncleansegs - = nsegs - newnsegs ;
2023-05-24 12:43:48 +03:00
/*
* If the sufile is successfully truncated , immediately adjust
* the segment allocation space while locking the semaphore
* " mi_sem " so that nilfs_sufile_alloc ( ) never allocates
* segments in the truncated space .
*/
sui - > allocmax = newnsegs - 1 ;
sui - > allocmin = 0 ;
2011-05-04 20:23:58 +04:00
}
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2011-05-04 20:23:58 +04:00
header = kaddr + bh_offset ( header_bh ) ;
header - > sh_ncleansegs = cpu_to_le64 ( sui - > ncleansegs ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2011-05-04 20:23:58 +04:00
2011-05-05 07:56:51 +04:00
mark_buffer_dirty ( header_bh ) ;
2011-05-04 20:23:58 +04:00
nilfs_mdt_mark_dirty ( sufile ) ;
nilfs_set_nsegments ( nilfs , newnsegs ) ;
out_header :
brelse ( header_bh ) ;
out :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-04-07 06:01:32 +04:00
/**
* nilfs_sufile_get_suinfo -
* @ sufile : inode of segment usage file
* @ segnum : segment number to start looking
2009-05-11 22:58:47 +04:00
* @ buf : array of suinfo
* @ sisz : byte size of suinfo
2009-04-07 06:01:32 +04:00
* @ nsi : size of suinfo array
*
* Description :
*
* Return Value : On success , 0 is returned and . . . . On error , one of the
* following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*/
2009-05-11 22:58:47 +04:00
ssize_t nilfs_sufile_get_suinfo ( struct inode * sufile , __u64 segnum , void * buf ,
2016-05-24 02:23:03 +03:00
unsigned int sisz , size_t nsi )
2009-04-07 06:01:32 +04:00
{
struct buffer_head * su_bh ;
struct nilfs_segment_usage * su ;
2009-05-11 22:58:47 +04:00
struct nilfs_suinfo * si = buf ;
2009-04-07 06:01:32 +04:00
size_t susz = NILFS_MDT ( sufile ) - > mi_entry_size ;
2011-05-05 07:56:51 +04:00
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
2009-04-07 06:01:32 +04:00
void * kaddr ;
unsigned long nsegs , segusages_per_block ;
ssize_t n ;
int ret , i , j ;
down_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
segusages_per_block = nilfs_sufile_segment_usages_per_block ( sufile ) ;
nsegs = min_t ( unsigned long ,
nilfs_sufile_get_nsegments ( sufile ) - segnum ,
nsi ) ;
for ( i = 0 ; i < nsegs ; i + = n , segnum + = n ) {
n = min_t ( unsigned long ,
segusages_per_block -
nilfs_sufile_get_offset ( sufile , segnum ) ,
nsegs - i ) ;
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 0 ,
& su_bh ) ;
if ( ret < 0 ) {
if ( ret ! = - ENOENT )
goto out ;
/* hole */
2009-05-11 22:58:47 +04:00
memset ( si , 0 , sisz * n ) ;
si = ( void * ) si + sisz * n ;
2009-04-07 06:01:32 +04:00
continue ;
}
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2009-04-07 06:01:32 +04:00
su = nilfs_sufile_block_get_segment_usage (
sufile , segnum , su_bh , kaddr ) ;
2009-05-11 22:58:47 +04:00
for ( j = 0 ; j < n ;
j + + , su = ( void * ) su + susz , si = ( void * ) si + sisz ) {
si - > sui_lastmod = le64_to_cpu ( su - > su_lastmod ) ;
si - > sui_nblocks = le32_to_cpu ( su - > su_nblocks ) ;
si - > sui_flags = le32_to_cpu ( su - > su_flags ) &
2016-08-03 00:05:28 +03:00
~ BIT ( NILFS_SEGMENT_USAGE_ACTIVE ) ;
2009-03-29 19:50:19 +04:00
if ( nilfs_segment_is_active ( nilfs , segnum + j ) )
2009-05-11 22:58:47 +04:00
si - > sui_flags | =
2016-08-03 00:05:28 +03:00
BIT ( NILFS_SEGMENT_USAGE_ACTIVE ) ;
2009-04-07 06:01:32 +04:00
}
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2009-04-07 06:01:32 +04:00
brelse ( su_bh ) ;
}
ret = nsegs ;
out :
up_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2009-11-12 17:56:43 +03:00
2014-04-04 01:50:27 +04:00
/**
* nilfs_sufile_set_suinfo - sets segment usage info
* @ sufile : inode of segment usage file
* @ buf : array of suinfo_update
* @ supsz : byte size of suinfo_update
* @ nsup : size of suinfo_update array
*
* Description : Takes an array of nilfs_suinfo_update structs and updates
* segment usage accordingly . Only the fields indicated by the sup_flags
* are updated .
*
* Return Value : On success , 0 is returned . On error , one of the
* following negative error codes is returned .
*
* % - EIO - I / O error .
*
* % - ENOMEM - Insufficient amount of memory available .
*
* % - EINVAL - Invalid values in input ( segment number , flags or nblocks )
*/
ssize_t nilfs_sufile_set_suinfo ( struct inode * sufile , void * buf ,
2016-05-24 02:23:03 +03:00
unsigned int supsz , size_t nsup )
2014-04-04 01:50:27 +04:00
{
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
struct buffer_head * header_bh , * bh ;
struct nilfs_suinfo_update * sup , * supend = buf + supsz * nsup ;
struct nilfs_segment_usage * su ;
void * kaddr ;
unsigned long blkoff , prev_blkoff ;
int cleansi , cleansu , dirtysi , dirtysu ;
long ncleaned = 0 , ndirtied = 0 ;
int ret = 0 ;
if ( unlikely ( nsup = = 0 ) )
return ret ;
for ( sup = buf ; sup < supend ; sup = ( void * ) sup + supsz ) {
if ( sup - > sup_segnum > = nilfs - > ns_nsegments
| | ( sup - > sup_flags &
( ~ 0UL < < __NR_NILFS_SUINFO_UPDATE_FIELDS ) )
| | ( nilfs_suinfo_update_nblocks ( sup ) & &
sup - > sup_sui . sui_nblocks >
nilfs - > ns_blocks_per_segment ) )
return - EINVAL ;
}
down_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
ret = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( ret < 0 )
goto out_sem ;
sup = buf ;
blkoff = nilfs_sufile_get_blkoff ( sufile , sup - > sup_segnum ) ;
ret = nilfs_mdt_get_block ( sufile , blkoff , 1 , NULL , & bh ) ;
if ( ret < 0 )
goto out_header ;
for ( ; ; ) {
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( bh - > b_page ) ;
2014-04-04 01:50:27 +04:00
su = nilfs_sufile_block_get_segment_usage (
sufile , sup - > sup_segnum , bh , kaddr ) ;
if ( nilfs_suinfo_update_lastmod ( sup ) )
su - > su_lastmod = cpu_to_le64 ( sup - > sup_sui . sui_lastmod ) ;
if ( nilfs_suinfo_update_nblocks ( sup ) )
su - > su_nblocks = cpu_to_le32 ( sup - > sup_sui . sui_nblocks ) ;
if ( nilfs_suinfo_update_flags ( sup ) ) {
/*
* Active flag is a virtual flag projected by running
* nilfs kernel code - drop it not to write it to
* disk .
*/
sup - > sup_sui . sui_flags & =
2016-08-03 00:05:28 +03:00
~ BIT ( NILFS_SEGMENT_USAGE_ACTIVE ) ;
2014-04-04 01:50:27 +04:00
cleansi = nilfs_suinfo_clean ( & sup - > sup_sui ) ;
cleansu = nilfs_segment_usage_clean ( su ) ;
dirtysi = nilfs_suinfo_dirty ( & sup - > sup_sui ) ;
dirtysu = nilfs_segment_usage_dirty ( su ) ;
if ( cleansi & & ! cleansu )
+ + ncleaned ;
else if ( ! cleansi & & cleansu )
- - ncleaned ;
if ( dirtysi & & ! dirtysu )
+ + ndirtied ;
else if ( ! dirtysi & & dirtysu )
- - ndirtied ;
su - > su_flags = cpu_to_le32 ( sup - > sup_sui . sui_flags ) ;
}
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2014-04-04 01:50:27 +04:00
sup = ( void * ) sup + supsz ;
if ( sup > = supend )
break ;
prev_blkoff = blkoff ;
blkoff = nilfs_sufile_get_blkoff ( sufile , sup - > sup_segnum ) ;
if ( blkoff = = prev_blkoff )
continue ;
/* get different block */
mark_buffer_dirty ( bh ) ;
put_bh ( bh ) ;
ret = nilfs_mdt_get_block ( sufile , blkoff , 1 , NULL , & bh ) ;
if ( unlikely ( ret < 0 ) )
goto out_mark ;
}
mark_buffer_dirty ( bh ) ;
put_bh ( bh ) ;
out_mark :
if ( ncleaned | | ndirtied ) {
nilfs_sufile_mod_counter ( header_bh , ( u64 ) ncleaned ,
( u64 ) ndirtied ) ;
NILFS_SUI ( sufile ) - > ncleansegs + = ncleaned ;
}
nilfs_mdt_mark_dirty ( sufile ) ;
out_header :
put_bh ( header_bh ) ;
out_sem :
up_write ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
return ret ;
}
2014-04-04 01:50:29 +04:00
/**
* nilfs_sufile_trim_fs ( ) - trim ioctl handle function
* @ sufile : inode of segment usage file
* @ range : fstrim_range structure
*
* start : First Byte to trim
* len : number of Bytes to trim from start
* minlen : minimum extent length in Bytes
*
* Decription : nilfs_sufile_trim_fs goes through all segments containing bytes
* from start to start + len . start is rounded up to the next block boundary
* and start + len is rounded down . For each clean segment blkdev_issue_discard
* function is invoked .
*
* Return Value : On success , 0 is returned or negative error code , otherwise .
*/
int nilfs_sufile_trim_fs ( struct inode * sufile , struct fstrim_range * range )
{
struct the_nilfs * nilfs = sufile - > i_sb - > s_fs_info ;
struct buffer_head * su_bh ;
struct nilfs_segment_usage * su ;
void * kaddr ;
size_t n , i , susz = NILFS_MDT ( sufile ) - > mi_entry_size ;
sector_t seg_start , seg_end , start_block , end_block ;
sector_t start = 0 , nblocks = 0 ;
u64 segnum , segnum_end , minlen , len , max_blocks , ndiscarded = 0 ;
int ret = 0 ;
unsigned int sects_per_block ;
sects_per_block = ( 1 < < nilfs - > ns_blocksize_bits ) /
bdev_logical_block_size ( nilfs - > ns_bdev ) ;
len = range - > len > > nilfs - > ns_blocksize_bits ;
minlen = range - > minlen > > nilfs - > ns_blocksize_bits ;
max_blocks = ( ( u64 ) nilfs - > ns_nsegments * nilfs - > ns_blocks_per_segment ) ;
if ( ! len | | range - > start > = max_blocks < < nilfs - > ns_blocksize_bits )
return - EINVAL ;
start_block = ( range - > start + nilfs - > ns_blocksize - 1 ) > >
nilfs - > ns_blocksize_bits ;
/*
* range - > len can be very large ( actually , it is set to
* ULLONG_MAX by default ) - truncate upper end of the range
* carefully so as not to overflow .
*/
if ( max_blocks - start_block < len )
end_block = max_blocks - 1 ;
else
end_block = start_block + len - 1 ;
segnum = nilfs_get_segnum_of_block ( nilfs , start_block ) ;
segnum_end = nilfs_get_segnum_of_block ( nilfs , end_block ) ;
down_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
while ( segnum < = segnum_end ) {
n = nilfs_sufile_segment_usages_in_block ( sufile , segnum ,
segnum_end ) ;
ret = nilfs_sufile_get_segment_usage_block ( sufile , segnum , 0 ,
& su_bh ) ;
if ( ret < 0 ) {
if ( ret ! = - ENOENT )
goto out_sem ;
/* hole */
segnum + = n ;
continue ;
}
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2014-04-04 01:50:29 +04:00
su = nilfs_sufile_block_get_segment_usage ( sufile , segnum ,
su_bh , kaddr ) ;
for ( i = 0 ; i < n ; + + i , + + segnum , su = ( void * ) su + susz ) {
if ( ! nilfs_segment_usage_clean ( su ) )
continue ;
nilfs_get_segment_range ( nilfs , segnum , & seg_start ,
& seg_end ) ;
if ( ! nblocks ) {
/* start new extent */
start = seg_start ;
nblocks = seg_end - seg_start + 1 ;
continue ;
}
if ( start + nblocks = = seg_start ) {
/* add to previous extent */
nblocks + = seg_end - seg_start + 1 ;
continue ;
}
/* discard previous extent */
if ( start < start_block ) {
nblocks - = start_block - start ;
start = start_block ;
}
if ( nblocks > = minlen ) {
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2014-04-04 01:50:29 +04:00
ret = blkdev_issue_discard ( nilfs - > ns_bdev ,
start * sects_per_block ,
nblocks * sects_per_block ,
2022-04-15 07:52:57 +03:00
GFP_NOFS ) ;
2014-04-04 01:50:29 +04:00
if ( ret < 0 ) {
put_bh ( su_bh ) ;
goto out_sem ;
}
ndiscarded + = nblocks ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( su_bh - > b_page ) ;
2014-04-04 01:50:29 +04:00
su = nilfs_sufile_block_get_segment_usage (
sufile , segnum , su_bh , kaddr ) ;
}
/* start new extent */
start = seg_start ;
nblocks = seg_end - seg_start + 1 ;
}
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2014-04-04 01:50:29 +04:00
put_bh ( su_bh ) ;
}
if ( nblocks ) {
/* discard last extent */
if ( start < start_block ) {
nblocks - = start_block - start ;
start = start_block ;
}
if ( start + nblocks > end_block + 1 )
nblocks = end_block - start + 1 ;
if ( nblocks > = minlen ) {
ret = blkdev_issue_discard ( nilfs - > ns_bdev ,
start * sects_per_block ,
nblocks * sects_per_block ,
2022-04-15 07:52:57 +03:00
GFP_NOFS ) ;
2014-04-04 01:50:29 +04:00
if ( ! ret )
ndiscarded + = nblocks ;
}
}
out_sem :
up_read ( & NILFS_MDT ( sufile ) - > mi_sem ) ;
range - > len = ndiscarded < < nilfs - > ns_blocksize_bits ;
return ret ;
}
2009-11-12 19:36:56 +03:00
/**
2010-09-05 07:20:59 +04:00
* nilfs_sufile_read - read or get sufile inode
* @ sb : super block instance
* @ susize : size of a segment usage entry
2009-11-12 19:36:56 +03:00
* @ raw_inode : on - disk sufile inode
2010-09-05 07:20:59 +04:00
* @ inodep : buffer to store the inode
2009-11-12 19:36:56 +03:00
*/
2010-09-05 07:20:59 +04:00
int nilfs_sufile_read ( struct super_block * sb , size_t susize ,
struct nilfs_inode * raw_inode , struct inode * * inodep )
2009-11-12 19:36:56 +03:00
{
2010-09-05 07:20:59 +04:00
struct inode * sufile ;
struct nilfs_sufile_info * sui ;
2009-11-12 21:41:55 +03:00
struct buffer_head * header_bh ;
struct nilfs_sufile_header * header ;
void * kaddr ;
2010-09-05 07:20:59 +04:00
int err ;
2009-11-12 21:41:55 +03:00
2014-04-04 01:50:31 +04:00
if ( susize > sb - > s_blocksize ) {
2020-08-12 04:35:49 +03:00
nilfs_err ( sb , " too large segment usage size: %zu bytes " ,
susize ) ;
2014-04-04 01:50:31 +04:00
return - EINVAL ;
} else if ( susize < NILFS_MIN_SEGMENT_USAGE_SIZE ) {
2020-08-12 04:35:49 +03:00
nilfs_err ( sb , " too small segment usage size: %zu bytes " ,
susize ) ;
2014-04-04 01:50:31 +04:00
return - EINVAL ;
}
2010-09-05 07:20:59 +04:00
sufile = nilfs_iget_locked ( sb , NULL , NILFS_SUFILE_INO ) ;
if ( unlikely ( ! sufile ) )
return - ENOMEM ;
if ( ! ( sufile - > i_state & I_NEW ) )
goto out ;
2009-11-12 21:41:55 +03:00
2010-09-05 07:20:59 +04:00
err = nilfs_mdt_init ( sufile , NILFS_MDT_GFP , sizeof ( * sui ) ) ;
if ( err )
goto failed ;
2009-11-12 19:36:56 +03:00
2010-09-05 07:20:59 +04:00
nilfs_mdt_set_entry_size ( sufile , susize ,
sizeof ( struct nilfs_sufile_header ) ) ;
err = nilfs_read_inode_common ( sufile , raw_inode ) ;
if ( err )
goto failed ;
err = nilfs_sufile_get_header_block ( sufile , & header_bh ) ;
if ( err )
goto failed ;
2009-11-12 17:56:43 +03:00
2010-09-05 07:20:59 +04:00
sui = NILFS_SUI ( sufile ) ;
2024-01-22 17:01:52 +03:00
kaddr = kmap_local_page ( header_bh - > b_page ) ;
2010-09-05 07:20:59 +04:00
header = kaddr + bh_offset ( header_bh ) ;
sui - > ncleansegs = le64_to_cpu ( header - > sh_ncleansegs ) ;
2024-01-22 17:01:52 +03:00
kunmap_local ( kaddr ) ;
2010-09-05 07:20:59 +04:00
brelse ( header_bh ) ;
2011-05-04 20:23:57 +04:00
sui - > allocmax = nilfs_sufile_get_nsegments ( sufile ) - 1 ;
sui - > allocmin = 0 ;
2010-09-05 07:20:59 +04:00
unlock_new_inode ( sufile ) ;
out :
* inodep = sufile ;
return 0 ;
failed :
iget_failed ( sufile ) ;
return err ;
2009-11-12 17:56:43 +03:00
}