2009-10-06 11:31:11 -07:00
# ifdef __KERNEL__
# include <linux / string.h>
# include <linux / slab.h>
# include <linux / bug.h>
# include <linux / kernel.h>
# ifndef dprintk
# define dprintk(args...)
# endif
# else
# include <string.h>
# include <stdio.h>
# include <stdlib.h>
# include <assert.h>
# define BUG_ON(x) assert(!(x))
# define dprintk(args...) /* printf(args) */
# define kmalloc(x, f) malloc(x)
# define kfree(x) free(x)
# endif
2010-04-06 15:14:15 -07:00
# include <linux/crush/crush.h>
# include <linux/crush/hash.h>
2012-04-24 14:38:37 +00:00
# include <linux/crush/mapper.h>
2009-10-06 11:31:11 -07:00
/*
* Implement the core CRUSH mapping algorithm .
*/
/**
* crush_find_rule - find a crush_rule id for a given ruleset , type , and size .
* @ map : the crush_map
* @ ruleset : the storage ruleset id ( user defined )
* @ type : storage ruleset type ( user defined )
* @ size : output set size
*/
2012-05-07 15:38:35 -07:00
int crush_find_rule ( const struct crush_map * map , int ruleset , int type , int size )
2009-10-06 11:31:11 -07:00
{
2012-05-07 15:38:35 -07:00
__u32 i ;
2009-10-06 11:31:11 -07:00
for ( i = 0 ; i < map - > max_rules ; i + + ) {
if ( map - > rules [ i ] & &
map - > rules [ i ] - > mask . ruleset = = ruleset & &
map - > rules [ i ] - > mask . type = = type & &
map - > rules [ i ] - > mask . min_size < = size & &
map - > rules [ i ] - > mask . max_size > = size )
return i ;
}
return - 1 ;
}
/*
* bucket choose methods
*
* For each bucket algorithm , we have a " choose " method that , given a
* crush input @ x and replica position ( usually , position in output set ) @ r ,
* will produce an item in the bucket .
*/
/*
* Choose based on a random permutation of the bucket .
*
* We used to use some prime number arithmetic to do this , but it
* wasn ' t very random , and had some other bad behaviors . Instead , we
* calculate an actual random permutation of the bucket members .
* Since this is expensive , we optimize for the r = 0 case , which
* captures the vast majority of calls .
*/
static int bucket_perm_choose ( struct crush_bucket * bucket ,
int x , int r )
{
2012-04-15 05:58:06 +00:00
unsigned int pr = r % bucket - > size ;
unsigned int i , s ;
2009-10-06 11:31:11 -07:00
/* start a new permutation if @x has changed */
2012-05-07 15:38:35 -07:00
if ( bucket - > perm_x ! = ( __u32 ) x | | bucket - > perm_n = = 0 ) {
2009-10-06 11:31:11 -07:00
dprintk ( " bucket %d new x=%d \n " , bucket - > id , x ) ;
bucket - > perm_x = x ;
/* optimize common r=0 case */
if ( pr = = 0 ) {
2009-11-07 20:18:22 -08:00
s = crush_hash32_3 ( bucket - > hash , x , bucket - > id , 0 ) %
2009-10-06 11:31:11 -07:00
bucket - > size ;
bucket - > perm [ 0 ] = s ;
bucket - > perm_n = 0xffff ; /* magic value, see below */
goto out ;
}
for ( i = 0 ; i < bucket - > size ; i + + )
bucket - > perm [ i ] = i ;
bucket - > perm_n = 0 ;
} else if ( bucket - > perm_n = = 0xffff ) {
/* clean up after the r=0 case above */
for ( i = 1 ; i < bucket - > size ; i + + )
bucket - > perm [ i ] = i ;
bucket - > perm [ bucket - > perm [ 0 ] ] = 0 ;
bucket - > perm_n = 1 ;
}
/* calculate permutation up to pr */
for ( i = 0 ; i < bucket - > perm_n ; i + + )
dprintk ( " perm_choose have %d: %d \n " , i , bucket - > perm [ i ] ) ;
while ( bucket - > perm_n < = pr ) {
2012-04-15 05:58:06 +00:00
unsigned int p = bucket - > perm_n ;
2009-10-06 11:31:11 -07:00
/* no point in swapping the final entry */
if ( p < bucket - > size - 1 ) {
2009-11-07 20:18:22 -08:00
i = crush_hash32_3 ( bucket - > hash , x , bucket - > id , p ) %
2009-10-06 11:31:11 -07:00
( bucket - > size - p ) ;
if ( i ) {
2012-04-15 05:58:06 +00:00
unsigned int t = bucket - > perm [ p + i ] ;
2009-10-06 11:31:11 -07:00
bucket - > perm [ p + i ] = bucket - > perm [ p ] ;
bucket - > perm [ p ] = t ;
}
dprintk ( " perm_choose swap %d with %d \n " , p , p + i ) ;
}
bucket - > perm_n + + ;
}
for ( i = 0 ; i < bucket - > size ; i + + )
dprintk ( " perm_choose %d: %d \n " , i , bucket - > perm [ i ] ) ;
s = bucket - > perm [ pr ] ;
out :
dprintk ( " perm_choose %d sz=%d x=%d r=%d (%d) s=%d \n " , bucket - > id ,
bucket - > size , x , r , pr , s ) ;
return bucket - > items [ s ] ;
}
/* uniform */
static int bucket_uniform_choose ( struct crush_bucket_uniform * bucket ,
int x , int r )
{
return bucket_perm_choose ( & bucket - > h , x , r ) ;
}
/* list */
static int bucket_list_choose ( struct crush_bucket_list * bucket ,
int x , int r )
{
int i ;
for ( i = bucket - > h . size - 1 ; i > = 0 ; i - - ) {
2009-11-07 20:18:22 -08:00
__u64 w = crush_hash32_4 ( bucket - > h . hash , x , bucket - > h . items [ i ] ,
r , bucket - > h . id ) ;
2009-10-06 11:31:11 -07:00
w & = 0xffff ;
dprintk ( " list_choose i=%d x=%d r=%d item %d weight %x "
" sw %x rand %llx " ,
i , x , r , bucket - > h . items [ i ] , bucket - > item_weights [ i ] ,
bucket - > sum_weights [ i ] , w ) ;
w * = bucket - > sum_weights [ i ] ;
w = w > > 16 ;
/*dprintk(" scaled %llx\n", w);*/
if ( w < bucket - > item_weights [ i ] )
return bucket - > h . items [ i ] ;
}
2012-05-07 15:35:24 -07:00
dprintk ( " bad list sums for bucket %d \n " , bucket - > h . id ) ;
return bucket - > h . items [ 0 ] ;
2009-10-06 11:31:11 -07:00
}
/* (binary) tree */
static int height ( int n )
{
int h = 0 ;
while ( ( n & 1 ) = = 0 ) {
h + + ;
n = n > > 1 ;
}
return h ;
}
static int left ( int x )
{
int h = height ( x ) ;
return x - ( 1 < < ( h - 1 ) ) ;
}
static int right ( int x )
{
int h = height ( x ) ;
return x + ( 1 < < ( h - 1 ) ) ;
}
static int terminal ( int x )
{
return x & 1 ;
}
static int bucket_tree_choose ( struct crush_bucket_tree * bucket ,
int x , int r )
{
2013-12-24 21:19:24 +02:00
int n ;
2009-10-06 11:31:11 -07:00
__u32 w ;
__u64 t ;
/* start at root */
n = bucket - > num_nodes > > 1 ;
while ( ! terminal ( n ) ) {
2013-12-24 21:19:24 +02:00
int l ;
2009-10-06 11:31:11 -07:00
/* pick point in [0, w) */
w = bucket - > node_weights [ n ] ;
2009-11-07 20:18:22 -08:00
t = ( __u64 ) crush_hash32_4 ( bucket - > h . hash , x , n , r ,
bucket - > h . id ) * ( __u64 ) w ;
2009-10-06 11:31:11 -07:00
t = t > > 32 ;
/* descend to the left or right? */
l = left ( n ) ;
if ( t < bucket - > node_weights [ l ] )
n = l ;
else
n = right ( n ) ;
}
return bucket - > h . items [ n > > 1 ] ;
}
/* straw */
static int bucket_straw_choose ( struct crush_bucket_straw * bucket ,
int x , int r )
{
2012-05-07 15:38:35 -07:00
__u32 i ;
2009-10-06 11:31:11 -07:00
int high = 0 ;
__u64 high_draw = 0 ;
__u64 draw ;
for ( i = 0 ; i < bucket - > h . size ; i + + ) {
2009-11-07 20:18:22 -08:00
draw = crush_hash32_3 ( bucket - > h . hash , x , bucket - > h . items [ i ] , r ) ;
2009-10-06 11:31:11 -07:00
draw & = 0xffff ;
draw * = bucket - > straws [ i ] ;
if ( i = = 0 | | draw > high_draw ) {
high = i ;
high_draw = draw ;
}
}
return bucket - > h . items [ high ] ;
}
static int crush_bucket_choose ( struct crush_bucket * in , int x , int r )
{
2010-06-24 12:58:14 -07:00
dprintk ( " crush_bucket_choose %d x=%d r=%d \n " , in - > id , x , r ) ;
2012-05-07 15:35:24 -07:00
BUG_ON ( in - > size = = 0 ) ;
2009-10-06 11:31:11 -07:00
switch ( in - > alg ) {
case CRUSH_BUCKET_UNIFORM :
return bucket_uniform_choose ( ( struct crush_bucket_uniform * ) in ,
x , r ) ;
case CRUSH_BUCKET_LIST :
return bucket_list_choose ( ( struct crush_bucket_list * ) in ,
x , r ) ;
case CRUSH_BUCKET_TREE :
return bucket_tree_choose ( ( struct crush_bucket_tree * ) in ,
x , r ) ;
case CRUSH_BUCKET_STRAW :
return bucket_straw_choose ( ( struct crush_bucket_straw * ) in ,
x , r ) ;
default :
2012-05-07 15:35:24 -07:00
dprintk ( " unknown bucket %d alg %d \n " , in - > id , in - > alg ) ;
2009-12-01 14:12:07 -08:00
return in - > items [ 0 ] ;
2009-10-06 11:31:11 -07:00
}
}
/*
* true if device is marked " out " ( failed , fully offloaded )
* of the cluster
*/
2013-12-24 21:19:24 +02:00
static int is_out ( const struct crush_map * map ,
const __u32 * weight , int weight_max ,
int item , int x )
2009-10-06 11:31:11 -07:00
{
2013-12-24 21:19:24 +02:00
if ( item > = weight_max )
return 1 ;
2010-07-05 09:44:17 -07:00
if ( weight [ item ] > = 0x10000 )
2009-10-06 11:31:11 -07:00
return 0 ;
if ( weight [ item ] = = 0 )
return 1 ;
2009-11-07 20:18:22 -08:00
if ( ( crush_hash32_2 ( CRUSH_HASH_RJENKINS1 , x , item ) & 0xffff )
< weight [ item ] )
2009-10-06 11:31:11 -07:00
return 0 ;
return 1 ;
}
/**
2013-12-24 21:19:25 +02:00
* crush_choose_firstn - choose numrep distinct items of given type
2009-10-06 11:31:11 -07:00
* @ map : the crush_map
* @ bucket : the bucket we are choose an item from
* @ x : crush input value
* @ numrep : the number of items to choose
* @ type : the type of item to choose
* @ out : pointer to output vector
* @ outpos : our position in that vector
* @ recurse_to_leaf : true if we want one device under each item of given type
* @ out2 : second output vector for leaf items ( if @ recurse_to_leaf )
*/
2013-12-24 21:19:25 +02:00
static int crush_choose_firstn ( const struct crush_map * map ,
struct crush_bucket * bucket ,
const __u32 * weight , int weight_max ,
int x , int numrep , int type ,
int * out , int outpos ,
2013-12-24 21:19:26 +02:00
unsigned int attempts ,
unsigned int recurse_attempts ,
2013-12-24 21:19:27 +02:00
unsigned int local_tries ,
unsigned int local_fallback_tries ,
2013-12-24 21:19:25 +02:00
int recurse_to_leaf ,
2013-12-24 21:19:26 +02:00
int * out2 )
2009-10-06 11:31:11 -07:00
{
int rep ;
2012-05-07 15:38:35 -07:00
unsigned int ftotal , flocal ;
2009-10-06 11:31:11 -07:00
int retry_descent , retry_bucket , skip_rep ;
struct crush_bucket * in = bucket ;
int r ;
int i ;
2009-10-07 10:59:34 -07:00
int item = 0 ;
2009-10-06 11:31:11 -07:00
int itemtype ;
int collide , reject ;
2010-06-24 12:58:14 -07:00
dprintk ( " CHOOSE%s bucket %d x %d outpos %d numrep %d \n " , recurse_to_leaf ? " _LEAF " : " " ,
bucket - > id , x , outpos , numrep ) ;
2009-10-06 11:31:11 -07:00
for ( rep = outpos ; rep < numrep ; rep + + ) {
/* keep trying until we get a non-out, non-colliding item */
ftotal = 0 ;
skip_rep = 0 ;
do {
retry_descent = 0 ;
in = bucket ; /* initial bucket */
/* choose through intervening buckets */
flocal = 0 ;
do {
2009-10-07 10:59:34 -07:00
collide = 0 ;
2009-10-06 11:31:11 -07:00
retry_bucket = 0 ;
r = rep ;
2013-12-24 21:19:25 +02:00
/* r' = r + f_total */
r + = ftotal ;
2009-10-06 11:31:11 -07:00
/* bucket choose */
2009-10-07 10:59:34 -07:00
if ( in - > size = = 0 ) {
reject = 1 ;
goto reject ;
}
2013-12-24 21:19:27 +02:00
if ( local_fallback_tries > 0 & &
2012-07-30 18:15:23 -07:00
flocal > = ( in - > size > > 1 ) & &
2013-12-24 21:19:27 +02:00
flocal > local_fallback_tries )
2009-10-06 11:31:11 -07:00
item = bucket_perm_choose ( in , x , r ) ;
else
item = crush_bucket_choose ( in , x , r ) ;
2012-05-07 15:35:24 -07:00
if ( item > = map - > max_devices ) {
dprintk ( " bad item %d \n " , item ) ;
skip_rep = 1 ;
break ;
}
2009-10-06 11:31:11 -07:00
/* desired type? */
if ( item < 0 )
itemtype = map - > buckets [ - 1 - item ] - > type ;
else
itemtype = 0 ;
dprintk ( " item %d type %d \n " , item , itemtype ) ;
/* keep going? */
if ( itemtype ! = type ) {
2012-05-07 15:35:24 -07:00
if ( item > = 0 | |
( - 1 - item ) > = map - > max_buckets ) {
dprintk ( " bad item type %d \n " , type ) ;
skip_rep = 1 ;
break ;
}
2009-10-06 11:31:11 -07:00
in = map - > buckets [ - 1 - item ] ;
2010-06-24 12:55:48 -07:00
retry_bucket = 1 ;
2009-10-06 11:31:11 -07:00
continue ;
}
/* collision? */
for ( i = 0 ; i < outpos ; i + + ) {
if ( out [ i ] = = item ) {
collide = 1 ;
break ;
}
}
2010-06-24 12:58:14 -07:00
reject = 0 ;
2013-01-15 18:49:09 -08:00
if ( ! collide & & recurse_to_leaf ) {
2010-06-24 12:58:14 -07:00
if ( item < 0 ) {
2013-12-24 21:19:25 +02:00
if ( crush_choose_firstn ( map ,
2010-06-24 12:58:14 -07:00
map - > buckets [ - 1 - item ] ,
2013-12-24 21:19:24 +02:00
weight , weight_max ,
2010-06-24 12:58:14 -07:00
x , outpos + 1 , 0 ,
out2 , outpos ,
2013-12-24 21:19:26 +02:00
recurse_attempts , 0 ,
2013-12-24 21:19:27 +02:00
local_tries ,
local_fallback_tries ,
2013-12-24 21:19:25 +02:00
0 ,
2010-06-24 12:58:14 -07:00
NULL ) < = outpos )
/* didn't get leaf */
reject = 1 ;
} else {
/* we already have a leaf! */
out2 [ outpos ] = item ;
}
}
if ( ! reject ) {
2009-10-06 11:31:11 -07:00
/* out? */
if ( itemtype = = 0 )
reject = is_out ( map , weight ,
2013-12-24 21:19:24 +02:00
weight_max ,
2009-10-06 11:31:11 -07:00
item , x ) ;
else
reject = 0 ;
}
2009-10-07 10:59:34 -07:00
reject :
2009-10-06 11:31:11 -07:00
if ( reject | | collide ) {
ftotal + + ;
flocal + + ;
2013-12-24 21:19:27 +02:00
if ( collide & & flocal < = local_tries )
2009-10-06 11:31:11 -07:00
/* retry locally a few times */
retry_bucket = 1 ;
2013-12-24 21:19:27 +02:00
else if ( local_fallback_tries > 0 & &
flocal < = in - > size + local_fallback_tries )
2009-10-06 11:31:11 -07:00
/* exhaustive bucket search */
retry_bucket = 1 ;
2013-12-24 21:19:26 +02:00
else if ( ftotal < = attempts )
2009-10-06 11:31:11 -07:00
/* then retry descent */
retry_descent = 1 ;
else
/* else give up */
skip_rep = 1 ;
dprintk ( " reject %d collide %d "
2012-05-07 15:38:35 -07:00
" ftotal %u flocal %u \n " ,
2009-10-06 11:31:11 -07:00
reject , collide , ftotal ,
flocal ) ;
}
} while ( retry_bucket ) ;
} while ( retry_descent ) ;
if ( skip_rep ) {
2013-12-24 21:19:25 +02:00
dprintk ( " skip rep \n " ) ;
continue ;
2009-10-06 11:31:11 -07:00
}
2010-06-24 12:58:14 -07:00
dprintk ( " CHOOSE got %d \n " , item ) ;
2009-10-06 11:31:11 -07:00
out [ outpos ] = item ;
outpos + + ;
}
2010-06-24 12:58:14 -07:00
dprintk ( " CHOOSE returns %d \n " , outpos ) ;
2009-10-06 11:31:11 -07:00
return outpos ;
}
2013-12-24 21:19:25 +02:00
/**
2013-12-24 21:19:25 +02:00
* crush_choose_indep : alternative breadth - first positionally stable mapping
2013-12-24 21:19:25 +02:00
*
*/
static void crush_choose_indep ( const struct crush_map * map ,
2013-12-24 21:19:26 +02:00
struct crush_bucket * bucket ,
const __u32 * weight , int weight_max ,
2013-12-24 21:19:25 +02:00
int x , int left , int numrep , int type ,
2013-12-24 21:19:26 +02:00
int * out , int outpos ,
unsigned int attempts ,
unsigned int recurse_attempts ,
int recurse_to_leaf ,
2013-12-24 21:19:25 +02:00
int * out2 ,
int parent_r )
2013-12-24 21:19:25 +02:00
{
struct crush_bucket * in = bucket ;
2013-12-24 21:19:25 +02:00
int endpos = outpos + left ;
2013-12-24 21:19:25 +02:00
int rep ;
unsigned int ftotal ;
int r ;
int i ;
int item = 0 ;
int itemtype ;
int collide ;
dprintk ( " CHOOSE%s INDEP bucket %d x %d outpos %d numrep %d \n " , recurse_to_leaf ? " _LEAF " : " " ,
bucket - > id , x , outpos , numrep ) ;
/* initially my result is undefined */
2013-12-24 21:19:25 +02:00
for ( rep = outpos ; rep < endpos ; rep + + ) {
2013-12-24 21:19:25 +02:00
out [ rep ] = CRUSH_ITEM_UNDEF ;
if ( out2 )
out2 [ rep ] = CRUSH_ITEM_UNDEF ;
}
2013-12-24 21:19:26 +02:00
for ( ftotal = 0 ; left > 0 & & ftotal < attempts ; ftotal + + ) {
2013-12-24 21:19:25 +02:00
for ( rep = outpos ; rep < endpos ; rep + + ) {
2013-12-24 21:19:25 +02:00
if ( out [ rep ] ! = CRUSH_ITEM_UNDEF )
continue ;
in = bucket ; /* initial bucket */
/* choose through intervening buckets */
for ( ; ; ) {
2013-12-24 21:19:25 +02:00
/* note: we base the choice on the position
* even in the nested call . that means that
* if the first layer chooses the same bucket
* in a different position , we will tend to
* choose a different item in that bucket .
* this will involve more devices in data
* movement and tend to distribute the load .
*/
2013-12-24 21:19:25 +02:00
r = rep + parent_r ;
2013-12-24 21:19:25 +02:00
/* be careful */
if ( in - > alg = = CRUSH_BUCKET_UNIFORM & &
in - > size % numrep = = 0 )
/* r'=r+(n+1)*f_total */
r + = ( numrep + 1 ) * ftotal ;
else
/* r' = r + n*f_total */
r + = numrep * ftotal ;
/* bucket choose */
if ( in - > size = = 0 ) {
dprintk ( " empty bucket \n " ) ;
break ;
}
item = crush_bucket_choose ( in , x , r ) ;
if ( item > = map - > max_devices ) {
dprintk ( " bad item %d \n " , item ) ;
out [ rep ] = CRUSH_ITEM_NONE ;
if ( out2 )
out2 [ rep ] = CRUSH_ITEM_NONE ;
left - - ;
break ;
}
/* desired type? */
if ( item < 0 )
itemtype = map - > buckets [ - 1 - item ] - > type ;
else
itemtype = 0 ;
dprintk ( " item %d type %d \n " , item , itemtype ) ;
/* keep going? */
if ( itemtype ! = type ) {
if ( item > = 0 | |
( - 1 - item ) > = map - > max_buckets ) {
dprintk ( " bad item type %d \n " , type ) ;
out [ rep ] = CRUSH_ITEM_NONE ;
if ( out2 )
out2 [ rep ] =
CRUSH_ITEM_NONE ;
left - - ;
break ;
}
in = map - > buckets [ - 1 - item ] ;
continue ;
}
/* collision? */
collide = 0 ;
2013-12-24 21:19:25 +02:00
for ( i = outpos ; i < endpos ; i + + ) {
2013-12-24 21:19:25 +02:00
if ( out [ i ] = = item ) {
collide = 1 ;
break ;
}
}
if ( collide )
break ;
if ( recurse_to_leaf ) {
if ( item < 0 ) {
crush_choose_indep ( map ,
2013-12-24 21:19:26 +02:00
map - > buckets [ - 1 - item ] ,
weight , weight_max ,
x , 1 , numrep , 0 ,
out2 , rep ,
recurse_attempts , 0 ,
0 , NULL , r ) ;
2013-12-24 21:19:25 +02:00
if ( out2 [ rep ] = = CRUSH_ITEM_NONE ) {
/* placed nothing; no leaf */
break ;
}
} else {
/* we already have a leaf! */
out2 [ rep ] = item ;
}
}
/* out? */
if ( itemtype = = 0 & &
is_out ( map , weight , weight_max , item , x ) )
break ;
/* yay! */
out [ rep ] = item ;
left - - ;
break ;
}
}
}
2013-12-24 21:19:25 +02:00
for ( rep = outpos ; rep < endpos ; rep + + ) {
2013-12-24 21:19:25 +02:00
if ( out [ rep ] = = CRUSH_ITEM_UNDEF ) {
out [ rep ] = CRUSH_ITEM_NONE ;
}
if ( out2 & & out2 [ rep ] = = CRUSH_ITEM_UNDEF ) {
out2 [ rep ] = CRUSH_ITEM_NONE ;
}
}
}
2009-10-06 11:31:11 -07:00
/**
* crush_do_rule - calculate a mapping with the given input and rule
* @ map : the crush_map
* @ ruleno : the rule id
* @ x : hash input
* @ result : pointer to result vector
* @ result_max : maximum result size
2013-12-24 21:19:24 +02:00
* @ weight : weight vector ( for map leaves )
* @ weight_max : size of weight vector
2013-12-24 21:19:24 +02:00
* @ scratch : scratch vector for private use ; must be > = 3 * result_max
2009-10-06 11:31:11 -07:00
*/
2012-05-07 15:38:35 -07:00
int crush_do_rule ( const struct crush_map * map ,
2009-10-06 11:31:11 -07:00
int ruleno , int x , int * result , int result_max ,
2013-12-24 21:19:24 +02:00
const __u32 * weight , int weight_max ,
int * scratch )
2009-10-06 11:31:11 -07:00
{
int result_len ;
2013-12-24 21:19:24 +02:00
int * a = scratch ;
int * b = scratch + result_max ;
int * c = scratch + result_max * 2 ;
2009-10-06 11:31:11 -07:00
int recurse_to_leaf ;
int * w ;
int wsize = 0 ;
int * o ;
int osize ;
int * tmp ;
struct crush_rule * rule ;
2012-05-07 15:38:35 -07:00
__u32 step ;
2009-10-06 11:31:11 -07:00
int i , j ;
int numrep ;
2013-12-24 21:19:26 +02:00
int choose_tries = map - > choose_total_tries ;
2013-12-24 21:19:27 +02:00
int choose_local_tries = map - > choose_local_tries ;
int choose_local_fallback_tries = map - > choose_local_fallback_tries ;
2013-12-24 21:19:26 +02:00
int choose_leaf_tries = 0 ;
2009-10-06 11:31:11 -07:00
2012-05-07 15:35:24 -07:00
if ( ( __u32 ) ruleno > = map - > max_rules ) {
dprintk ( " bad ruleno %d \n " , ruleno ) ;
return 0 ;
}
2009-10-06 11:31:11 -07:00
rule = map - > rules [ ruleno ] ;
result_len = 0 ;
w = a ;
o = b ;
for ( step = 0 ; step < rule - > len ; step + + ) {
2013-12-24 21:19:24 +02:00
int firstn = 0 ;
2012-05-07 15:35:48 -07:00
struct crush_rule_step * curstep = & rule - > steps [ step ] ;
switch ( curstep - > op ) {
2009-10-06 11:31:11 -07:00
case CRUSH_RULE_TAKE :
2012-05-07 15:35:48 -07:00
w [ 0 ] = curstep - > arg1 ;
2009-10-06 11:31:11 -07:00
wsize = 1 ;
break ;
2013-12-24 21:19:26 +02:00
case CRUSH_RULE_SET_CHOOSE_TRIES :
if ( curstep - > arg1 > 0 )
choose_tries = curstep - > arg1 ;
break ;
2013-12-24 21:19:26 +02:00
case CRUSH_RULE_SET_CHOOSELEAF_TRIES :
2013-12-24 21:19:26 +02:00
if ( curstep - > arg1 > 0 )
choose_leaf_tries = curstep - > arg1 ;
break ;
2013-12-24 21:19:27 +02:00
case CRUSH_RULE_SET_CHOOSE_LOCAL_TRIES :
if ( curstep - > arg1 > 0 )
choose_local_tries = curstep - > arg1 ;
break ;
case CRUSH_RULE_SET_CHOOSE_LOCAL_FALLBACK_TRIES :
if ( curstep - > arg1 > 0 )
choose_local_fallback_tries = curstep - > arg1 ;
break ;
2013-12-24 21:19:26 +02:00
case CRUSH_RULE_CHOOSELEAF_FIRSTN :
2009-10-06 11:31:11 -07:00
case CRUSH_RULE_CHOOSE_FIRSTN :
firstn = 1 ;
2012-05-07 15:35:48 -07:00
/* fall through */
2013-12-24 21:19:26 +02:00
case CRUSH_RULE_CHOOSELEAF_INDEP :
2009-10-06 11:31:11 -07:00
case CRUSH_RULE_CHOOSE_INDEP :
2012-05-07 15:35:24 -07:00
if ( wsize = = 0 )
break ;
2009-10-06 11:31:11 -07:00
recurse_to_leaf =
2012-05-07 15:35:48 -07:00
curstep - > op = =
2013-12-24 21:19:26 +02:00
CRUSH_RULE_CHOOSELEAF_FIRSTN | |
2012-05-07 15:35:48 -07:00
curstep - > op = =
2013-12-24 21:19:26 +02:00
CRUSH_RULE_CHOOSELEAF_INDEP ;
2009-10-06 11:31:11 -07:00
/* reset output */
osize = 0 ;
for ( i = 0 ; i < wsize ; i + + ) {
/*
* see CRUSH_N , CRUSH_N_MINUS macros .
* basically , numrep < = 0 means relative to
* the provided result_max
*/
2012-05-07 15:35:48 -07:00
numrep = curstep - > arg1 ;
2009-10-06 11:31:11 -07:00
if ( numrep < = 0 ) {
numrep + = result_max ;
if ( numrep < = 0 )
continue ;
}
j = 0 ;
2013-12-24 21:19:25 +02:00
if ( firstn ) {
2013-12-24 21:19:26 +02:00
int recurse_tries ;
if ( choose_leaf_tries )
recurse_tries =
choose_leaf_tries ;
else if ( map - > chooseleaf_descend_once )
recurse_tries = 1 ;
else
recurse_tries = choose_tries ;
2013-12-24 21:19:25 +02:00
osize + = crush_choose_firstn (
map ,
map - > buckets [ - 1 - w [ i ] ] ,
weight , weight_max ,
x , numrep ,
curstep - > arg2 ,
o + osize , j ,
2013-12-24 21:19:26 +02:00
choose_tries ,
2013-12-24 21:19:26 +02:00
recurse_tries ,
2013-12-24 21:19:27 +02:00
choose_local_tries ,
choose_local_fallback_tries ,
2013-12-24 21:19:25 +02:00
recurse_to_leaf ,
2013-12-24 21:19:26 +02:00
c + osize ) ;
2013-12-24 21:19:25 +02:00
} else {
2013-12-24 21:19:25 +02:00
crush_choose_indep (
map ,
map - > buckets [ - 1 - w [ i ] ] ,
weight , weight_max ,
2013-12-24 21:19:25 +02:00
x , numrep , numrep ,
2013-12-24 21:19:25 +02:00
curstep - > arg2 ,
o + osize , j ,
2013-12-24 21:19:26 +02:00
choose_tries ,
2013-12-24 21:19:26 +02:00
choose_leaf_tries ?
choose_leaf_tries : 1 ,
2013-12-24 21:19:25 +02:00
recurse_to_leaf ,
2013-12-24 21:19:25 +02:00
c + osize ,
0 ) ;
2013-12-24 21:19:25 +02:00
osize + = numrep ;
}
2009-10-06 11:31:11 -07:00
}
if ( recurse_to_leaf )
/* copy final _leaf_ values to output set */
memcpy ( o , c , osize * sizeof ( * o ) ) ;
2013-12-24 21:19:24 +02:00
/* swap o and w arrays */
2009-10-06 11:31:11 -07:00
tmp = o ;
o = w ;
w = tmp ;
wsize = osize ;
break ;
case CRUSH_RULE_EMIT :
for ( i = 0 ; i < wsize & & result_len < result_max ; i + + ) {
result [ result_len ] = w [ i ] ;
result_len + + ;
}
wsize = 0 ;
break ;
default :
2012-05-07 15:35:24 -07:00
dprintk ( " unknown op %d at step %d \n " ,
curstep - > op , step ) ;
break ;
2009-10-06 11:31:11 -07:00
}
}
2011-12-07 09:10:26 -08:00
return result_len ;
2009-10-06 11:31:11 -07:00
}