2019-05-31 11:09:56 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2006-01-16 19:50:04 +03:00
/*
* Copyright ( C ) Sistina Software , Inc . 1997 - 2003 All rights reserved .
2006-05-18 23:09:15 +04:00
* Copyright ( C ) 2004 - 2006 Red Hat , Inc . All rights reserved .
2006-01-16 19:50:04 +03:00
*/
# include <linux/slab.h>
# include <linux/spinlock.h>
# include <linux/completion.h>
# include <linux/buffer_head.h>
# include <linux/xattr.h>
2006-02-28 01:23:27 +03:00
# include <linux/gfs2_ondisk.h>
2013-12-20 17:16:52 +04:00
# include <linux/posix_acl_xattr.h>
2016-12-24 22:46:01 +03:00
# include <linux/uaccess.h>
2006-01-16 19:50:04 +03:00
# include "gfs2.h"
2006-02-28 01:23:27 +03:00
# include "incore.h"
2006-01-16 19:50:04 +03:00
# include "acl.h"
2009-08-26 21:51:04 +04:00
# include "xattr.h"
2006-01-16 19:50:04 +03:00
# include "glock.h"
# include "inode.h"
# include "meta_io.h"
# include "quota.h"
# include "rgrp.h"
2017-08-18 17:15:13 +03:00
# include "super.h"
2006-01-16 19:50:04 +03:00
# include "trans.h"
2006-02-28 01:23:27 +03:00
# include "util.h"
2006-01-16 19:50:04 +03:00
/**
* ea_calc_size - returns the acutal number of bytes the request will take up
* ( not counting any unstuffed data blocks )
* @ sdp :
* @ er :
* @ size :
*
* Returns : 1 if the EA should be stuffed
*/
2009-08-26 21:41:32 +04:00
static int ea_calc_size ( struct gfs2_sbd * sdp , unsigned int nsize , size_t dsize ,
2006-01-16 19:50:04 +03:00
unsigned int * size )
{
2009-08-26 21:41:32 +04:00
unsigned int jbsize = sdp - > sd_jbsize ;
/* Stuffed */
* size = ALIGN ( sizeof ( struct gfs2_ea_header ) + nsize + dsize , 8 ) ;
if ( * size < = jbsize )
2006-01-16 19:50:04 +03:00
return 1 ;
2009-08-26 21:41:32 +04:00
/* Unstuffed */
* size = ALIGN ( sizeof ( struct gfs2_ea_header ) + nsize +
( sizeof ( __be64 ) * DIV_ROUND_UP ( dsize , jbsize ) ) , 8 ) ;
2006-01-16 19:50:04 +03:00
return 0 ;
}
2009-08-26 21:41:32 +04:00
static int ea_check_size ( struct gfs2_sbd * sdp , unsigned int nsize , size_t dsize )
2006-01-16 19:50:04 +03:00
{
unsigned int size ;
2009-08-26 21:41:32 +04:00
if ( dsize > GFS2_EA_MAX_DATA_LEN )
2006-01-16 19:50:04 +03:00
return - ERANGE ;
2009-08-26 21:41:32 +04:00
ea_calc_size ( sdp , nsize , dsize , & size ) ;
2006-01-16 19:50:04 +03:00
/* This can only happen with 512 byte blocks */
if ( size > sdp - > sd_jbsize )
return - ERANGE ;
return 0 ;
}
2006-09-05 21:15:18 +04:00
typedef int ( * ea_call_t ) ( struct gfs2_inode * ip , struct buffer_head * bh ,
2006-01-16 19:50:04 +03:00
struct gfs2_ea_header * ea ,
2006-09-05 21:15:18 +04:00
struct gfs2_ea_header * prev , void * private ) ;
2006-01-16 19:50:04 +03:00
static int ea_foreach_i ( struct gfs2_inode * ip , struct buffer_head * bh ,
ea_call_t ea_call , void * data )
{
struct gfs2_ea_header * ea , * prev = NULL ;
int error = 0 ;
2006-06-14 23:32:57 +04:00
if ( gfs2_metatype_check ( GFS2_SB ( & ip - > i_inode ) , bh , GFS2_METATYPE_EA ) )
2006-01-16 19:50:04 +03:00
return - EIO ;
for ( ea = GFS2_EA_BH2FIRST ( bh ) ; ; prev = ea , ea = GFS2_EA2NEXT ( ea ) ) {
if ( ! GFS2_EA_REC_LEN ( ea ) )
goto fail ;
2006-09-05 21:15:18 +04:00
if ( ! ( bh - > b_data < = ( char * ) ea & & ( char * ) GFS2_EA2NEXT ( ea ) < =
bh - > b_data + bh - > b_size ) )
2006-01-16 19:50:04 +03:00
goto fail ;
if ( ! GFS2_EATYPE_VALID ( ea - > ea_type ) )
goto fail ;
error = ea_call ( ip , bh , ea , prev , data ) ;
if ( error )
return error ;
if ( GFS2_EA_IS_LAST ( ea ) ) {
if ( ( char * ) GFS2_EA2NEXT ( ea ) ! =
bh - > b_data + bh - > b_size )
goto fail ;
break ;
}
}
return error ;
2006-09-04 20:04:26 +04:00
fail :
2006-01-16 19:50:04 +03:00
gfs2_consist_inode ( ip ) ;
return - EIO ;
}
static int ea_foreach ( struct gfs2_inode * ip , ea_call_t ea_call , void * data )
{
struct buffer_head * bh , * eabh ;
2006-10-14 18:46:30 +04:00
__be64 * eablk , * end ;
2006-01-16 19:50:04 +03:00
int error ;
2015-11-12 00:00:35 +03:00
error = gfs2_meta_read ( ip - > i_gl , ip - > i_eattr , DIO_WAIT , 0 , & bh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2008-11-04 13:05:22 +03:00
if ( ! ( ip - > i_diskflags & GFS2_DIF_EA_INDIRECT ) ) {
2006-01-16 19:50:04 +03:00
error = ea_foreach_i ( ip , bh , ea_call , data ) ;
goto out ;
}
2006-06-14 23:32:57 +04:00
if ( gfs2_metatype_check ( GFS2_SB ( & ip - > i_inode ) , bh , GFS2_METATYPE_IN ) ) {
2006-01-16 19:50:04 +03:00
error = - EIO ;
goto out ;
}
2006-10-14 18:46:30 +04:00
eablk = ( __be64 * ) ( bh - > b_data + sizeof ( struct gfs2_meta_header ) ) ;
2006-06-14 23:32:57 +04:00
end = eablk + GFS2_SB ( & ip - > i_inode ) - > sd_inptrs ;
2006-01-16 19:50:04 +03:00
for ( ; eablk < end ; eablk + + ) {
2006-09-04 20:49:07 +04:00
u64 bn ;
2006-01-16 19:50:04 +03:00
if ( ! * eablk )
break ;
bn = be64_to_cpu ( * eablk ) ;
2015-11-12 00:00:35 +03:00
error = gfs2_meta_read ( ip - > i_gl , bn , DIO_WAIT , 0 , & eabh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
break ;
error = ea_foreach_i ( ip , eabh , ea_call , data ) ;
brelse ( eabh ) ;
if ( error )
break ;
}
2006-09-04 20:04:26 +04:00
out :
2006-01-16 19:50:04 +03:00
brelse ( bh ) ;
return error ;
}
struct ea_find {
2009-08-26 21:41:32 +04:00
int type ;
const char * name ;
size_t namel ;
2006-01-16 19:50:04 +03:00
struct gfs2_ea_location * ef_el ;
} ;
static int ea_find_i ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea , struct gfs2_ea_header * prev ,
void * private )
{
struct ea_find * ef = private ;
if ( ea - > ea_type = = GFS2_EATYPE_UNUSED )
return 0 ;
2009-08-26 21:41:32 +04:00
if ( ea - > ea_type = = ef - > type ) {
if ( ea - > ea_name_len = = ef - > namel & &
! memcmp ( GFS2_EA2NAME ( ea ) , ef - > name , ea - > ea_name_len ) ) {
2006-01-16 19:50:04 +03:00
struct gfs2_ea_location * el = ef - > ef_el ;
get_bh ( bh ) ;
el - > el_bh = bh ;
el - > el_ea = ea ;
el - > el_prev = prev ;
return 1 ;
}
}
return 0 ;
}
2009-10-02 15:00:00 +04:00
static int gfs2_ea_find ( struct gfs2_inode * ip , int type , const char * name ,
struct gfs2_ea_location * el )
2006-01-16 19:50:04 +03:00
{
struct ea_find ef ;
int error ;
2009-08-26 21:41:32 +04:00
ef . type = type ;
ef . name = name ;
ef . namel = strlen ( name ) ;
2006-01-16 19:50:04 +03:00
ef . ef_el = el ;
memset ( el , 0 , sizeof ( struct gfs2_ea_location ) ) ;
error = ea_foreach ( ip , ea_find_i , & ef ) ;
if ( error > 0 )
return 0 ;
return error ;
}
/**
* ea_dealloc_unstuffed -
* @ ip :
* @ bh :
* @ ea :
* @ prev :
* @ private :
*
* Take advantage of the fact that all unstuffed blocks are
* allocated from the same RG . But watch , this may not always
* be true .
*
* Returns : errno
*/
static int ea_dealloc_unstuffed ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea ,
struct gfs2_ea_header * prev , void * private )
{
int * leave = private ;
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct gfs2_rgrpd * rgd ;
struct gfs2_holder rg_gh ;
2006-10-14 18:46:30 +04:00
__be64 * dataptrs ;
u64 bn = 0 ;
2006-09-04 20:49:07 +04:00
u64 bstart = 0 ;
2006-01-16 19:50:04 +03:00
unsigned int blen = 0 ;
unsigned int blks = 0 ;
unsigned int x ;
int error ;
2012-04-05 06:11:16 +04:00
error = gfs2_rindex_update ( sdp ) ;
if ( error )
return error ;
2006-01-16 19:50:04 +03:00
if ( GFS2_EA_IS_STUFFED ( ea ) )
return 0 ;
dataptrs = GFS2_EA2DATAPTRS ( ea ) ;
2006-09-05 21:15:18 +04:00
for ( x = 0 ; x < ea - > ea_num_ptrs ; x + + , dataptrs + + ) {
2006-01-16 19:50:04 +03:00
if ( * dataptrs ) {
blks + + ;
bn = be64_to_cpu ( * dataptrs ) ;
}
2006-09-05 21:15:18 +04:00
}
2006-01-16 19:50:04 +03:00
if ( ! blks )
return 0 ;
2012-02-08 16:58:32 +04:00
rgd = gfs2_blk2rgrpd ( sdp , bn , 1 ) ;
2006-01-16 19:50:04 +03:00
if ( ! rgd ) {
gfs2_consist_inode ( ip ) ;
return - EIO ;
}
error = gfs2_glock_nq_init ( rgd - > rd_gl , LM_ST_EXCLUSIVE , 0 , & rg_gh ) ;
if ( error )
return error ;
2007-06-01 17:11:58 +04:00
error = gfs2_trans_begin ( sdp , rgd - > rd_length + RES_DINODE +
2006-09-05 21:15:18 +04:00
RES_EATTR + RES_STATFS + RES_QUOTA , blks ) ;
2006-01-16 19:50:04 +03:00
if ( error )
goto out_gunlock ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , bh ) ;
2006-01-16 19:50:04 +03:00
dataptrs = GFS2_EA2DATAPTRS ( ea ) ;
for ( x = 0 ; x < ea - > ea_num_ptrs ; x + + , dataptrs + + ) {
if ( ! * dataptrs )
break ;
bn = be64_to_cpu ( * dataptrs ) ;
if ( bstart + blen = = bn )
blen + + ;
else {
if ( bstart )
2018-10-04 17:36:02 +03:00
gfs2_free_meta ( ip , rgd , bstart , blen ) ;
2006-01-16 19:50:04 +03:00
bstart = bn ;
blen = 1 ;
}
* dataptrs = 0 ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , - 1 ) ;
2006-01-16 19:50:04 +03:00
}
if ( bstart )
2018-10-04 17:36:02 +03:00
gfs2_free_meta ( ip , rgd , bstart , blen ) ;
2006-01-16 19:50:04 +03:00
if ( prev & & ! leave ) {
2006-09-04 20:49:07 +04:00
u32 len ;
2006-01-16 19:50:04 +03:00
len = GFS2_EA_REC_LEN ( prev ) + GFS2_EA_REC_LEN ( ea ) ;
prev - > ea_rec_len = cpu_to_be32 ( len ) ;
if ( GFS2_EA_IS_LAST ( ea ) )
prev - > ea_flags | = GFS2_EAFLAG_LAST ;
} else {
ea - > ea_type = GFS2_EATYPE_UNUSED ;
ea - > ea_num_ptrs = 0 ;
}
2017-10-04 17:21:19 +03:00
ip - > i_inode . i_ctime = current_time ( & ip - > i_inode ) ;
2018-02-21 18:54:46 +03:00
__mark_inode_dirty ( & ip - > i_inode , I_DIRTY_DATASYNC ) ;
2006-01-16 19:50:04 +03:00
gfs2_trans_end ( sdp ) ;
2006-09-04 20:04:26 +04:00
out_gunlock :
2006-01-16 19:50:04 +03:00
gfs2_glock_dq_uninit ( & rg_gh ) ;
return error ;
}
static int ea_remove_unstuffed ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea ,
struct gfs2_ea_header * prev , int leave )
{
int error ;
2012-07-19 16:12:40 +04:00
error = gfs2_rindex_update ( GFS2_SB ( & ip - > i_inode ) ) ;
if ( error )
return error ;
2013-02-01 05:49:26 +04:00
error = gfs2_quota_hold ( ip , NO_UID_QUOTA_CHANGE , NO_GID_QUOTA_CHANGE ) ;
2006-01-16 19:50:04 +03:00
if ( error )
goto out_alloc ;
2006-09-05 21:15:18 +04:00
error = ea_dealloc_unstuffed ( ip , bh , ea , prev , ( leave ) ? & error : NULL ) ;
2006-01-16 19:50:04 +03:00
gfs2_quota_unhold ( ip ) ;
2006-09-04 20:04:26 +04:00
out_alloc :
2006-01-16 19:50:04 +03:00
return error ;
}
struct ea_list {
struct gfs2_ea_request * ei_er ;
unsigned int ei_size ;
} ;
static int ea_list_i ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea , struct gfs2_ea_header * prev ,
void * private )
{
struct ea_list * ei = private ;
struct gfs2_ea_request * er = ei - > ei_er ;
2018-08-03 12:57:52 +03:00
unsigned int ea_size ;
char * prefix ;
unsigned int l ;
2006-01-16 19:50:04 +03:00
if ( ea - > ea_type = = GFS2_EATYPE_UNUSED )
return 0 ;
2018-08-03 12:57:52 +03:00
switch ( ea - > ea_type ) {
case GFS2_EATYPE_USR :
prefix = " user. " ;
l = 5 ;
break ;
case GFS2_EATYPE_SYS :
prefix = " system. " ;
l = 7 ;
break ;
case GFS2_EATYPE_SECURITY :
prefix = " security. " ;
l = 9 ;
break ;
default :
BUG ( ) ;
}
2006-01-16 19:50:04 +03:00
2018-08-03 12:57:52 +03:00
ea_size = l + ea - > ea_name_len + 1 ;
if ( er - > er_data_len ) {
2006-01-16 19:50:04 +03:00
if ( ei - > ei_size + ea_size > er - > er_data_len )
return - ERANGE ;
2006-05-22 18:36:25 +04:00
memcpy ( er - > er_data + ei - > ei_size , prefix , l ) ;
memcpy ( er - > er_data + ei - > ei_size + l , GFS2_EA2NAME ( ea ) ,
2006-01-16 19:50:04 +03:00
ea - > ea_name_len ) ;
2018-08-03 12:57:52 +03:00
er - > er_data [ ei - > ei_size + ea_size - 1 ] = 0 ;
2006-01-16 19:50:04 +03:00
}
ei - > ei_size + = ea_size ;
return 0 ;
}
/**
2009-08-26 21:41:32 +04:00
* gfs2_listxattr - List gfs2 extended attributes
* @ dentry : The dentry whose inode we are interested in
* @ buffer : The buffer to write the results
* @ size : The size of the buffer
2006-01-16 19:50:04 +03:00
*
* Returns : actual size of data on success , - errno on error
*/
2009-08-26 21:41:32 +04:00
ssize_t gfs2_listxattr ( struct dentry * dentry , char * buffer , size_t size )
2006-01-16 19:50:04 +03:00
{
2015-03-18 01:25:59 +03:00
struct gfs2_inode * ip = GFS2_I ( d_inode ( dentry ) ) ;
2009-08-26 21:41:32 +04:00
struct gfs2_ea_request er ;
2006-01-16 19:50:04 +03:00
struct gfs2_holder i_gh ;
int error ;
2009-08-26 21:41:32 +04:00
memset ( & er , 0 , sizeof ( struct gfs2_ea_request ) ) ;
if ( size ) {
er . er_data = buffer ;
er . er_data_len = size ;
2006-01-16 19:50:04 +03:00
}
2006-09-05 21:15:18 +04:00
error = gfs2_glock_nq_init ( ip - > i_gl , LM_ST_SHARED , LM_FLAG_ANY , & i_gh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2008-11-03 17:28:42 +03:00
if ( ip - > i_eattr ) {
2009-08-26 21:41:32 +04:00
struct ea_list ei = { . ei_er = & er , . ei_size = 0 } ;
2006-01-16 19:50:04 +03:00
error = ea_foreach ( ip , ea_list_i , & ei ) ;
if ( ! error )
error = ei . ei_size ;
}
gfs2_glock_dq_uninit ( & i_gh ) ;
return error ;
}
/**
2012-07-26 14:26:36 +04:00
* ea_iter_unstuffed - copies the unstuffed xattr data to / from the
* request buffer
2006-09-05 21:15:18 +04:00
* @ ip : The GFS2 inode
* @ ea : The extended attribute header structure
2012-07-26 14:26:36 +04:00
* @ din : The data to be copied in
* @ dout : The data to be copied out ( one of din , dout will be NULL )
2006-01-16 19:50:04 +03:00
*
* Returns : errno
*/
2012-07-26 14:26:36 +04:00
static int gfs2_iter_unstuffed ( struct gfs2_inode * ip , struct gfs2_ea_header * ea ,
const char * din , char * dout )
2006-01-16 19:50:04 +03:00
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct buffer_head * * bh ;
unsigned int amount = GFS2_EA_DATA_LEN ( ea ) ;
2006-02-28 01:23:27 +03:00
unsigned int nptrs = DIV_ROUND_UP ( amount , sdp - > sd_jbsize ) ;
2006-10-14 18:46:30 +04:00
__be64 * dataptrs = GFS2_EA2DATAPTRS ( ea ) ;
2006-01-16 19:50:04 +03:00
unsigned int x ;
int error = 0 ;
2012-07-26 14:26:36 +04:00
unsigned char * pos ;
unsigned cp_size ;
2006-01-16 19:50:04 +03:00
2008-04-09 17:33:41 +04:00
bh = kcalloc ( nptrs , sizeof ( struct buffer_head * ) , GFP_NOFS ) ;
2006-01-16 19:50:04 +03:00
if ( ! bh )
return - ENOMEM ;
for ( x = 0 ; x < nptrs ; x + + ) {
2015-11-12 00:00:35 +03:00
error = gfs2_meta_read ( ip - > i_gl , be64_to_cpu ( * dataptrs ) , 0 , 0 ,
2006-09-22 01:05:23 +04:00
bh + x ) ;
2006-01-16 19:50:04 +03:00
if ( error ) {
while ( x - - )
brelse ( bh [ x ] ) ;
goto out ;
}
dataptrs + + ;
}
for ( x = 0 ; x < nptrs ; x + + ) {
2006-09-22 01:05:23 +04:00
error = gfs2_meta_wait ( sdp , bh [ x ] ) ;
2006-01-16 19:50:04 +03:00
if ( error ) {
for ( ; x < nptrs ; x + + )
brelse ( bh [ x ] ) ;
goto out ;
}
if ( gfs2_metatype_check ( sdp , bh [ x ] , GFS2_METATYPE_ED ) ) {
for ( ; x < nptrs ; x + + )
brelse ( bh [ x ] ) ;
error = - EIO ;
goto out ;
}
2012-07-26 14:26:36 +04:00
pos = bh [ x ] - > b_data + sizeof ( struct gfs2_meta_header ) ;
cp_size = ( sdp - > sd_jbsize > amount ) ? amount : sdp - > sd_jbsize ;
2006-01-16 19:50:04 +03:00
2012-07-26 14:26:36 +04:00
if ( dout ) {
memcpy ( dout , pos , cp_size ) ;
dout + = sdp - > sd_jbsize ;
}
if ( din ) {
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , bh [ x ] ) ;
2012-07-26 14:26:36 +04:00
memcpy ( pos , din , cp_size ) ;
din + = sdp - > sd_jbsize ;
}
2006-01-16 19:50:04 +03:00
2012-07-26 14:26:36 +04:00
amount - = sdp - > sd_jbsize ;
2006-01-16 19:50:04 +03:00
brelse ( bh [ x ] ) ;
}
2006-09-04 20:04:26 +04:00
out :
2006-01-16 19:50:04 +03:00
kfree ( bh ) ;
return error ;
}
2009-10-02 15:00:00 +04:00
static int gfs2_ea_get_copy ( struct gfs2_inode * ip , struct gfs2_ea_location * el ,
char * data , size_t size )
2006-01-16 19:50:04 +03:00
{
2009-08-26 21:41:32 +04:00
int ret ;
size_t len = GFS2_EA_DATA_LEN ( el - > el_ea ) ;
if ( len > size )
return - ERANGE ;
2006-01-16 19:50:04 +03:00
if ( GFS2_EA_IS_STUFFED ( el - > el_ea ) ) {
2009-08-26 21:41:32 +04:00
memcpy ( data , GFS2_EA2DATA ( el - > el_ea ) , len ) ;
return len ;
}
2012-07-26 14:26:36 +04:00
ret = gfs2_iter_unstuffed ( ip , el - > el_ea , NULL , data ) ;
2009-08-26 21:41:32 +04:00
if ( ret < 0 )
return ret ;
return len ;
2006-01-16 19:50:04 +03:00
}
2009-10-02 15:00:00 +04:00
int gfs2_xattr_acl_get ( struct gfs2_inode * ip , const char * name , char * * ppdata )
{
struct gfs2_ea_location el ;
int error ;
int len ;
char * data ;
error = gfs2_ea_find ( ip , GFS2_EATYPE_SYS , name , & el ) ;
if ( error )
return error ;
if ( ! el . el_ea )
goto out ;
if ( ! GFS2_EA_DATA_LEN ( el . el_ea ) )
goto out ;
len = GFS2_EA_DATA_LEN ( el . el_ea ) ;
data = kmalloc ( len , GFP_NOFS ) ;
error = - ENOMEM ;
if ( data = = NULL )
goto out ;
error = gfs2_ea_get_copy ( ip , & el , data , len ) ;
2011-11-09 16:54:43 +04:00
if ( error < 0 )
kfree ( data ) ;
else
* ppdata = data ;
2009-10-02 15:00:00 +04:00
out :
brelse ( el . el_bh ) ;
return error ;
}
2006-01-16 19:50:04 +03:00
/**
2009-08-26 21:41:32 +04:00
* gfs2_xattr_get - Get a GFS2 extended attribute
* @ inode : The inode
* @ name : The name of the extended attribute
* @ buffer : The buffer to write the result into
* @ size : The size of the buffer
2009-11-13 12:52:56 +03:00
* @ type : The type of extended attribute
2006-01-16 19:50:04 +03:00
*
* Returns : actual size of data on success , - errno on error
*/
2016-05-13 04:59:17 +03:00
static int __gfs2_xattr_get ( struct inode * inode , const char * name ,
void * buffer , size_t size , int type )
2006-01-16 19:50:04 +03:00
{
2016-04-11 03:48:24 +03:00
struct gfs2_inode * ip = GFS2_I ( inode ) ;
2006-01-16 19:50:04 +03:00
struct gfs2_ea_location el ;
int error ;
2008-11-03 17:28:42 +03:00
if ( ! ip - > i_eattr )
2006-01-16 19:50:04 +03:00
return - ENODATA ;
2009-08-26 21:41:32 +04:00
if ( strlen ( name ) > GFS2_EA_MAX_NAME_LEN )
return - EINVAL ;
2006-01-16 19:50:04 +03:00
2009-08-26 21:41:32 +04:00
error = gfs2_ea_find ( ip , type , name , & el ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
if ( ! el . el_ea )
return - ENODATA ;
2009-09-14 12:50:57 +04:00
if ( size )
2009-08-26 21:41:32 +04:00
error = gfs2_ea_get_copy ( ip , & el , buffer , size ) ;
else
2006-01-16 19:50:04 +03:00
error = GFS2_EA_DATA_LEN ( el . el_ea ) ;
brelse ( el . el_bh ) ;
return error ;
}
2016-05-13 04:59:17 +03:00
static int gfs2_xattr_get ( const struct xattr_handler * handler ,
struct dentry * unused , struct inode * inode ,
const char * name , void * buffer , size_t size )
{
struct gfs2_inode * ip = GFS2_I ( inode ) ;
struct gfs2_holder gh ;
int ret ;
/* During lookup, SELinux calls this function with the glock locked. */
if ( ! gfs2_glock_is_locked_by_me ( ip - > i_gl ) ) {
ret = gfs2_glock_nq_init ( ip - > i_gl , LM_ST_SHARED , LM_FLAG_ANY , & gh ) ;
if ( ret )
return ret ;
2017-10-13 01:39:38 +03:00
} else {
gfs2_holder_mark_uninitialized ( & gh ) ;
2016-05-13 04:59:17 +03:00
}
ret = __gfs2_xattr_get ( inode , name , buffer , size , handler - > flags ) ;
2017-10-13 01:39:38 +03:00
if ( gfs2_holder_initialized ( & gh ) )
2016-05-13 04:59:17 +03:00
gfs2_glock_dq_uninit ( & gh ) ;
return ret ;
}
2006-01-16 19:50:04 +03:00
/**
* ea_alloc_blk - allocates a new block for extended attributes .
* @ ip : A pointer to the inode that ' s getting extended attributes
2006-09-05 21:15:18 +04:00
* @ bhp : Pointer to pointer to a struct buffer_head
2006-01-16 19:50:04 +03:00
*
* Returns : errno
*/
static int ea_alloc_blk ( struct gfs2_inode * ip , struct buffer_head * * bhp )
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct gfs2_ea_header * ea ;
2008-02-06 13:11:15 +03:00
unsigned int n = 1 ;
2006-09-04 20:49:07 +04:00
u64 block ;
2009-05-20 13:48:47 +04:00
int error ;
2006-01-16 19:50:04 +03:00
2011-11-18 19:58:32 +04:00
error = gfs2_alloc_blocks ( ip , & block , & n , 0 , NULL ) ;
2009-05-20 13:48:47 +04:00
if ( error )
return error ;
2019-04-05 14:18:23 +03:00
gfs2_trans_remove_revoke ( sdp , block , 1 ) ;
2006-01-16 19:50:04 +03:00
* bhp = gfs2_meta_new ( ip - > i_gl , block ) ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , * bhp ) ;
2006-01-16 19:50:04 +03:00
gfs2_metatype_set ( * bhp , GFS2_METATYPE_EA , GFS2_FORMAT_EA ) ;
gfs2_buffer_clear_tail ( * bhp , sizeof ( struct gfs2_meta_header ) ) ;
ea = GFS2_EA_BH2FIRST ( * bhp ) ;
ea - > ea_rec_len = cpu_to_be32 ( sdp - > sd_jbsize ) ;
ea - > ea_type = GFS2_EATYPE_UNUSED ;
ea - > ea_flags = GFS2_EAFLAG_LAST ;
ea - > ea_num_ptrs = 0 ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , 1 ) ;
2006-01-16 19:50:04 +03:00
return 0 ;
}
/**
* ea_write - writes the request info to an ea , creating new blocks if
* necessary
2006-09-05 21:15:18 +04:00
* @ ip : inode that is being modified
* @ ea : the location of the new ea in a block
2006-01-16 19:50:04 +03:00
* @ er : the write request
*
* Note : does not update ea_rec_len or the GFS2_EAFLAG_LAST bin of ea_flags
*
* returns : errno
*/
static int ea_write ( struct gfs2_inode * ip , struct gfs2_ea_header * ea ,
struct gfs2_ea_request * er )
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2009-05-20 13:48:47 +04:00
int error ;
2006-01-16 19:50:04 +03:00
ea - > ea_data_len = cpu_to_be32 ( er - > er_data_len ) ;
ea - > ea_name_len = er - > er_name_len ;
ea - > ea_type = er - > er_type ;
ea - > __pad = 0 ;
memcpy ( GFS2_EA2NAME ( ea ) , er - > er_name , er - > er_name_len ) ;
if ( GFS2_EAREQ_SIZE_STUFFED ( er ) < = sdp - > sd_jbsize ) {
ea - > ea_num_ptrs = 0 ;
memcpy ( GFS2_EA2DATA ( ea ) , er - > er_data , er - > er_data_len ) ;
} else {
2006-10-14 18:46:30 +04:00
__be64 * dataptr = GFS2_EA2DATAPTRS ( ea ) ;
2006-01-16 19:50:04 +03:00
const char * data = er - > er_data ;
unsigned int data_len = er - > er_data_len ;
unsigned int copy ;
unsigned int x ;
2006-02-28 01:23:27 +03:00
ea - > ea_num_ptrs = DIV_ROUND_UP ( er - > er_data_len , sdp - > sd_jbsize ) ;
2006-01-16 19:50:04 +03:00
for ( x = 0 ; x < ea - > ea_num_ptrs ; x + + ) {
struct buffer_head * bh ;
2006-09-04 20:49:07 +04:00
u64 block ;
2006-01-16 19:50:04 +03:00
int mh_size = sizeof ( struct gfs2_meta_header ) ;
2008-02-06 13:11:15 +03:00
unsigned int n = 1 ;
2006-01-16 19:50:04 +03:00
2011-11-18 19:58:32 +04:00
error = gfs2_alloc_blocks ( ip , & block , & n , 0 , NULL ) ;
2009-05-20 13:48:47 +04:00
if ( error )
return error ;
2019-04-05 14:18:23 +03:00
gfs2_trans_remove_revoke ( sdp , block , 1 ) ;
2006-01-16 19:50:04 +03:00
bh = gfs2_meta_new ( ip - > i_gl , block ) ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , bh ) ;
2006-01-16 19:50:04 +03:00
gfs2_metatype_set ( bh , GFS2_METATYPE_ED , GFS2_FORMAT_ED ) ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , 1 ) ;
2006-01-16 19:50:04 +03:00
2006-09-05 21:15:18 +04:00
copy = data_len > sdp - > sd_jbsize ? sdp - > sd_jbsize :
data_len ;
2006-01-16 19:50:04 +03:00
memcpy ( bh - > b_data + mh_size , data , copy ) ;
if ( copy < sdp - > sd_jbsize )
memset ( bh - > b_data + mh_size + copy , 0 ,
sdp - > sd_jbsize - copy ) ;
2006-09-05 21:15:18 +04:00
* dataptr + + = cpu_to_be64 ( bh - > b_blocknr ) ;
2006-01-16 19:50:04 +03:00
data + = copy ;
data_len - = copy ;
brelse ( bh ) ;
}
gfs2_assert_withdraw ( sdp , ! data_len ) ;
}
return 0 ;
}
typedef int ( * ea_skeleton_call_t ) ( struct gfs2_inode * ip ,
2006-09-05 21:15:18 +04:00
struct gfs2_ea_request * er , void * private ) ;
2006-01-16 19:50:04 +03:00
static int ea_alloc_skeleton ( struct gfs2_inode * ip , struct gfs2_ea_request * er ,
unsigned int blks ,
2006-09-05 21:15:18 +04:00
ea_skeleton_call_t skeleton_call , void * private )
2006-01-16 19:50:04 +03:00
{
2013-10-02 14:13:25 +04:00
struct gfs2_alloc_parms ap = { . target = blks } ;
2006-01-16 19:50:04 +03:00
int error ;
2012-07-19 16:12:40 +04:00
error = gfs2_rindex_update ( GFS2_SB ( & ip - > i_inode ) ) ;
if ( error )
return error ;
2015-03-18 20:03:41 +03:00
error = gfs2_quota_lock_check ( ip , & ap ) ;
2006-01-16 19:50:04 +03:00
if ( error )
2012-05-18 17:28:23 +04:00
return error ;
2006-01-16 19:50:04 +03:00
2013-10-02 14:13:25 +04:00
error = gfs2_inplace_reserve ( ip , & ap ) ;
2006-01-16 19:50:04 +03:00
if ( error )
goto out_gunlock_q ;
2006-06-14 23:32:57 +04:00
error = gfs2_trans_begin ( GFS2_SB ( & ip - > i_inode ) ,
2012-07-30 17:53:19 +04:00
blks + gfs2_rg_blocks ( ip , blks ) +
2006-01-16 19:50:04 +03:00
RES_DINODE + RES_STATFS + RES_QUOTA , 0 ) ;
if ( error )
goto out_ipres ;
error = skeleton_call ( ip , er , private ) ;
if ( error )
goto out_end_trans ;
2017-10-04 17:21:19 +03:00
ip - > i_inode . i_ctime = current_time ( & ip - > i_inode ) ;
2018-02-21 18:54:46 +03:00
__mark_inode_dirty ( & ip - > i_inode , I_DIRTY_DATASYNC ) ;
2006-01-16 19:50:04 +03:00
2006-09-04 20:04:26 +04:00
out_end_trans :
2006-06-14 23:32:57 +04:00
gfs2_trans_end ( GFS2_SB ( & ip - > i_inode ) ) ;
2006-09-04 20:04:26 +04:00
out_ipres :
2006-01-16 19:50:04 +03:00
gfs2_inplace_release ( ip ) ;
2006-09-04 20:04:26 +04:00
out_gunlock_q :
2006-01-16 19:50:04 +03:00
gfs2_quota_unlock ( ip ) ;
return error ;
}
static int ea_init_i ( struct gfs2_inode * ip , struct gfs2_ea_request * er ,
void * private )
{
struct buffer_head * bh ;
int error ;
error = ea_alloc_blk ( ip , & bh ) ;
if ( error )
return error ;
2008-11-03 17:28:42 +03:00
ip - > i_eattr = bh - > b_blocknr ;
2006-01-16 19:50:04 +03:00
error = ea_write ( ip , GFS2_EA_BH2FIRST ( bh ) , er ) ;
brelse ( bh ) ;
return error ;
}
/**
* ea_init - initializes a new eattr block
* @ ip :
* @ er :
*
* Returns : errno
*/
2009-08-26 21:41:32 +04:00
static int ea_init ( struct gfs2_inode * ip , int type , const char * name ,
const void * data , size_t size )
2006-01-16 19:50:04 +03:00
{
2009-08-26 21:41:32 +04:00
struct gfs2_ea_request er ;
2006-06-14 23:32:57 +04:00
unsigned int jbsize = GFS2_SB ( & ip - > i_inode ) - > sd_jbsize ;
2006-01-16 19:50:04 +03:00
unsigned int blks = 1 ;
2009-08-26 21:41:32 +04:00
er . er_type = type ;
er . er_name = name ;
er . er_name_len = strlen ( name ) ;
er . er_data = ( void * ) data ;
er . er_data_len = size ;
if ( GFS2_EAREQ_SIZE_STUFFED ( & er ) > jbsize )
blks + = DIV_ROUND_UP ( er . er_data_len , jbsize ) ;
2006-01-16 19:50:04 +03:00
2009-08-26 21:41:32 +04:00
return ea_alloc_skeleton ( ip , & er , blks , ea_init_i , NULL ) ;
2006-01-16 19:50:04 +03:00
}
static struct gfs2_ea_header * ea_split_ea ( struct gfs2_ea_header * ea )
{
2006-09-04 20:49:07 +04:00
u32 ea_size = GFS2_EA_SIZE ( ea ) ;
2006-02-27 20:00:42 +03:00
struct gfs2_ea_header * new = ( struct gfs2_ea_header * ) ( ( char * ) ea +
ea_size ) ;
2006-09-04 20:49:07 +04:00
u32 new_size = GFS2_EA_REC_LEN ( ea ) - ea_size ;
2006-01-16 19:50:04 +03:00
int last = ea - > ea_flags & GFS2_EAFLAG_LAST ;
ea - > ea_rec_len = cpu_to_be32 ( ea_size ) ;
ea - > ea_flags ^ = last ;
new - > ea_rec_len = cpu_to_be32 ( new_size ) ;
new - > ea_flags = last ;
return new ;
}
static void ea_set_remove_stuffed ( struct gfs2_inode * ip ,
struct gfs2_ea_location * el )
{
struct gfs2_ea_header * ea = el - > el_ea ;
struct gfs2_ea_header * prev = el - > el_prev ;
2006-09-04 20:49:07 +04:00
u32 len ;
2006-01-16 19:50:04 +03:00
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , el - > el_bh ) ;
2006-01-16 19:50:04 +03:00
if ( ! prev | | ! GFS2_EA_IS_STUFFED ( ea ) ) {
ea - > ea_type = GFS2_EATYPE_UNUSED ;
return ;
} else if ( GFS2_EA2NEXT ( prev ) ! = ea ) {
prev = GFS2_EA2NEXT ( prev ) ;
2006-06-14 23:32:57 +04:00
gfs2_assert_withdraw ( GFS2_SB ( & ip - > i_inode ) , GFS2_EA2NEXT ( prev ) = = ea ) ;
2006-01-16 19:50:04 +03:00
}
len = GFS2_EA_REC_LEN ( prev ) + GFS2_EA_REC_LEN ( ea ) ;
prev - > ea_rec_len = cpu_to_be32 ( len ) ;
if ( GFS2_EA_IS_LAST ( ea ) )
prev - > ea_flags | = GFS2_EAFLAG_LAST ;
}
struct ea_set {
int ea_split ;
struct gfs2_ea_request * es_er ;
struct gfs2_ea_location * es_el ;
struct buffer_head * es_bh ;
struct gfs2_ea_header * es_ea ;
} ;
static int ea_set_simple_noalloc ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea , struct ea_set * es )
{
struct gfs2_ea_request * er = es - > es_er ;
int error ;
2006-06-14 23:32:57 +04:00
error = gfs2_trans_begin ( GFS2_SB ( & ip - > i_inode ) , RES_DINODE + 2 * RES_EATTR , 0 ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , bh ) ;
2006-01-16 19:50:04 +03:00
if ( es - > ea_split )
ea = ea_split_ea ( ea ) ;
ea_write ( ip , ea , er ) ;
if ( es - > es_el )
ea_set_remove_stuffed ( ip , es - > es_el ) ;
2016-09-14 17:48:04 +03:00
ip - > i_inode . i_ctime = current_time ( & ip - > i_inode ) ;
2018-02-21 18:54:46 +03:00
__mark_inode_dirty ( & ip - > i_inode , I_DIRTY_DATASYNC ) ;
2017-10-04 17:21:19 +03:00
2006-06-14 23:32:57 +04:00
gfs2_trans_end ( GFS2_SB ( & ip - > i_inode ) ) ;
2006-01-16 19:50:04 +03:00
return error ;
}
static int ea_set_simple_alloc ( struct gfs2_inode * ip ,
struct gfs2_ea_request * er , void * private )
{
struct ea_set * es = private ;
struct gfs2_ea_header * ea = es - > es_ea ;
int error ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , es - > es_bh ) ;
2006-01-16 19:50:04 +03:00
if ( es - > ea_split )
ea = ea_split_ea ( ea ) ;
error = ea_write ( ip , ea , er ) ;
if ( error )
return error ;
if ( es - > es_el )
ea_set_remove_stuffed ( ip , es - > es_el ) ;
return 0 ;
}
static int ea_set_simple ( struct gfs2_inode * ip , struct buffer_head * bh ,
struct gfs2_ea_header * ea , struct gfs2_ea_header * prev ,
void * private )
{
struct ea_set * es = private ;
unsigned int size ;
int stuffed ;
int error ;
2009-08-26 21:41:32 +04:00
stuffed = ea_calc_size ( GFS2_SB ( & ip - > i_inode ) , es - > es_er - > er_name_len ,
es - > es_er - > er_data_len , & size ) ;
2006-01-16 19:50:04 +03:00
if ( ea - > ea_type = = GFS2_EATYPE_UNUSED ) {
if ( GFS2_EA_REC_LEN ( ea ) < size )
return 0 ;
if ( ! GFS2_EA_IS_STUFFED ( ea ) ) {
error = ea_remove_unstuffed ( ip , bh , ea , prev , 1 ) ;
if ( error )
return error ;
}
es - > ea_split = 0 ;
} else if ( GFS2_EA_REC_LEN ( ea ) - GFS2_EA_SIZE ( ea ) > = size )
es - > ea_split = 1 ;
else
return 0 ;
if ( stuffed ) {
error = ea_set_simple_noalloc ( ip , bh , ea , es ) ;
if ( error )
return error ;
} else {
unsigned int blks ;
es - > es_bh = bh ;
es - > es_ea = ea ;
2006-02-28 01:23:27 +03:00
blks = 2 + DIV_ROUND_UP ( es - > es_er - > er_data_len ,
2006-06-14 23:32:57 +04:00
GFS2_SB ( & ip - > i_inode ) - > sd_jbsize ) ;
2006-01-16 19:50:04 +03:00
error = ea_alloc_skeleton ( ip , es - > es_er , blks ,
ea_set_simple_alloc , es ) ;
if ( error )
return error ;
}
return 1 ;
}
static int ea_set_block ( struct gfs2_inode * ip , struct gfs2_ea_request * er ,
void * private )
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct buffer_head * indbh , * newbh ;
2006-10-14 18:46:30 +04:00
__be64 * eablk ;
2006-01-16 19:50:04 +03:00
int error ;
int mh_size = sizeof ( struct gfs2_meta_header ) ;
2008-11-04 13:05:22 +03:00
if ( ip - > i_diskflags & GFS2_DIF_EA_INDIRECT ) {
2006-10-14 18:46:30 +04:00
__be64 * end ;
2006-01-16 19:50:04 +03:00
2015-11-12 00:00:35 +03:00
error = gfs2_meta_read ( ip - > i_gl , ip - > i_eattr , DIO_WAIT , 0 ,
2006-09-22 01:05:23 +04:00
& indbh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
if ( gfs2_metatype_check ( sdp , indbh , GFS2_METATYPE_IN ) ) {
error = - EIO ;
goto out ;
}
2006-10-14 18:46:30 +04:00
eablk = ( __be64 * ) ( indbh - > b_data + mh_size ) ;
2006-01-16 19:50:04 +03:00
end = eablk + sdp - > sd_inptrs ;
for ( ; eablk < end ; eablk + + )
if ( ! * eablk )
break ;
if ( eablk = = end ) {
error = - ENOSPC ;
goto out ;
}
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , indbh ) ;
2006-01-16 19:50:04 +03:00
} else {
2006-09-04 20:49:07 +04:00
u64 blk ;
2008-02-06 13:11:15 +03:00
unsigned int n = 1 ;
2011-11-18 19:58:32 +04:00
error = gfs2_alloc_blocks ( ip , & blk , & n , 0 , NULL ) ;
2009-05-20 13:48:47 +04:00
if ( error )
return error ;
2019-04-05 14:18:23 +03:00
gfs2_trans_remove_revoke ( sdp , blk , 1 ) ;
2006-01-16 19:50:04 +03:00
indbh = gfs2_meta_new ( ip - > i_gl , blk ) ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , indbh ) ;
2006-01-16 19:50:04 +03:00
gfs2_metatype_set ( indbh , GFS2_METATYPE_IN , GFS2_FORMAT_IN ) ;
gfs2_buffer_clear_tail ( indbh , mh_size ) ;
2006-10-14 18:46:30 +04:00
eablk = ( __be64 * ) ( indbh - > b_data + mh_size ) ;
2008-11-03 17:28:42 +03:00
* eablk = cpu_to_be64 ( ip - > i_eattr ) ;
ip - > i_eattr = blk ;
2008-11-04 13:05:22 +03:00
ip - > i_diskflags | = GFS2_DIF_EA_INDIRECT ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , 1 ) ;
2006-01-16 19:50:04 +03:00
eablk + + ;
}
error = ea_alloc_blk ( ip , & newbh ) ;
if ( error )
goto out ;
2006-09-04 20:49:07 +04:00
* eablk = cpu_to_be64 ( ( u64 ) newbh - > b_blocknr ) ;
2006-01-16 19:50:04 +03:00
error = ea_write ( ip , GFS2_EA_BH2FIRST ( newbh ) , er ) ;
brelse ( newbh ) ;
if ( error )
goto out ;
if ( private )
2006-09-05 21:15:18 +04:00
ea_set_remove_stuffed ( ip , private ) ;
2006-01-16 19:50:04 +03:00
2006-09-04 20:04:26 +04:00
out :
2006-01-16 19:50:04 +03:00
brelse ( indbh ) ;
return error ;
}
2009-08-26 21:41:32 +04:00
static int ea_set_i ( struct gfs2_inode * ip , int type , const char * name ,
const void * value , size_t size , struct gfs2_ea_location * el )
2006-01-16 19:50:04 +03:00
{
2009-08-26 21:41:32 +04:00
struct gfs2_ea_request er ;
2006-01-16 19:50:04 +03:00
struct ea_set es ;
unsigned int blks = 2 ;
int error ;
2009-08-26 21:41:32 +04:00
er . er_type = type ;
er . er_name = name ;
er . er_data = ( void * ) value ;
er . er_name_len = strlen ( name ) ;
er . er_data_len = size ;
2006-01-16 19:50:04 +03:00
memset ( & es , 0 , sizeof ( struct ea_set ) ) ;
2009-08-26 21:41:32 +04:00
es . es_er = & er ;
2006-01-16 19:50:04 +03:00
es . es_el = el ;
error = ea_foreach ( ip , ea_set_simple , & es ) ;
if ( error > 0 )
return 0 ;
if ( error )
return error ;
2008-11-04 13:05:22 +03:00
if ( ! ( ip - > i_diskflags & GFS2_DIF_EA_INDIRECT ) )
2006-01-16 19:50:04 +03:00
blks + + ;
2009-08-26 21:41:32 +04:00
if ( GFS2_EAREQ_SIZE_STUFFED ( & er ) > GFS2_SB ( & ip - > i_inode ) - > sd_jbsize )
blks + = DIV_ROUND_UP ( er . er_data_len , GFS2_SB ( & ip - > i_inode ) - > sd_jbsize ) ;
2006-01-16 19:50:04 +03:00
2009-08-26 21:41:32 +04:00
return ea_alloc_skeleton ( ip , & er , blks , ea_set_block , el ) ;
2006-01-16 19:50:04 +03:00
}
static int ea_set_remove_unstuffed ( struct gfs2_inode * ip ,
struct gfs2_ea_location * el )
{
if ( el - > el_prev & & GFS2_EA2NEXT ( el - > el_prev ) ! = el - > el_ea ) {
el - > el_prev = GFS2_EA2NEXT ( el - > el_prev ) ;
2006-06-14 23:32:57 +04:00
gfs2_assert_withdraw ( GFS2_SB ( & ip - > i_inode ) ,
2006-01-16 19:50:04 +03:00
GFS2_EA2NEXT ( el - > el_prev ) = = el - > el_ea ) ;
}
2009-09-14 12:50:57 +04:00
return ea_remove_unstuffed ( ip , el - > el_bh , el - > el_ea , el - > el_prev , 0 ) ;
2006-01-16 19:50:04 +03:00
}
static int ea_remove_stuffed ( struct gfs2_inode * ip , struct gfs2_ea_location * el )
{
struct gfs2_ea_header * ea = el - > el_ea ;
struct gfs2_ea_header * prev = el - > el_prev ;
int error ;
2006-06-14 23:32:57 +04:00
error = gfs2_trans_begin ( GFS2_SB ( & ip - > i_inode ) , RES_DINODE + RES_EATTR , 0 ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , el - > el_bh ) ;
2006-01-16 19:50:04 +03:00
if ( prev ) {
2006-09-04 20:49:07 +04:00
u32 len ;
2006-01-16 19:50:04 +03:00
len = GFS2_EA_REC_LEN ( prev ) + GFS2_EA_REC_LEN ( ea ) ;
prev - > ea_rec_len = cpu_to_be32 ( len ) ;
if ( GFS2_EA_IS_LAST ( ea ) )
prev - > ea_flags | = GFS2_EAFLAG_LAST ;
2009-08-26 21:41:32 +04:00
} else {
2006-01-16 19:50:04 +03:00
ea - > ea_type = GFS2_EATYPE_UNUSED ;
2009-08-26 21:41:32 +04:00
}
2006-01-16 19:50:04 +03:00
2017-10-04 17:21:19 +03:00
ip - > i_inode . i_ctime = current_time ( & ip - > i_inode ) ;
2018-02-21 18:54:46 +03:00
__mark_inode_dirty ( & ip - > i_inode , I_DIRTY_DATASYNC ) ;
2006-01-16 19:50:04 +03:00
2006-06-14 23:32:57 +04:00
gfs2_trans_end ( GFS2_SB ( & ip - > i_inode ) ) ;
2006-01-16 19:50:04 +03:00
return error ;
}
2009-08-26 21:41:32 +04:00
/**
* gfs2_xattr_remove - Remove a GFS2 extended attribute
2009-11-13 12:52:56 +03:00
* @ ip : The inode
2009-08-26 21:41:32 +04:00
* @ type : The type of the extended attribute
* @ name : The name of the extended attribute
*
* This is not called directly by the VFS since we use the ( common )
* scheme of making a " set with NULL data " mean a remove request . Note
* that this is different from a set with zero length data .
*
* Returns : 0 , or errno on failure
*/
2009-11-13 12:52:56 +03:00
static int gfs2_xattr_remove ( struct gfs2_inode * ip , int type , const char * name )
2006-01-16 19:50:04 +03:00
{
struct gfs2_ea_location el ;
int error ;
2008-11-03 17:28:42 +03:00
if ( ! ip - > i_eattr )
2006-01-16 19:50:04 +03:00
return - ENODATA ;
2009-08-26 21:41:32 +04:00
error = gfs2_ea_find ( ip , type , name , & el ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
if ( ! el . el_ea )
return - ENODATA ;
if ( GFS2_EA_IS_STUFFED ( el . el_ea ) )
error = ea_remove_stuffed ( ip , & el ) ;
else
2009-08-26 21:41:32 +04:00
error = ea_remove_unstuffed ( ip , el . el_bh , el . el_ea , el . el_prev , 0 ) ;
2006-01-16 19:50:04 +03:00
brelse ( el . el_bh ) ;
return error ;
}
/**
2009-11-13 12:52:56 +03:00
* __gfs2_xattr_set - Set ( or remove ) a GFS2 extended attribute
* @ ip : The inode
2009-08-26 21:41:32 +04:00
* @ name : The name of the extended attribute
* @ value : The value of the extended attribute ( NULL for remove )
* @ size : The size of the @ value argument
* @ flags : Create or Replace
2009-11-13 12:52:56 +03:00
* @ type : The type of the extended attribute
2006-01-16 19:50:04 +03:00
*
2009-08-26 21:41:32 +04:00
* See gfs2_xattr_remove ( ) for details of the removal of xattrs .
*
* Returns : 0 or errno on failure
2006-01-16 19:50:04 +03:00
*/
2009-11-13 12:52:56 +03:00
int __gfs2_xattr_set ( struct inode * inode , const char * name ,
const void * value , size_t size , int flags , int type )
2006-01-16 19:50:04 +03:00
{
2009-08-26 21:41:32 +04:00
struct gfs2_inode * ip = GFS2_I ( inode ) ;
2009-11-13 12:52:56 +03:00
struct gfs2_sbd * sdp = GFS2_SB ( inode ) ;
2009-08-26 21:41:32 +04:00
struct gfs2_ea_location el ;
unsigned int namel = strlen ( name ) ;
2006-01-16 19:50:04 +03:00
int error ;
2009-08-26 21:41:32 +04:00
if ( IS_IMMUTABLE ( inode ) | | IS_APPEND ( inode ) )
return - EPERM ;
if ( namel > GFS2_EA_MAX_NAME_LEN )
return - ERANGE ;
2006-01-16 19:50:04 +03:00
2017-08-30 15:26:30 +03:00
if ( value = = NULL ) {
error = gfs2_xattr_remove ( ip , type , name ) ;
if ( error = = - ENODATA & & ! ( flags & XATTR_REPLACE ) )
error = 0 ;
return error ;
}
2009-08-26 21:41:32 +04:00
if ( ea_check_size ( sdp , namel , size ) )
return - ERANGE ;
if ( ! ip - > i_eattr ) {
if ( flags & XATTR_REPLACE )
return - ENODATA ;
return ea_init ( ip , type , name , value , size ) ;
}
error = gfs2_ea_find ( ip , type , name , & el ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2009-08-26 21:41:32 +04:00
if ( el . el_ea ) {
if ( ip - > i_diskflags & GFS2_DIF_APPENDONLY ) {
brelse ( el . el_bh ) ;
return - EPERM ;
}
2006-01-16 19:50:04 +03:00
2009-08-26 21:41:32 +04:00
error = - EEXIST ;
if ( ! ( flags & XATTR_CREATE ) ) {
int unstuffed = ! GFS2_EA_IS_STUFFED ( el . el_ea ) ;
error = ea_set_i ( ip , type , name , value , size , & el ) ;
if ( ! error & & unstuffed )
ea_set_remove_unstuffed ( ip , & el ) ;
}
brelse ( el . el_bh ) ;
return error ;
}
error = - ENODATA ;
if ( ! ( flags & XATTR_REPLACE ) )
error = ea_set_i ( ip , type , name , value , size , NULL ) ;
2006-01-16 19:50:04 +03:00
return error ;
}
2015-10-04 20:18:51 +03:00
static int gfs2_xattr_set ( const struct xattr_handler * handler ,
2016-05-27 17:19:30 +03:00
struct dentry * unused , struct inode * inode ,
const char * name , const void * value ,
size_t size , int flags )
2009-11-13 12:52:56 +03:00
{
2016-05-13 04:59:17 +03:00
struct gfs2_inode * ip = GFS2_I ( inode ) ;
struct gfs2_holder gh ;
int ret ;
2020-02-27 21:47:53 +03:00
ret = gfs2_qa_get ( ip ) ;
2016-05-13 04:59:17 +03:00
if ( ret )
return ret ;
2017-10-13 01:39:38 +03:00
/* May be called from gfs_setattr with the glock locked. */
if ( ! gfs2_glock_is_locked_by_me ( ip - > i_gl ) ) {
ret = gfs2_glock_nq_init ( ip - > i_gl , LM_ST_EXCLUSIVE , 0 , & gh ) ;
if ( ret )
2020-02-27 21:47:53 +03:00
goto out ;
2017-10-13 01:39:38 +03:00
} else {
2020-02-27 21:47:53 +03:00
if ( WARN_ON_ONCE ( ip - > i_gl - > gl_state ! = LM_ST_EXCLUSIVE ) ) {
ret = - EIO ;
goto out ;
}
2017-10-13 01:39:38 +03:00
gfs2_holder_mark_uninitialized ( & gh ) ;
}
2016-05-13 04:59:17 +03:00
ret = __gfs2_xattr_set ( inode , name , value , size , flags , handler - > flags ) ;
2017-10-13 01:39:38 +03:00
if ( gfs2_holder_initialized ( & gh ) )
gfs2_glock_dq_uninit ( & gh ) ;
2020-02-27 21:47:53 +03:00
out :
gfs2_qa_put ( ip ) ;
2016-05-13 04:59:17 +03:00
return ret ;
2009-11-13 12:52:56 +03:00
}
2006-01-16 19:50:04 +03:00
static int ea_dealloc_indirect ( struct gfs2_inode * ip )
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct gfs2_rgrp_list rlist ;
2018-10-04 17:36:02 +03:00
struct gfs2_rgrpd * rgd ;
2006-01-16 19:50:04 +03:00
struct buffer_head * indbh , * dibh ;
2006-10-14 18:46:30 +04:00
__be64 * eablk , * end ;
2006-01-16 19:50:04 +03:00
unsigned int rg_blocks = 0 ;
2006-09-04 20:49:07 +04:00
u64 bstart = 0 ;
2006-01-16 19:50:04 +03:00
unsigned int blen = 0 ;
unsigned int blks = 0 ;
unsigned int x ;
int error ;
2012-04-05 06:11:16 +04:00
error = gfs2_rindex_update ( sdp ) ;
if ( error )
return error ;
2006-01-16 19:50:04 +03:00
memset ( & rlist , 0 , sizeof ( struct gfs2_rgrp_list ) ) ;
2015-11-12 00:00:35 +03:00
error = gfs2_meta_read ( ip - > i_gl , ip - > i_eattr , DIO_WAIT , 0 , & indbh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
if ( gfs2_metatype_check ( sdp , indbh , GFS2_METATYPE_IN ) ) {
error = - EIO ;
goto out ;
}
2006-10-14 18:46:30 +04:00
eablk = ( __be64 * ) ( indbh - > b_data + sizeof ( struct gfs2_meta_header ) ) ;
2006-01-16 19:50:04 +03:00
end = eablk + sdp - > sd_inptrs ;
for ( ; eablk < end ; eablk + + ) {
2006-09-04 20:49:07 +04:00
u64 bn ;
2006-01-16 19:50:04 +03:00
if ( ! * eablk )
break ;
bn = be64_to_cpu ( * eablk ) ;
if ( bstart + blen = = bn )
blen + + ;
else {
if ( bstart )
2011-09-02 19:08:09 +04:00
gfs2_rlist_add ( ip , & rlist , bstart ) ;
2006-01-16 19:50:04 +03:00
bstart = bn ;
blen = 1 ;
}
blks + + ;
}
if ( bstart )
2011-09-02 19:08:09 +04:00
gfs2_rlist_add ( ip , & rlist , bstart ) ;
2006-01-16 19:50:04 +03:00
else
goto out ;
2018-10-04 02:06:23 +03:00
gfs2_rlist_alloc ( & rlist ) ;
2006-01-16 19:50:04 +03:00
for ( x = 0 ; x < rlist . rl_rgrps ; x + + ) {
2018-10-04 17:36:02 +03:00
rgd = gfs2_glock2rgrp ( rlist . rl_ghs [ x ] . gh_gl ) ;
2007-06-01 17:11:58 +04:00
rg_blocks + = rgd - > rd_length ;
2006-01-16 19:50:04 +03:00
}
error = gfs2_glock_nq_m ( rlist . rl_rgrps , rlist . rl_ghs ) ;
if ( error )
goto out_rlist_free ;
2006-09-05 21:15:18 +04:00
error = gfs2_trans_begin ( sdp , rg_blocks + RES_DINODE + RES_INDIRECT +
RES_STATFS + RES_QUOTA , blks ) ;
2006-01-16 19:50:04 +03:00
if ( error )
goto out_gunlock ;
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , indbh ) ;
2006-01-16 19:50:04 +03:00
2006-10-14 18:46:30 +04:00
eablk = ( __be64 * ) ( indbh - > b_data + sizeof ( struct gfs2_meta_header ) ) ;
2006-01-16 19:50:04 +03:00
bstart = 0 ;
2018-10-04 17:36:02 +03:00
rgd = NULL ;
2006-01-16 19:50:04 +03:00
blen = 0 ;
for ( ; eablk < end ; eablk + + ) {
2006-09-04 20:49:07 +04:00
u64 bn ;
2006-01-16 19:50:04 +03:00
if ( ! * eablk )
break ;
bn = be64_to_cpu ( * eablk ) ;
if ( bstart + blen = = bn )
blen + + ;
else {
if ( bstart )
2018-10-04 17:36:02 +03:00
gfs2_free_meta ( ip , rgd , bstart , blen ) ;
2006-01-16 19:50:04 +03:00
bstart = bn ;
2018-10-04 17:36:02 +03:00
rgd = gfs2_blk2rgrpd ( sdp , bstart , true ) ;
2006-01-16 19:50:04 +03:00
blen = 1 ;
}
* eablk = 0 ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , - 1 ) ;
2006-01-16 19:50:04 +03:00
}
if ( bstart )
2018-10-04 17:36:02 +03:00
gfs2_free_meta ( ip , rgd , bstart , blen ) ;
2006-01-16 19:50:04 +03:00
2008-11-04 13:05:22 +03:00
ip - > i_diskflags & = ~ GFS2_DIF_EA_INDIRECT ;
2006-01-16 19:50:04 +03:00
error = gfs2_meta_inode_buffer ( ip , & dibh ) ;
if ( ! error ) {
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , dibh ) ;
2006-10-31 23:07:05 +03:00
gfs2_dinode_out ( ip , dibh - > b_data ) ;
2006-01-16 19:50:04 +03:00
brelse ( dibh ) ;
}
gfs2_trans_end ( sdp ) ;
2006-09-04 20:04:26 +04:00
out_gunlock :
2006-01-16 19:50:04 +03:00
gfs2_glock_dq_m ( rlist . rl_rgrps , rlist . rl_ghs ) ;
2006-09-04 20:04:26 +04:00
out_rlist_free :
2006-01-16 19:50:04 +03:00
gfs2_rlist_free ( & rlist ) ;
2006-09-04 20:04:26 +04:00
out :
2006-01-16 19:50:04 +03:00
brelse ( indbh ) ;
return error ;
}
static int ea_dealloc_block ( struct gfs2_inode * ip )
{
2006-06-14 23:32:57 +04:00
struct gfs2_sbd * sdp = GFS2_SB ( & ip - > i_inode ) ;
2006-01-16 19:50:04 +03:00
struct gfs2_rgrpd * rgd ;
struct buffer_head * dibh ;
2011-11-21 22:36:17 +04:00
struct gfs2_holder gh ;
2006-01-16 19:50:04 +03:00
int error ;
2012-04-05 06:11:16 +04:00
error = gfs2_rindex_update ( sdp ) ;
if ( error )
return error ;
2012-02-08 16:58:32 +04:00
rgd = gfs2_blk2rgrpd ( sdp , ip - > i_eattr , 1 ) ;
2006-01-16 19:50:04 +03:00
if ( ! rgd ) {
gfs2_consist_inode ( ip ) ;
return - EIO ;
}
2011-11-21 22:36:17 +04:00
error = gfs2_glock_nq_init ( rgd - > rd_gl , LM_ST_EXCLUSIVE , 0 , & gh ) ;
2006-01-16 19:50:04 +03:00
if ( error )
return error ;
2006-09-05 21:15:18 +04:00
error = gfs2_trans_begin ( sdp , RES_RG_BIT + RES_DINODE + RES_STATFS +
RES_QUOTA , 1 ) ;
2006-01-16 19:50:04 +03:00
if ( error )
goto out_gunlock ;
2018-10-04 17:36:02 +03:00
gfs2_free_meta ( ip , rgd , ip - > i_eattr , 1 ) ;
2006-01-16 19:50:04 +03:00
2008-11-03 17:28:42 +03:00
ip - > i_eattr = 0 ;
2008-02-12 17:17:27 +03:00
gfs2_add_inode_blocks ( & ip - > i_inode , - 1 ) ;
2006-01-16 19:50:04 +03:00
error = gfs2_meta_inode_buffer ( ip , & dibh ) ;
if ( ! error ) {
2012-12-14 16:36:02 +04:00
gfs2_trans_add_meta ( ip - > i_gl , dibh ) ;
2006-10-31 23:07:05 +03:00
gfs2_dinode_out ( ip , dibh - > b_data ) ;
2006-01-16 19:50:04 +03:00
brelse ( dibh ) ;
}
gfs2_trans_end ( sdp ) ;
2006-09-04 20:04:26 +04:00
out_gunlock :
2011-11-21 22:36:17 +04:00
gfs2_glock_dq_uninit ( & gh ) ;
2006-01-16 19:50:04 +03:00
return error ;
}
/**
* gfs2_ea_dealloc - deallocate the extended attribute fork
* @ ip : the inode
*
* Returns : errno
*/
int gfs2_ea_dealloc ( struct gfs2_inode * ip )
{
int error ;
2012-07-19 16:12:40 +04:00
error = gfs2_rindex_update ( GFS2_SB ( & ip - > i_inode ) ) ;
if ( error )
return error ;
2013-02-01 05:49:26 +04:00
error = gfs2_quota_hold ( ip , NO_UID_QUOTA_CHANGE , NO_GID_QUOTA_CHANGE ) ;
2006-01-16 19:50:04 +03:00
if ( error )
2012-05-18 17:28:23 +04:00
return error ;
2006-01-16 19:50:04 +03:00
error = ea_foreach ( ip , ea_dealloc_unstuffed , NULL ) ;
if ( error )
2011-08-31 19:38:29 +04:00
goto out_quota ;
2006-01-16 19:50:04 +03:00
2008-11-04 13:05:22 +03:00
if ( ip - > i_diskflags & GFS2_DIF_EA_INDIRECT ) {
2006-01-16 19:50:04 +03:00
error = ea_dealloc_indirect ( ip ) ;
if ( error )
2011-08-31 19:38:29 +04:00
goto out_quota ;
2006-01-16 19:50:04 +03:00
}
error = ea_dealloc_block ( ip ) ;
2006-09-04 20:04:26 +04:00
out_quota :
2006-01-16 19:50:04 +03:00
gfs2_quota_unhold ( ip ) ;
return error ;
}
2010-05-14 04:53:23 +04:00
static const struct xattr_handler gfs2_xattr_user_handler = {
2009-08-26 21:41:32 +04:00
. prefix = XATTR_USER_PREFIX ,
2009-11-13 12:52:56 +03:00
. flags = GFS2_EATYPE_USR ,
. get = gfs2_xattr_get ,
. set = gfs2_xattr_set ,
2009-08-26 21:41:32 +04:00
} ;
2010-05-14 04:53:23 +04:00
static const struct xattr_handler gfs2_xattr_security_handler = {
2009-08-26 21:41:32 +04:00
. prefix = XATTR_SECURITY_PREFIX ,
2009-11-13 12:52:56 +03:00
. flags = GFS2_EATYPE_SECURITY ,
. get = gfs2_xattr_get ,
. set = gfs2_xattr_set ,
2009-08-26 21:41:32 +04:00
} ;
2010-05-14 04:53:23 +04:00
const struct xattr_handler * gfs2_xattr_handlers [ ] = {
2009-08-26 21:41:32 +04:00
& gfs2_xattr_user_handler ,
& gfs2_xattr_security_handler ,
2013-12-20 17:16:52 +04:00
& posix_acl_access_xattr_handler ,
& posix_acl_default_xattr_handler ,
2009-08-26 21:41:32 +04:00
NULL ,
} ;