2010-04-07 02:14:15 +04:00
# include <linux/ceph/ceph_debug.h>
2009-10-06 22:31:09 +04:00
# include <linux/bug.h>
# include <linux/err.h>
# include <linux/random.h>
# include <linux/slab.h>
# include <linux/types.h>
2010-04-07 02:14:15 +04:00
# include <linux/ceph/mdsmap.h>
# include <linux/ceph/messenger.h>
# include <linux/ceph/decode.h>
2009-10-06 22:31:09 +04:00
# include "super.h"
/*
* choose a random mds that is " up " ( i . e . has a state > 0 ) , or - 1.
*/
int ceph_mdsmap_get_random_mds ( struct ceph_mdsmap * m )
{
int n = 0 ;
int i ;
2013-04-10 01:49:11 +04:00
/* special case for one mds */
if ( 1 = = m - > m_max_mds & & m - > m_info [ 0 ] . state > 0 )
return 0 ;
2009-10-06 22:31:09 +04:00
/* count */
for ( i = 0 ; i < m - > m_max_mds ; i + + )
if ( m - > m_info [ i ] . state > 0 )
n + + ;
if ( n = = 0 )
return - 1 ;
/* pick */
2013-04-10 01:49:11 +04:00
n = prandom_u32 ( ) % n ;
2009-10-06 22:31:09 +04:00
i = 0 ;
for ( i = 0 ; n > 0 ; i + + , n - - )
while ( m - > m_info [ i ] . state < = 0 )
i + + ;
return i ;
}
/*
* Decode an MDS map
*
* Ignore any fields we don ' t care about ( there are quite a few of
* them ) .
*/
struct ceph_mdsmap * ceph_mdsmap_decode ( void * * p , void * end )
{
struct ceph_mdsmap * m ;
2009-12-15 02:13:47 +03:00
const void * start = * p ;
2009-10-06 22:31:09 +04:00
int i , j , n ;
int err = - EINVAL ;
2016-03-31 10:53:01 +03:00
u8 mdsmap_v , mdsmap_cv ;
2009-10-06 22:31:09 +04:00
m = kzalloc ( sizeof ( * m ) , GFP_NOFS ) ;
if ( m = = NULL )
return ERR_PTR ( - ENOMEM ) ;
2016-03-31 10:53:01 +03:00
ceph_decode_need ( p , end , 1 + 1 , bad ) ;
mdsmap_v = ceph_decode_8 ( p ) ;
mdsmap_cv = ceph_decode_8 ( p ) ;
if ( mdsmap_v > = 4 ) {
u32 mdsmap_len ;
ceph_decode_32_safe ( p , end , mdsmap_len , bad ) ;
if ( end < * p + mdsmap_len )
goto bad ;
end = * p + mdsmap_len ;
2013-02-23 22:41:09 +04:00
}
2009-10-06 22:31:09 +04:00
ceph_decode_need ( p , end , 8 * sizeof ( u32 ) + sizeof ( u64 ) , bad ) ;
2009-10-14 20:59:09 +04:00
m - > m_epoch = ceph_decode_32 ( p ) ;
m - > m_client_epoch = ceph_decode_32 ( p ) ;
m - > m_last_failure = ceph_decode_32 ( p ) ;
m - > m_root = ceph_decode_32 ( p ) ;
m - > m_session_timeout = ceph_decode_32 ( p ) ;
m - > m_session_autoclose = ceph_decode_32 ( p ) ;
m - > m_max_file_size = ceph_decode_64 ( p ) ;
m - > m_max_mds = ceph_decode_32 ( p ) ;
2009-10-06 22:31:09 +04:00
m - > m_info = kcalloc ( m - > m_max_mds , sizeof ( * m - > m_info ) , GFP_NOFS ) ;
if ( m - > m_info = = NULL )
goto badmem ;
/* pick out active nodes from mds_info (state > 0) */
2009-10-14 20:59:09 +04:00
n = ceph_decode_32 ( p ) ;
2009-10-06 22:31:09 +04:00
for ( i = 0 ; i < n ; i + + ) {
2009-11-20 02:31:50 +03:00
u64 global_id ;
2009-10-06 22:31:09 +04:00
u32 namelen ;
s32 mds , inc , state ;
u64 state_seq ;
2016-03-31 10:53:01 +03:00
u8 info_v ;
void * info_end = NULL ;
2009-10-06 22:31:09 +04:00
struct ceph_entity_addr addr ;
u32 num_export_targets ;
void * pexport_targets = NULL ;
2010-06-18 01:19:01 +04:00
struct ceph_timespec laggy_since ;
2013-05-29 15:46:56 +04:00
struct ceph_mds_info * info ;
2009-10-06 22:31:09 +04:00
2016-03-31 10:53:01 +03:00
ceph_decode_need ( p , end , sizeof ( u64 ) + 1 , bad ) ;
2009-11-20 02:31:50 +03:00
global_id = ceph_decode_64 ( p ) ;
2016-03-31 10:53:01 +03:00
info_v = ceph_decode_8 ( p ) ;
if ( info_v > = 4 ) {
u32 info_len ;
u8 info_cv ;
ceph_decode_need ( p , end , 1 + sizeof ( u32 ) , bad ) ;
info_cv = ceph_decode_8 ( p ) ;
info_len = ceph_decode_32 ( p ) ;
info_end = * p + info_len ;
if ( info_end > end )
goto bad ;
}
ceph_decode_need ( p , end , sizeof ( u64 ) + sizeof ( u32 ) , bad ) ;
2009-11-20 02:31:50 +03:00
* p + = sizeof ( u64 ) ;
2009-10-14 20:59:09 +04:00
namelen = ceph_decode_32 ( p ) ; /* skip mds name */
2009-10-06 22:31:09 +04:00
* p + = namelen ;
ceph_decode_need ( p , end ,
2009-10-08 03:38:19 +04:00
4 * sizeof ( u32 ) + sizeof ( u64 ) +
2009-10-06 22:31:09 +04:00
sizeof ( addr ) + sizeof ( struct ceph_timespec ) ,
bad ) ;
2009-10-14 20:59:09 +04:00
mds = ceph_decode_32 ( p ) ;
inc = ceph_decode_32 ( p ) ;
state = ceph_decode_32 ( p ) ;
state_seq = ceph_decode_64 ( p ) ;
2009-11-20 02:31:50 +03:00
ceph_decode_copy ( p , & addr , sizeof ( addr ) ) ;
ceph_decode_addr ( & addr ) ;
2010-06-18 01:19:01 +04:00
ceph_decode_copy ( p , & laggy_since , sizeof ( laggy_since ) ) ;
2009-10-06 22:31:09 +04:00
* p + = sizeof ( u32 ) ;
ceph_decode_32_safe ( p , end , namelen , bad ) ;
2009-10-08 03:38:19 +04:00
* p + = namelen ;
2016-03-31 10:53:01 +03:00
if ( info_v > = 2 ) {
2009-10-06 22:31:09 +04:00
ceph_decode_32_safe ( p , end , num_export_targets , bad ) ;
pexport_targets = * p ;
2009-10-08 03:38:19 +04:00
* p + = num_export_targets * sizeof ( u32 ) ;
2009-10-06 22:31:09 +04:00
} else {
num_export_targets = 0 ;
}
2016-03-31 10:53:01 +03:00
if ( info_end & & * p ! = info_end ) {
if ( * p > info_end )
goto bad ;
* p = info_end ;
}
2009-11-20 02:31:50 +03:00
dout ( " mdsmap_decode %d/%d %lld mds%d.%d %s %s \n " ,
2010-04-07 02:14:15 +04:00
i + 1 , n , global_id , mds , inc ,
ceph_pr_addr ( & addr . in_addr ) ,
2009-10-06 22:31:09 +04:00
ceph_mds_state_name ( state ) ) ;
2013-05-29 15:46:56 +04:00
if ( mds < 0 | | mds > = m - > m_max_mds | | state < = 0 )
continue ;
info = & m - > m_info [ mds ] ;
info - > global_id = global_id ;
info - > state = state ;
info - > addr = addr ;
info - > laggy = ( laggy_since . tv_sec ! = 0 | |
laggy_since . tv_nsec ! = 0 ) ;
info - > num_export_targets = num_export_targets ;
if ( num_export_targets ) {
info - > export_targets = kcalloc ( num_export_targets ,
sizeof ( u32 ) , GFP_NOFS ) ;
if ( info - > export_targets = = NULL )
goto badmem ;
for ( j = 0 ; j < num_export_targets ; j + + )
info - > export_targets [ j ] =
ceph_decode_32 ( & pexport_targets ) ;
} else {
info - > export_targets = NULL ;
2009-10-06 22:31:09 +04:00
}
}
/* pg_pools */
ceph_decode_32_safe ( p , end , n , bad ) ;
m - > m_num_data_pg_pools = n ;
2013-02-23 22:41:09 +04:00
m - > m_data_pg_pools = kcalloc ( n , sizeof ( u64 ) , GFP_NOFS ) ;
2009-10-06 22:31:09 +04:00
if ( ! m - > m_data_pg_pools )
goto badmem ;
2013-02-23 22:41:09 +04:00
ceph_decode_need ( p , end , sizeof ( u64 ) * ( n + 1 ) , bad ) ;
2009-10-06 22:31:09 +04:00
for ( i = 0 ; i < n ; i + + )
2013-02-23 22:41:09 +04:00
m - > m_data_pg_pools [ i ] = ceph_decode_64 ( p ) ;
m - > m_cas_pg_pool = ceph_decode_64 ( p ) ;
2009-10-06 22:31:09 +04:00
/* ok, we don't care about the rest. */
2016-03-31 10:53:01 +03:00
* p = end ;
2009-10-06 22:31:09 +04:00
dout ( " mdsmap_decode success epoch %u \n " , m - > m_epoch ) ;
return m ;
badmem :
err = - ENOMEM ;
bad :
pr_err ( " corrupt mdsmap \n " ) ;
2009-12-15 02:13:47 +03:00
print_hex_dump ( KERN_DEBUG , " mdsmap: " ,
DUMP_PREFIX_OFFSET , 16 , 1 ,
start , end - start , true ) ;
2009-10-06 22:31:09 +04:00
ceph_mdsmap_destroy ( m ) ;
2013-05-28 18:59:00 +04:00
return ERR_PTR ( err ) ;
2009-10-06 22:31:09 +04:00
}
void ceph_mdsmap_destroy ( struct ceph_mdsmap * m )
{
int i ;
for ( i = 0 ; i < m - > m_max_mds ; i + + )
kfree ( m - > m_info [ i ] . export_targets ) ;
kfree ( m - > m_info ) ;
kfree ( m - > m_data_pg_pools ) ;
kfree ( m ) ;
}