afr: thin-arbiter lock release fixes

- pass fop state instead of afr local to
afr_ta_dom_lock_check_and_release()

- avoid afr_lock_release_synctask() being called simultaneosuly from
notify code path and transaction (post-op) code path due to races.

- Check if the post-op on TA is valid based on event_gen checks.

- Invalidate in-memory information when we get TA child down.

Note: Thi patch addresses some pending review comments of commit
053b1309dc8fbc05fcde5223e734da9f694cf5cc
(https://review.gluster.org/#/c/glusterfs/+/20095/)

fixes: bz#1709130
Change-Id: I2ccd7e1b53362f9f3fed8680aecb23b5011eb18c
Signed-off-by: Ravishankar N <ravishankar@redhat.com>
(cherry picked from commit 9ab2747da78061882f6734df4b265bce11adaef1)
This commit is contained in:
Ravishankar N 2019-04-09 09:44:33 +05:30
parent a1fa0379b7
commit 9f225fa2c4
3 changed files with 93 additions and 47 deletions

View File

@ -5366,6 +5366,11 @@ afr_handle_inodelk_contention(xlator_t *this, struct gf_upcall *upcall)
}
LOCK(&priv->lock);
{
if (priv->release_ta_notify_dom_lock == _gf_true) {
/* Ignore multiple release requests from shds.*/
UNLOCK(&priv->lock);
return;
}
priv->release_ta_notify_dom_lock = _gf_true;
inmem_count = priv->ta_in_mem_txn_count;
onwire_count = priv->ta_on_wire_txn_count;
@ -5373,7 +5378,7 @@ afr_handle_inodelk_contention(xlator_t *this, struct gf_upcall *upcall)
UNLOCK(&priv->lock);
if (inmem_count || onwire_count)
/* lock release will happen in txn code path after
* inflight or on-wire txns are over.*/
* in-memory or on-wire txns are over.*/
return;
afr_ta_lock_release_synctask(this);
@ -5531,6 +5536,7 @@ afr_notify(xlator_t *this, int32_t event, void *data, void *data2)
if (priv->thin_arbiter_count &&
(idx == AFR_CHILD_THIN_ARBITER)) {
priv->ta_child_up = 1;
priv->ta_event_gen++;
break;
}
__afr_handle_child_up_event(this, child_xlator, idx,
@ -5542,6 +5548,8 @@ afr_notify(xlator_t *this, int32_t event, void *data, void *data2)
if (priv->thin_arbiter_count &&
(idx == AFR_CHILD_THIN_ARBITER)) {
priv->ta_child_up = 0;
priv->ta_event_gen++;
afr_ta_locked_priv_invalidate(priv);
break;
}
__afr_handle_child_down_event(this, child_xlator, idx,
@ -5702,6 +5710,8 @@ afr_local_init(afr_local_t *local, afr_private_t *priv, int32_t *op_errno)
local->ta_child_up = priv->ta_child_up;
local->ta_failed_subvol = AFR_CHILD_UNKNOWN;
local->read_txn_query_child = AFR_CHILD_UNKNOWN;
local->ta_event_gen = priv->ta_event_gen;
local->fop_state = TA_SUCCESS;
}
local->is_new_entry = _gf_false;

View File

@ -74,6 +74,14 @@ afr_changelog_post_op_done(call_frame_t *frame, xlator_t *this);
static void
afr_changelog_post_op_fail(call_frame_t *frame, xlator_t *this, int op_errno);
void
afr_ta_locked_priv_invalidate(afr_private_t *priv)
{
priv->ta_bad_child_index = AFR_CHILD_UNKNOWN;
priv->release_ta_notify_dom_lock = _gf_false;
priv->ta_notify_dom_lock_offset = 0;
}
static void
afr_ta_process_waitq(xlator_t *this)
{
@ -127,17 +135,16 @@ afr_release_notify_lock_for_ta(void *opaque)
flock.l_len = 1;
ret = syncop_inodelk(priv->children[THIN_ARBITER_BRICK_INDEX],
AFR_TA_DOM_NOTIFY, &loc, F_SETLK, &flock, NULL, NULL);
if (!ret) {
LOCK(&priv->lock);
priv->ta_bad_child_index = AFR_CHILD_UNKNOWN;
priv->release_ta_notify_dom_lock = _gf_false;
priv->ta_notify_dom_lock_offset = 0;
UNLOCK(&priv->lock);
} else {
if (ret) {
gf_msg(this->name, GF_LOG_ERROR, -ret, AFR_MSG_THIN_ARB,
"Failed to unlock AFR_TA_DOM_NOTIFY lock.");
}
LOCK(&priv->lock);
{
afr_ta_locked_priv_invalidate(priv);
}
UNLOCK(&priv->lock);
out:
loc_wipe(&loc);
return ret;
@ -665,7 +672,7 @@ afr_set_pending_dict(afr_private_t *priv, dict_t *xattr, int **pending)
}
static void
afr_ta_dom_lock_check_and_release(afr_local_t *local, xlator_t *this)
afr_ta_dom_lock_check_and_release(afr_ta_fop_state_t fop_state, xlator_t *this)
{
afr_private_t *priv = this->private;
unsigned int inmem_count = 0;
@ -675,17 +682,25 @@ afr_ta_dom_lock_check_and_release(afr_local_t *local, xlator_t *this)
LOCK(&priv->lock);
{
/*Once we get notify lock release upcall notification,
if two fop states are non empty/non zero, we will
not release lock.
1 - If anything in memory txn
2 - If anything in onwire or onwireq
if any of the fop state counters are non-zero, we will
not release the lock.
*/
if (local->fop_state == TA_INFO_IN_MEMORY_SUCCESS) {
inmem_count = --priv->ta_in_mem_txn_count;
} else {
inmem_count = priv->ta_in_mem_txn_count;
}
onwire_count = priv->ta_on_wire_txn_count;
inmem_count = priv->ta_in_mem_txn_count;
switch (fop_state) {
case TA_GET_INFO_FROM_TA_FILE:
onwire_count = --priv->ta_on_wire_txn_count;
break;
case TA_INFO_IN_MEMORY_SUCCESS:
case TA_INFO_IN_MEMORY_FAILED:
inmem_count = --priv->ta_in_mem_txn_count;
break;
case TA_WAIT_FOR_NOTIFY_LOCK_REL:
GF_ASSERT(0);
break;
case TA_SUCCESS:
break;
}
release = priv->release_ta_notify_dom_lock;
}
UNLOCK(&priv->lock);
@ -697,7 +712,7 @@ afr_ta_dom_lock_check_and_release(afr_local_t *local, xlator_t *this)
}
static void
afr_ta_process_onwireq(afr_local_t *local, xlator_t *this)
afr_ta_process_onwireq(afr_ta_fop_state_t fop_state, xlator_t *this)
{
afr_private_t *priv = this->private;
afr_local_t *entry = NULL;
@ -710,15 +725,6 @@ afr_ta_process_onwireq(afr_local_t *local, xlator_t *this)
LOCK(&priv->lock);
{
if (--priv->ta_on_wire_txn_count == 0) {
UNLOCK(&priv->lock);
/*Only one write fop came and after taking notify
*lock and before doing xattrop, it has received
*lock contention upcall, so this is the only place
*to find this out and release the lock*/
afr_ta_dom_lock_check_and_release(local, this);
return;
}
bad_child = priv->ta_bad_child_index;
if (bad_child == AFR_CHILD_UNKNOWN) {
/*The previous on-wire ta_post_op was a failure. Just dequeue
@ -739,9 +745,6 @@ afr_ta_process_onwireq(afr_local_t *local, xlator_t *this)
while (!list_empty(&onwireq)) {
entry = list_entry(onwireq.next, afr_local_t, ta_onwireq);
list_del_init(&entry->ta_onwireq);
LOCK(&priv->lock);
--priv->ta_on_wire_txn_count;
UNLOCK(&priv->lock);
if (entry->ta_failed_subvol == bad_child) {
afr_post_op_handle_success(entry->transaction.frame, this);
} else {
@ -764,7 +767,7 @@ afr_changelog_post_op_done(call_frame_t *frame, xlator_t *this)
if (priv->thin_arbiter_count) {
/*fop should not come here with TA_WAIT_FOR_NOTIFY_LOCK_REL state */
afr_ta_dom_lock_check_and_release(frame->local, this);
afr_ta_dom_lock_check_and_release(local->fop_state, this);
}
/* Fail the FOP if post-op did not succeed on quorum no. of bricks. */
@ -1172,12 +1175,23 @@ afr_ta_post_op_done(int ret, call_frame_t *frame, void *opaque)
{
xlator_t *this = NULL;
afr_local_t *local = NULL;
call_frame_t *txn_frame = NULL;
afr_ta_fop_state_t fop_state;
local = (afr_local_t *)opaque;
fop_state = local->fop_state;
txn_frame = local->transaction.frame;
this = frame->this;
if (ret == 0) {
/*Mark pending xattrs on the up data brick.*/
afr_post_op_handle_success(txn_frame, this);
} else {
afr_post_op_handle_failure(txn_frame, this, -ret);
}
STACK_DESTROY(frame->root);
afr_ta_process_onwireq(local, this);
afr_ta_process_onwireq(fop_state, this);
return 0;
}
@ -1210,13 +1224,25 @@ out:
return changelog;
}
static void
afr_ta_locked_xattrop_validate(afr_private_t *priv, afr_local_t *local,
gf_boolean_t *valid)
{
if (priv->ta_event_gen > local->ta_event_gen) {
/* We can't trust the ta's response anymore.*/
afr_ta_locked_priv_invalidate(priv);
*valid = _gf_false;
return;
}
return;
}
static int
afr_ta_post_op_do(void *opaque)
{
afr_local_t *local = NULL;
afr_private_t *priv = NULL;
xlator_t *this = NULL;
call_frame_t *txn_frame = NULL;
dict_t *xattr = NULL;
unsigned char *pending = NULL;
int **changelog = NULL;
@ -1227,10 +1253,10 @@ afr_ta_post_op_do(void *opaque)
};
int i = 0;
int ret = 0;
gf_boolean_t valid = _gf_true;
local = (afr_local_t *)opaque;
txn_frame = local->transaction.frame;
this = txn_frame->this;
this = local->transaction.frame->this;
priv = this->private;
ret = afr_fill_ta_loc(this, &loc);
@ -1268,22 +1294,31 @@ afr_ta_post_op_do(void *opaque)
ret = syncop_xattrop(priv->children[THIN_ARBITER_BRICK_INDEX], &loc,
GF_XATTROP_ADD_ARRAY, xattr, NULL, NULL, NULL);
if (ret) {
gf_msg(this->name, GF_LOG_ERROR, -ret, AFR_MSG_THIN_ARB,
"Post-op on thin-arbiter id file %s failed for gfid %s.",
priv->pending_key[THIN_ARBITER_BRICK_INDEX],
uuid_utoa(local->inode->gfid));
}
LOCK(&priv->lock);
{
if (ret == 0) {
priv->ta_bad_child_index = failed_subvol;
} else if (ret == -EINVAL) {
priv->ta_bad_child_index = success_subvol;
ret = -EIO; /* TA failed the fop. Return EIO to application. */
}
afr_ta_locked_xattrop_validate(priv, local, &valid);
}
UNLOCK(&priv->lock);
if (ret) {
gf_msg(this->name, GF_LOG_ERROR, -ret, AFR_MSG_THIN_ARB,
"Post-op on thin-arbiter id file %s failed for gfid %s.",
if (valid == _gf_false) {
gf_msg(this->name, GF_LOG_ERROR, EIO, AFR_MSG_THIN_ARB,
"Post-op on thin-arbiter id file %s for gfid %s invalidated due "
"to event-gen mismatch.",
priv->pending_key[THIN_ARBITER_BRICK_INDEX],
uuid_utoa(local->inode->gfid));
if (ret == -EINVAL)
ret = -EIO; /* TA failed the fop. Return EIO to application. */
ret = -EIO;
}
afr_ta_post_op_unlock(this, &loc);
@ -1296,12 +1331,6 @@ out:
loc_wipe(&loc);
if (ret == 0) {
/*Mark pending xattrs on the up data brick.*/
afr_post_op_handle_success(local->transaction.frame, this);
} else {
afr_post_op_handle_failure(local->transaction.frame, this, -ret);
}
return ret;
}
@ -1360,6 +1389,7 @@ afr_ta_set_fop_state(afr_private_t *priv, afr_local_t *local,
/* Post-op on TA not needed as the fop succeeded only on the
* in-memory bad data brick and not the good one. Fail the fop.*/
local->fop_state = TA_INFO_IN_MEMORY_FAILED;
priv->ta_in_mem_txn_count++;
}
}
UNLOCK(&priv->lock);

View File

@ -126,6 +126,7 @@ typedef enum {
*on BAD brick - Success*/
TA_INFO_IN_MEMORY_FAILED, /*Bad brick info is in memory and fop failed
*on GOOD brick - Failed*/
TA_SUCCESS, /*FOP succeeded on both data bricks.*/
} afr_ta_fop_state_t;
struct afr_nfsd {
@ -148,6 +149,7 @@ typedef struct _afr_private {
uuid_t ta_gfid;
unsigned char ta_child_up;
int ta_bad_child_index;
int ta_event_gen;
off_t ta_notify_dom_lock_offset;
gf_boolean_t release_ta_notify_dom_lock;
unsigned int ta_in_mem_txn_count;
@ -887,6 +889,7 @@ typedef struct _afr_local {
struct list_head ta_onwireq;
afr_ta_fop_state_t fop_state;
int ta_failed_subvol;
int ta_event_gen;
gf_boolean_t is_new_entry;
} afr_local_t;
@ -1326,6 +1329,9 @@ afr_ta_has_quorum(afr_private_t *priv, afr_local_t *local);
void
afr_ta_lock_release_synctask(xlator_t *this);
void
afr_ta_locked_priv_invalidate(afr_private_t *priv);
gf_boolean_t
afr_lookup_has_quorum(call_frame_t *frame, xlator_t *this,
unsigned char *subvols);