blob: 7a48a82333c4e233ae4f9acca074589e7ebab535 [file] [log] [blame]
/*
* mdadm - manage Linux "md" devices aka RAID arrays.
*
* Copyright (C) 2011 Neil Brown <neilb@suse.de>
*
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* Author: Neil Brown
* Email: <neilb@suse.de>
*/
#include "mdadm.h"
#include "mdadm_internal.h"
#include "debug.h"
#include "mapfile.h"
#include "policy.h"
/* faulty stuff */
#define WriteTransient 0
#define ReadTransient 1
#define WritePersistent 2
#define ReadPersistent 3
#define WriteAll 4 /* doesn't go to device */
#define ReadFixable 5
#define Modes 6
#define ClearErrors 31
#define ClearFaults 30
enum r0layout {
RAID0_ORIG_LAYOUT = 1,
RAID0_ALT_MULTIZONE_LAYOUT = 2,
};
/* name/number mappings */
mapping_t r5layout[] = {
{ "left-asymmetric", ALGORITHM_LEFT_ASYMMETRIC},
{ "right-asymmetric", ALGORITHM_RIGHT_ASYMMETRIC},
{ "left-symmetric", ALGORITHM_LEFT_SYMMETRIC},
{ "right-symmetric", ALGORITHM_RIGHT_SYMMETRIC},
{ "default", ALGORITHM_LEFT_SYMMETRIC},
{ "la", ALGORITHM_LEFT_ASYMMETRIC},
{ "ra", ALGORITHM_RIGHT_ASYMMETRIC},
{ "ls", ALGORITHM_LEFT_SYMMETRIC},
{ "rs", ALGORITHM_RIGHT_SYMMETRIC},
{ "parity-first", ALGORITHM_PARITY_0},
{ "parity-last", ALGORITHM_PARITY_N},
{ "ddf-zero-restart", ALGORITHM_RIGHT_ASYMMETRIC},
{ "ddf-N-restart", ALGORITHM_LEFT_ASYMMETRIC},
{ "ddf-N-continue", ALGORITHM_LEFT_SYMMETRIC},
{ NULL, UnSet }
};
mapping_t r6layout[] = {
{ "left-asymmetric", ALGORITHM_LEFT_ASYMMETRIC},
{ "right-asymmetric", ALGORITHM_RIGHT_ASYMMETRIC},
{ "left-symmetric", ALGORITHM_LEFT_SYMMETRIC},
{ "right-symmetric", ALGORITHM_RIGHT_SYMMETRIC},
{ "default", ALGORITHM_LEFT_SYMMETRIC},
{ "la", ALGORITHM_LEFT_ASYMMETRIC},
{ "ra", ALGORITHM_RIGHT_ASYMMETRIC},
{ "ls", ALGORITHM_LEFT_SYMMETRIC},
{ "rs", ALGORITHM_RIGHT_SYMMETRIC},
{ "parity-first", ALGORITHM_PARITY_0},
{ "parity-last", ALGORITHM_PARITY_N},
{ "ddf-zero-restart", ALGORITHM_ROTATING_ZERO_RESTART},
{ "ddf-N-restart", ALGORITHM_ROTATING_N_RESTART},
{ "ddf-N-continue", ALGORITHM_ROTATING_N_CONTINUE},
{ "left-asymmetric-6", ALGORITHM_LEFT_ASYMMETRIC_6},
{ "right-asymmetric-6", ALGORITHM_RIGHT_ASYMMETRIC_6},
{ "left-symmetric-6", ALGORITHM_LEFT_SYMMETRIC_6},
{ "right-symmetric-6", ALGORITHM_RIGHT_SYMMETRIC_6},
{ "parity-first-6", ALGORITHM_PARITY_0_6},
{ NULL, UnSet }
};
/* raid0 layout is only needed because of a bug in 3.14 which changed
* the effective layout of raid0 arrays with varying device sizes.
*/
mapping_t r0layout[] = {
{ "original", RAID0_ORIG_LAYOUT},
{ "alternate", RAID0_ALT_MULTIZONE_LAYOUT},
{ "1", 1}, /* aka ORIG */
{ "2", 2}, /* aka ALT */
{ "dangerous", 0},
{ NULL, UnSet},
};
mapping_t pers[] = {
{ "linear", LEVEL_LINEAR},
{ "raid0", 0},
{ "0", 0},
{ "stripe", 0},
{ "raid1", 1},
{ "1", 1},
{ "mirror", 1},
{ "raid4", 4},
{ "4", 4},
{ "raid5", 5},
{ "5", 5},
{ "multipath", LEVEL_MULTIPATH},
{ "mp", LEVEL_MULTIPATH},
{ "raid6", 6},
{ "6", 6},
{ "raid10", 10},
{ "10", 10},
{ "faulty", LEVEL_FAULTY},
{ "container", LEVEL_CONTAINER},
{ NULL, UnSet }
};
mapping_t faultylayout[] = {
{ "write-transient", WriteTransient },
{ "wt", WriteTransient },
{ "read-transient", ReadTransient },
{ "rt", ReadTransient },
{ "write-persistent", WritePersistent },
{ "wp", WritePersistent },
{ "read-persistent", ReadPersistent },
{ "rp", ReadPersistent },
{ "write-all", WriteAll },
{ "wa", WriteAll },
{ "read-fixable", ReadFixable },
{ "rf", ReadFixable },
{ "clear", ClearErrors},
{ "flush", ClearFaults},
{ "none", ClearErrors},
{ "default", ClearErrors},
{ NULL, UnSet }
};
mapping_t consistency_policies[] = {
{ "unknown", CONSISTENCY_POLICY_UNKNOWN},
{ "none", CONSISTENCY_POLICY_NONE},
{ "resync", CONSISTENCY_POLICY_RESYNC},
{ "bitmap", CONSISTENCY_POLICY_BITMAP},
{ "journal", CONSISTENCY_POLICY_JOURNAL},
{ "ppl", CONSISTENCY_POLICY_PPL},
{ NULL, CONSISTENCY_POLICY_UNKNOWN }
};
mapping_t sysfs_array_states[] = {
{ "active-idle", ARRAY_ACTIVE_IDLE },
{ "active", ARRAY_ACTIVE },
{ "clear", ARRAY_CLEAR },
{ "inactive", ARRAY_INACTIVE },
{ "suspended", ARRAY_SUSPENDED },
{ "readonly", ARRAY_READONLY },
{ "read-auto", ARRAY_READ_AUTO },
{ "clean", ARRAY_CLEAN },
{ "write-pending", ARRAY_WRITE_PENDING },
{ "broken", ARRAY_BROKEN },
{ NULL, ARRAY_UNKNOWN_STATE }
};
mapping_t assemble_statuses[] = {
{ "but cannot be started", INCR_NO },
{ "but not safe to start", INCR_UNSAFE },
{ "and started", INCR_YES },
{ NULL, INCR_ALREADY }
};
char *map_num(mapping_t *map, int num)
{
while (map->name) {
if (map->num == num)
return map->name;
map++;
}
return NULL;
}
static int map_name(mapping_t *map, char *name)
{
while (map->name && strcmp(map->name, name) != 0)
map++;
return map->num;
}
int mdadm_get_layout(int level, char *name)
{
int layout;
switch(level) {
case UnSet:
pr_err("raid level must be given before layout.\n");
return -EINVAL;
case 0:
layout = map_name(r0layout, optarg);
if (layout == UnSet) {
pr_err("layout %s not understood for raid0.\n", name);
return -EINVAL;
}
break;
case 5:
layout = map_name(r5layout, optarg);
if (layout == UnSet) {
pr_err("layout %s not understood for raid5.\n", name);
return -EINVAL;
}
break;
case 6:
layout = map_name(r6layout, optarg);
if (layout == UnSet) {
pr_err("layout %s not understood for raid6.\n", name);
return -EINVAL;
}
break;
case 10:
layout = parse_layout_10(optarg);
if (layout < 0) {
pr_err("layout for raid10 must be 'nNN', 'oNN' or 'fNN' where NN is a number, not %s\n", name);
return -EINVAL;
}
break;
case LEVEL_FAULTY:
/* Faulty
* modeNNN
*/
layout = parse_layout_faulty(optarg);
if (layout < 0) {
pr_err("layout %s not understood for faulty.\n",
optarg);
return -EINVAL;
}
break;
default:
pr_err("layout not meaningful for %s arrays.\n",
map_num(pers, level));
return -EAGAIN;
}
return layout;
}
int mdadm_default_layout(int level, int verbose)
{
int layout = 0; /* no layout */
switch(level) {
default:
break;
case 10:
layout = 0x102; /* near=2, far=1 */
if (verbose > 0)
pr_err("layout defaults to n1\n");
break;
case 5:
case 6:
layout = map_name(r5layout, "default");
if (verbose > 0)
pr_err("layout defaults to %s\n",
map_num(r5layout, layout));
break;
case 0:
layout = RAID0_ORIG_LAYOUT;
break;
case LEVEL_FAULTY:
layout = map_name(faultylayout, "default");
if (verbose > 0)
pr_err("layout defaults to %s\n",
map_num(faultylayout, layout));
break;
}
return layout;
}
int mdadm_faulty_layout(char *name)
{
return map_name(faultylayout, name);
}
char *mdadm_raid_layout_name(int level, int layout)
{
switch (level) {
case 0:
return map_num(r0layout, layout);
case 5:
return map_num(r5layout, layout);
case 6:
return map_num(r6layout, layout);
default:
break;
}
return NULL;
}
int mdadm_raid_layout_num(int level, char *layout)
{
switch (level) {
case 0:
return map_name(r0layout, layout);
case 5:
return map_name(r5layout, layout);
case 6:
return map_name(r6layout, layout);
default:
break;
}
return -1;
}
int mdadm_personality_num(char *name)
{
return map_name(pers, name);
}
char *mdadm_personality_name(int num)
{
return map_num(pers, num);
}
int mdadm_consistency_policy_num(char *name)
{
return map_name(consistency_policies, name);
}
char *mdadm_consistency_policy_name(int num)
{
return map_num(consistency_policies, num);
}
int mdadm_array_state_num(char *name)
{
return map_name(sysfs_array_states, name);
}
char *mdadm_array_state_name(int num)
{
return map_num(sysfs_array_states, num);
}
char *mdadm_assemble_status(int status)
{
return map_num(assemble_statuses, status);
}