Skip to content

Commit

Permalink
Routine to parse a rotor vector configuration.
Browse files Browse the repository at this point in the history
Probably the configuration should be split into multiple properties, one per rotor, and limit separate from types.

Format:

[spec<=limit];spec

with spec being a comma-separated list of vdev-guids and generic type specifiers: ssd, ssd-raidz, mixed, hdd, and hdd-raidz.

limit gives the limit in kbytes of which allocations are allowed within the rotor vector.  The last rotor has no limit.  (And vdevs which are not matched by the guids or the generic types are placed in the last rotor.)

For the moment things are a bit difficult due to the 32 character limit of the comment field.  That is also why the limit is implicitly expressed in units of 1024 bytes.

Example:

ssd<=4;mixed<=64;123,hdd

Allocations less than 4 kbytes are allocated on an ssd-only vdev (mirror or not).
Allocations less than 64 kbytes end up on ssd/hdd mixed (mirrors, such raidz makes no sense).
Other allocations end up on remaining disks.
123 represents av vdev guid.
  • Loading branch information
inkdot7 committed Dec 10, 2016
1 parent 12917df commit 522f535
Show file tree
Hide file tree
Showing 2 changed files with 189 additions and 2 deletions.
2 changes: 2 additions & 0 deletions include/sys/metaslab.h
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,8 @@ uint64_t metaslab_class_get_space(metaslab_class_t *);
uint64_t metaslab_class_get_dspace(metaslab_class_t *);
uint64_t metaslab_class_get_deferred(metaslab_class_t *);

void metaslab_parse_rotor_config(metaslab_class_t *, char *);

metaslab_group_t *metaslab_group_create(metaslab_class_t *, vdev_t *);
void metaslab_group_destroy(metaslab_group_t *);
void metaslab_group_set_rotor_category(metaslab_group_t *, boolean_t);
Expand Down
189 changes: 187 additions & 2 deletions module/zfs/metaslab.c
Original file line number Diff line number Diff line change
Expand Up @@ -192,8 +192,6 @@ metaslab_class_create(spa_t *spa, metaslab_ops_t *ops, char *rvconfig)
metaslab_class_t *mc;
int i;

(void) rvconfig;

mc = kmem_zalloc(sizeof (metaslab_class_t), KM_SLEEP);

mc->mc_spa = spa;
Expand All @@ -204,6 +202,8 @@ metaslab_class_create(spa_t *spa, metaslab_ops_t *ops, char *rvconfig)
refcount_create_tracked(&mc->mc_alloc_slots);
mc->mc_max_nrot = -1;

metaslab_parse_rotor_config(mc, rvconfig);

return (mc);
}

Expand Down Expand Up @@ -580,6 +580,191 @@ metaslab_group_alloc_update(metaslab_group_t *mg)
mutex_exit(&mg->mg_lock);
}

