drbd: Print common state changes of all volumes as connection state changes
Signed-off-by: Philipp Reisner <philipp.reisner@linbit.com> Signed-off-by: Lars Ellenberg <lars.ellenberg@linbit.com>
This commit is contained in:
Родитель
88ef594ed7
Коммит
435693e89b
|
@ -350,29 +350,29 @@ void print_st_err(struct drbd_conf *mdev, union drbd_state os,
|
||||||
print_st(mdev, "wanted", ns);
|
print_st(mdev, "wanted", ns);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_state_change(struct drbd_conf *mdev, union drbd_state os, union drbd_state ns,
|
static long print_state_change(char *pb, union drbd_state os, union drbd_state ns,
|
||||||
enum chg_state_flags flags)
|
enum chg_state_flags flags)
|
||||||
{
|
{
|
||||||
char *pbp, pb[300];
|
char *pbp;
|
||||||
pbp = pb;
|
pbp = pb;
|
||||||
*pbp = 0;
|
*pbp = 0;
|
||||||
if (ns.role != os.role)
|
if (ns.role != os.role && flags & CS_DC_ROLE)
|
||||||
pbp += sprintf(pbp, "role( %s -> %s ) ",
|
pbp += sprintf(pbp, "role( %s -> %s ) ",
|
||||||
drbd_role_str(os.role),
|
drbd_role_str(os.role),
|
||||||
drbd_role_str(ns.role));
|
drbd_role_str(ns.role));
|
||||||
if (ns.peer != os.peer)
|
if (ns.peer != os.peer && flags & CS_DC_PEER)
|
||||||
pbp += sprintf(pbp, "peer( %s -> %s ) ",
|
pbp += sprintf(pbp, "peer( %s -> %s ) ",
|
||||||
drbd_role_str(os.peer),
|
drbd_role_str(os.peer),
|
||||||
drbd_role_str(ns.peer));
|
drbd_role_str(ns.peer));
|
||||||
if (ns.conn != os.conn && !(flags & CS_NO_CSTATE_CHG))
|
if (ns.conn != os.conn && flags & CS_DC_CONN)
|
||||||
pbp += sprintf(pbp, "conn( %s -> %s ) ",
|
pbp += sprintf(pbp, "conn( %s -> %s ) ",
|
||||||
drbd_conn_str(os.conn),
|
drbd_conn_str(os.conn),
|
||||||
drbd_conn_str(ns.conn));
|
drbd_conn_str(ns.conn));
|
||||||
if (ns.disk != os.disk)
|
if (ns.disk != os.disk && flags & CS_DC_DISK)
|
||||||
pbp += sprintf(pbp, "disk( %s -> %s ) ",
|
pbp += sprintf(pbp, "disk( %s -> %s ) ",
|
||||||
drbd_disk_str(os.disk),
|
drbd_disk_str(os.disk),
|
||||||
drbd_disk_str(ns.disk));
|
drbd_disk_str(ns.disk));
|
||||||
if (ns.pdsk != os.pdsk)
|
if (ns.pdsk != os.pdsk && flags & CS_DC_PDSK)
|
||||||
pbp += sprintf(pbp, "pdsk( %s -> %s ) ",
|
pbp += sprintf(pbp, "pdsk( %s -> %s ) ",
|
||||||
drbd_disk_str(os.pdsk),
|
drbd_disk_str(os.pdsk),
|
||||||
drbd_disk_str(ns.pdsk));
|
drbd_disk_str(ns.pdsk));
|
||||||
|
@ -392,10 +392,29 @@ static void print_state_change(struct drbd_conf *mdev, union drbd_state os, unio
|
||||||
pbp += sprintf(pbp, "user_isp( %d -> %d ) ",
|
pbp += sprintf(pbp, "user_isp( %d -> %d ) ",
|
||||||
os.user_isp,
|
os.user_isp,
|
||||||
ns.user_isp);
|
ns.user_isp);
|
||||||
if (pbp != pb)
|
|
||||||
|
return pbp - pb;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void drbd_pr_state_change(struct drbd_conf *mdev, union drbd_state os, union drbd_state ns,
|
||||||
|
enum chg_state_flags flags)
|
||||||
|
{
|
||||||
|
char pb[300];
|
||||||
|
|
||||||
|
if (print_state_change(pb, os, ns, flags ^ CS_DC_MASK))
|
||||||
dev_info(DEV, "%s\n", pb);
|
dev_info(DEV, "%s\n", pb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void conn_pr_state_change(struct drbd_tconn *tconn, union drbd_state os, union drbd_state ns,
|
||||||
|
enum chg_state_flags flags)
|
||||||
|
{
|
||||||
|
char pb[300];
|
||||||
|
|
||||||
|
if (print_state_change(pb, os, ns, flags))
|
||||||
|
conn_info(tconn, "%s\n", pb);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* is_valid_state() - Returns an SS_ error code if ns is not valid
|
* is_valid_state() - Returns an SS_ error code if ns is not valid
|
||||||
* @mdev: DRBD device.
|
* @mdev: DRBD device.
|
||||||
|
@ -827,7 +846,7 @@ __drbd_set_state(struct drbd_conf *mdev, union drbd_state ns,
|
||||||
if (warn_sync_abort)
|
if (warn_sync_abort)
|
||||||
dev_warn(DEV, "%s aborted.\n", warn_sync_abort);
|
dev_warn(DEV, "%s aborted.\n", warn_sync_abort);
|
||||||
|
|
||||||
print_state_change(mdev, os, ns, flags);
|
drbd_pr_state_change(mdev, os, ns, flags);
|
||||||
|
|
||||||
/* if we are going -> D_FAILED or D_DISKLESS, grab one extra reference
|
/* if we are going -> D_FAILED or D_DISKLESS, grab one extra reference
|
||||||
* on the ldev here, to be sure the transition -> D_DISKLESS resp.
|
* on the ldev here, to be sure the transition -> D_DISKLESS resp.
|
||||||
|
@ -1364,98 +1383,41 @@ static int w_after_conn_state_ch(struct drbd_work *w, int unused)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_conn_state_change(struct drbd_tconn *tconn, enum drbd_conns oc, enum drbd_conns nc)
|
void conn_old_common_state(struct drbd_tconn *tconn, union drbd_state *pcs, enum chg_state_flags *pf)
|
||||||
{
|
{
|
||||||
char *pbp, pb[300];
|
enum chg_state_flags flags = ~0;
|
||||||
pbp = pb;
|
|
||||||
*pbp = 0;
|
|
||||||
if (nc != oc)
|
|
||||||
pbp += sprintf(pbp, "conn( %s -> %s ) ",
|
|
||||||
drbd_conn_str(oc),
|
|
||||||
drbd_conn_str(nc));
|
|
||||||
|
|
||||||
conn_info(tconn, "%s\n", pb);
|
|
||||||
}
|
|
||||||
|
|
||||||
enum sp_state {
|
|
||||||
OC_UNINITIALIZED,
|
|
||||||
OC_CONSISTENT,
|
|
||||||
OC_INCONSISTENT,
|
|
||||||
} oc_state;
|
|
||||||
|
|
||||||
static void common_state_part(enum sp_state *sps, int *sp, int nsp)
|
|
||||||
{
|
|
||||||
switch (*sps) {
|
|
||||||
case OC_UNINITIALIZED:
|
|
||||||
*sp = nsp;
|
|
||||||
*sps = OC_CONSISTENT;
|
|
||||||
break;
|
|
||||||
case OC_CONSISTENT:
|
|
||||||
if (*sp != nsp)
|
|
||||||
*sps = OC_INCONSISTENT;
|
|
||||||
break;
|
|
||||||
case OC_INCONSISTENT:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void conn_old_common_state(struct drbd_tconn *tconn, union drbd_state *pcs, union drbd_state *pmask)
|
|
||||||
{
|
|
||||||
union drbd_state css = {}; /* common state state */
|
|
||||||
union drbd_state os, cs = {}; /* old_state, common_state */
|
union drbd_state os, cs = {}; /* old_state, common_state */
|
||||||
union drbd_state mask = {};
|
|
||||||
enum sp_state sps; /* state part state */
|
|
||||||
int sp; /* state part */
|
|
||||||
struct drbd_conf *mdev;
|
struct drbd_conf *mdev;
|
||||||
int vnr;
|
int vnr, first_vol = 1;
|
||||||
|
|
||||||
idr_for_each_entry(&tconn->volumes, mdev, vnr) {
|
idr_for_each_entry(&tconn->volumes, mdev, vnr) {
|
||||||
os = mdev->state;
|
os = mdev->state;
|
||||||
|
|
||||||
sps = css.role;
|
if (first_vol) {
|
||||||
sp = cs.role;
|
cs = os;
|
||||||
common_state_part(&sps, &sp, os.role);
|
first_vol = 0;
|
||||||
css.role = sps;
|
continue;
|
||||||
cs.role = sp;
|
}
|
||||||
|
|
||||||
sps = css.peer;
|
if (cs.role != os.role)
|
||||||
sp = cs.peer;
|
flags &= ~CS_DC_ROLE;
|
||||||
common_state_part(&sps, &sp, os.peer);
|
|
||||||
css.peer = sps;
|
|
||||||
cs.peer = sp;
|
|
||||||
|
|
||||||
sps = css.conn;
|
if (cs.peer != os.peer)
|
||||||
sp = cs.conn;
|
flags &= ~CS_DC_PEER;
|
||||||
common_state_part(&sps, &sp, os.conn);
|
|
||||||
css.conn = sps;
|
|
||||||
cs.conn = sp;
|
|
||||||
|
|
||||||
sps = css.disk;
|
if (cs.conn != os.conn)
|
||||||
sp = cs.disk;
|
flags &= ~CS_DC_CONN;
|
||||||
common_state_part(&sps, &sp, os.disk);
|
|
||||||
css.disk = sps;
|
|
||||||
cs.disk = sp;
|
|
||||||
|
|
||||||
sps = css.pdsk;
|
if (cs.disk != os.disk)
|
||||||
sp = cs.pdsk;
|
flags &= ~CS_DC_DISK;
|
||||||
common_state_part(&sps, &sp, os.pdsk);
|
|
||||||
css.pdsk = sps;
|
if (cs.pdsk != os.pdsk)
|
||||||
cs.pdsk = sp;
|
flags &= ~CS_DC_PDSK;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (css.role == OC_CONSISTENT)
|
*pf |= CS_DC_MASK;
|
||||||
mask.role = R_MASK;
|
*pf &= flags;
|
||||||
if (css.peer == OC_CONSISTENT)
|
|
||||||
mask.peer = R_MASK;
|
|
||||||
if (css.conn == OC_CONSISTENT)
|
|
||||||
mask.conn = C_MASK;
|
|
||||||
if (css.disk == OC_CONSISTENT)
|
|
||||||
mask.disk = D_MASK;
|
|
||||||
if (css.pdsk == OC_CONSISTENT)
|
|
||||||
mask.pdsk = D_MASK;
|
|
||||||
|
|
||||||
*pcs = cs;
|
*pcs = cs;
|
||||||
*pmask = mask;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum drbd_state_rv
|
static enum drbd_state_rv
|
||||||
|
@ -1541,7 +1503,7 @@ _conn_rq_cond(struct drbd_tconn *tconn, union drbd_state mask, union drbd_state
|
||||||
rv = tconn->cstate != C_WF_REPORT_PARAMS ? SS_CW_NO_NEED : SS_UNKNOWN_ERROR;
|
rv = tconn->cstate != C_WF_REPORT_PARAMS ? SS_CW_NO_NEED : SS_UNKNOWN_ERROR;
|
||||||
|
|
||||||
if (rv == SS_UNKNOWN_ERROR)
|
if (rv == SS_UNKNOWN_ERROR)
|
||||||
rv = conn_is_valid_transition(tconn, mask, val, CS_NO_CSTATE_CHG);
|
rv = conn_is_valid_transition(tconn, mask, val, 0);
|
||||||
|
|
||||||
if (rv == SS_SUCCESS)
|
if (rv == SS_SUCCESS)
|
||||||
rv = SS_UNKNOWN_ERROR; /* cont waiting, otherwise fail. */
|
rv = SS_UNKNOWN_ERROR; /* cont waiting, otherwise fail. */
|
||||||
|
@ -1583,7 +1545,7 @@ _conn_request_state(struct drbd_tconn *tconn, union drbd_state mask, union drbd_
|
||||||
enum drbd_state_rv rv = SS_SUCCESS;
|
enum drbd_state_rv rv = SS_SUCCESS;
|
||||||
struct after_conn_state_chg_work *acscw;
|
struct after_conn_state_chg_work *acscw;
|
||||||
enum drbd_conns oc = tconn->cstate;
|
enum drbd_conns oc = tconn->cstate;
|
||||||
union drbd_state ms, os_val, os_mask;
|
union drbd_state ms, os;
|
||||||
|
|
||||||
rv = is_valid_conn_transition(oc, val.conn);
|
rv = is_valid_conn_transition(oc, val.conn);
|
||||||
if (rv < SS_SUCCESS)
|
if (rv < SS_SUCCESS)
|
||||||
|
@ -1600,19 +1562,14 @@ _conn_request_state(struct drbd_tconn *tconn, union drbd_state mask, union drbd_
|
||||||
goto abort;
|
goto abort;
|
||||||
}
|
}
|
||||||
|
|
||||||
conn_old_common_state(tconn, &os_val, &os_mask);
|
conn_old_common_state(tconn, &os, &flags);
|
||||||
if (os_mask.conn == C_MASK) {
|
|
||||||
oc = os_val.conn;
|
|
||||||
print_conn_state_change(tconn, oc, val.conn);
|
|
||||||
flags |= CS_NO_CSTATE_CHG;
|
|
||||||
}
|
|
||||||
|
|
||||||
ms = conn_set_state(tconn, mask, val, flags);
|
ms = conn_set_state(tconn, mask, val, flags);
|
||||||
ms.conn = val.conn;
|
ms.conn = val.conn;
|
||||||
|
conn_pr_state_change(tconn, os, ms, flags);
|
||||||
|
|
||||||
acscw = kmalloc(sizeof(*acscw), GFP_ATOMIC);
|
acscw = kmalloc(sizeof(*acscw), GFP_ATOMIC);
|
||||||
if (acscw) {
|
if (acscw) {
|
||||||
acscw->oc = oc;
|
acscw->oc = os.conn;
|
||||||
acscw->nms = ms;
|
acscw->nms = ms;
|
||||||
acscw->flags = flags;
|
acscw->flags = flags;
|
||||||
acscw->w.cb = w_after_conn_state_ch;
|
acscw->w.cb = w_after_conn_state_ch;
|
||||||
|
|
|
@ -57,13 +57,18 @@ struct drbd_tconn;
|
||||||
__ns.T2 = (S2); __ns.T3 = (S3); __ns; })
|
__ns.T2 = (S2); __ns.T3 = (S3); __ns; })
|
||||||
|
|
||||||
enum chg_state_flags {
|
enum chg_state_flags {
|
||||||
CS_HARD = 1,
|
CS_HARD = 1 << 0,
|
||||||
CS_VERBOSE = 2,
|
CS_VERBOSE = 1 << 1,
|
||||||
CS_WAIT_COMPLETE = 4,
|
CS_WAIT_COMPLETE = 1 << 2,
|
||||||
CS_SERIALIZE = 8,
|
CS_SERIALIZE = 1 << 3,
|
||||||
CS_ORDERED = CS_WAIT_COMPLETE + CS_SERIALIZE,
|
CS_ORDERED = CS_WAIT_COMPLETE + CS_SERIALIZE,
|
||||||
CS_NO_CSTATE_CHG = 16, /* Do not display changes in cstate. Internal to drbd_state.c */
|
CS_LOCAL_ONLY = 1 << 4, /* Do not consider a device pair wide state change */
|
||||||
CS_LOCAL_ONLY = 32, /* Do not consider a device pair wide state change */
|
CS_DC_ROLE = 1 << 5, /* DC = display as connection state change */
|
||||||
|
CS_DC_PEER = 1 << 6,
|
||||||
|
CS_DC_CONN = 1 << 7,
|
||||||
|
CS_DC_DISK = 1 << 8,
|
||||||
|
CS_DC_PDSK = 1 << 9,
|
||||||
|
CS_DC_MASK = CS_DC_ROLE + CS_DC_PEER + CS_DC_CONN + CS_DC_DISK + CS_DC_PDSK,
|
||||||
};
|
};
|
||||||
|
|
||||||
extern enum drbd_state_rv drbd_change_state(struct drbd_conf *mdev,
|
extern enum drbd_state_rv drbd_change_state(struct drbd_conf *mdev,
|
||||||
|
|
Загрузка…
Ссылка в новой задаче