/*
* Please do not judge the rotor vector approach based on the ugliness
* of this parsing routine. :-)
*/
void
metaslab_parse_rotor_config(metaslab_class_t *mc, char *rotorvector)
{
int nrot = 0;
char *endconfig;

if (rotorvector == NULL)
return;

/*
* The list consist of colon-separated categories.
* Each catagory is a comma-separated list of vdev guids or
* types (ssd,ssd-raidz,mixed,hdd,hdd-raidz), followed
* by <= and the allocation threshold for the category.
* ('<=' since users likely want to specify that rather than '<')
* The threshold for the last rotor is not to be given, as we
* always allocate somewhere.
*/

/* Clear the configuration. */
memset(mc->mc_rotvec_threshold, 0, sizeof (mc->mc_rotvec_threshold));
memset(mc->mc_rotvec_vdev_guids, 0, sizeof (mc->mc_rotvec_vdev_guids));
memset(mc->mc_rotvec_categories, 0, sizeof (mc->mc_rotvec_categories));

/*
* The returns when encountering a malformed configuration are
* ok in the sense that we just do not fill the fields any
* further.
*/

/*
* It would probably be better to have separate properties
* for each vector category and each vector threshold.
* Much less parsing logics.
*/

endconfig = rotorvector + strlen(rotorvector);

while (rotorvector < endconfig) {
char *nextrotor;
char *semicolon, *lessthan;
char *endtypes;
int nguids = 0;

nextrotor = endconfig;
semicolon = strchr(rotorvector, ';');
if (semicolon == NULL)
semicolon = endconfig;
else
nextrotor = semicolon+1;
lessthan = strstr(rotorvector, "<=");
if ((lessthan == NULL && semicolon != endconfig) ||
lessthan > semicolon)
return; /* malformed, missing '<=' in this item */
if (lessthan != NULL && semicolon == endconfig)
return; /* malformed, '<=' for last item */

endtypes = (lessthan) ? lessthan : semicolon;

while (rotorvector < endtypes) {
char *comma, *nexttype;
size_t len;

nexttype = endtypes;
comma = strchr(rotorvector, ',');
if (comma == NULL || comma > endtypes)
comma = endtypes;
else
nexttype = comma+1;

len = comma-rotorvector;

if (strncmp(rotorvector, "ssd", len) == 0)
mc->mc_rotvec_categories[nrot] |= 0x01;
else if (strncmp(rotorvector, "ssd-raidz", len) == 0)
mc->mc_rotvec_categories[nrot] |= 0x02;
else if (strncmp(rotorvector, "mixed", len) == 0)
mc->mc_rotvec_categories[nrot] |= 0x04;
else if (strncmp(rotorvector, "hdd", len) == 0)
mc->mc_rotvec_categories[nrot] |= 0x08;
else if (strncmp(rotorvector, "hdd-raidz", len) == 0)
mc->mc_rotvec_categories[nrot] |= 0x10;
else {
/* It must be a vdev guid. */
uint64_t guid;
#ifdef _KERNEL
char tmpstr[64];
size_t len = comma-rotorvector;
strncpy(tmpstr, rotorvector, len);
tmpstr[len] = 0;
#endif
#ifdef _KERNEL
if (kstrtoull(tmpstr, 0, &guid) != 0)
return; /* malformed configuration */
#else
char *endptr;
guid = strtoull(rotorvector, &endptr, 0);
if (endptr != comma)
return; /* malformed configuration */
#endif
if (nguids >= 5)
return; /* too many guids... */
mc->mc_rotvec_vdev_guids[nrot][nguids] = guid;
nguids++;
}
rotorvector = nexttype;
}

if (lessthan) {
uint64_t threshold;
#ifdef _KERNEL
char tmpstr[64];
size_t len = semicolon-(lessthan+2);
strncpy(tmpstr, lessthan+2, len);
tmpstr[len] = 0;
#endif
#ifdef _KERNEL
if (kstrtoull(tmpstr, 0, &threshold) != 0)
return; /* malformed configuration */
#else
char *endptr;
threshold = strtoull(lessthan+2, &endptr, 0);
if (endptr != semicolon)
return; /* malformed configuration */
#endif
/*
* To live with the 32 character limit for the
* comment field, we multiply the threshold by
* 1024 internally.
*/
mc->mc_rotvec_threshold[nrot] = threshold * 1024;
}
rotorvector = nextrotor;
nrot++;
}

#if 0
#ifdef _KERNEL
{
int i;

for (i = 0; i < METASLAB_CLASS_ROTORS; i++) {
int j;

printk("rotvec[%d]: limit:%llu typemask:%02x guids:",
i,
mc->mc_rotvec_threshold[i],
mc->mc_rotvec_categories[i]);
for (j = 0; j < 5 && mc->mc_rotvec_vdev_guids[i][j];
j++) {
printk(" %llu", mc->mc_rotvec_vdev_guids[i][j]);
}
printk("\n");
}
}
#endif
#endif
}

int
metaslab_vdev_rotor_category(vdev_t *vd)
{
/*
* Five categories, from faster to slower:
*
* 0: nonrot (SSD) disk or mirror
* 1: nonrot (SSD) raidz
* 2: mixed nonrot+rot anything (raidz makes little sense)
* 3: rot (HDD) disk or mirror
* 4: rot (HDD) raidz
*/

if (vd->vdev_nonrot) {
return ((vd->vdev_ops != &vdev_raidz_ops) ? 0 : 1);
} else if (vd->vdev_nonrot_mix) {
return (2);
} else {
return ((vd->vdev_ops != &vdev_raidz_ops) ? 3 : 4);
}
}

metaslab_group_t *
metaslab_group_create(metaslab_class_t *mc, vdev_t *vd)
{
Expand Down

0 comments on commit 522f535

Please sign in to comment.