mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-01 10:42:11 +00:00
dlm: fix add_scan and del_scan usage
Remove a few calls to add_scan() and del_scan() in cases where the rsb is a dir record, so the rsb should never be placed on the scan list at all. Add WARN_ON to catch cases where this is done. Signed-off-by: David Teigland <teigland@redhat.com>
This commit is contained in:
parent
4f5957a980
commit
c217adfc8c
@ -420,6 +420,9 @@ static void del_scan(struct dlm_ls *ls, struct dlm_rsb *r)
|
|||||||
{
|
{
|
||||||
struct dlm_rsb *first;
|
struct dlm_rsb *first;
|
||||||
|
|
||||||
|
/* active rsbs should never be on the scan list */
|
||||||
|
WARN_ON(!rsb_flag(r, RSB_INACTIVE));
|
||||||
|
|
||||||
spin_lock_bh(&ls->ls_scan_lock);
|
spin_lock_bh(&ls->ls_scan_lock);
|
||||||
r->res_toss_time = 0;
|
r->res_toss_time = 0;
|
||||||
|
|
||||||
@ -457,17 +460,16 @@ static void add_scan(struct dlm_ls *ls, struct dlm_rsb *r)
|
|||||||
int our_nodeid = dlm_our_nodeid();
|
int our_nodeid = dlm_our_nodeid();
|
||||||
struct dlm_rsb *first;
|
struct dlm_rsb *first;
|
||||||
|
|
||||||
/* If we're the directory record for this rsb, and
|
/* A dir record for a remote master rsb should never be on the scan list. */
|
||||||
* we're not the master of it, then we need to wait
|
WARN_ON(!dlm_no_directory(ls) &&
|
||||||
* for the master node to send us a dir remove for
|
(r->res_master_nodeid != our_nodeid) &&
|
||||||
* before removing the dir record.
|
(dlm_dir_nodeid(r) == our_nodeid));
|
||||||
*/
|
|
||||||
if (!dlm_no_directory(ls) &&
|
/* An active rsb should never be on the scan list. */
|
||||||
(r->res_master_nodeid != our_nodeid) &&
|
WARN_ON(!rsb_flag(r, RSB_INACTIVE));
|
||||||
(dlm_dir_nodeid(r) == our_nodeid)) {
|
|
||||||
del_scan(ls, r);
|
/* An rsb should not already be on the scan list. */
|
||||||
return;
|
WARN_ON(!list_empty(&r->res_scan_list));
|
||||||
}
|
|
||||||
|
|
||||||
spin_lock_bh(&ls->ls_scan_lock);
|
spin_lock_bh(&ls->ls_scan_lock);
|
||||||
/* set the new rsb absolute expire time in the rsb */
|
/* set the new rsb absolute expire time in the rsb */
|
||||||
@ -479,12 +481,6 @@ static void add_scan(struct dlm_ls *ls, struct dlm_rsb *r)
|
|||||||
list_add_tail(&r->res_scan_list, &ls->ls_scan_list);
|
list_add_tail(&r->res_scan_list, &ls->ls_scan_list);
|
||||||
enable_scan_timer(ls, r->res_toss_time);
|
enable_scan_timer(ls, r->res_toss_time);
|
||||||
} else {
|
} else {
|
||||||
/* check if the rsb was already queued, if so delete
|
|
||||||
* it from the toss queue
|
|
||||||
*/
|
|
||||||
if (!list_empty(&r->res_scan_list))
|
|
||||||
list_del(&r->res_scan_list);
|
|
||||||
|
|
||||||
/* try to get the maybe new first element and then add
|
/* try to get the maybe new first element and then add
|
||||||
* to this rsb with the oldest expire time to the end
|
* to this rsb with the oldest expire time to the end
|
||||||
* of the queue. If the list was empty before this
|
* of the queue. If the list was empty before this
|
||||||
@ -807,10 +803,12 @@ static int find_rsb_dir(struct dlm_ls *ls, const void *name, int len,
|
|||||||
r->res_first_lkid = 0;
|
r->res_first_lkid = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* A dir record will not be on the scan list. */
|
||||||
|
if (r->res_dir_nodeid != our_nodeid)
|
||||||
|
del_scan(ls, r);
|
||||||
list_move(&r->res_slow_list, &ls->ls_slow_active);
|
list_move(&r->res_slow_list, &ls->ls_slow_active);
|
||||||
rsb_clear_flag(r, RSB_INACTIVE);
|
rsb_clear_flag(r, RSB_INACTIVE);
|
||||||
kref_init(&r->res_ref);
|
kref_init(&r->res_ref); /* ref is now used in active state */
|
||||||
del_scan(ls, r);
|
|
||||||
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
||||||
|
|
||||||
goto out;
|
goto out;
|
||||||
@ -1272,7 +1270,10 @@ int dlm_master_lookup(struct dlm_ls *ls, int from_nodeid, const char *name,
|
|||||||
__dlm_master_lookup(ls, r, our_nodeid, from_nodeid, true, flags,
|
__dlm_master_lookup(ls, r, our_nodeid, from_nodeid, true, flags,
|
||||||
r_nodeid, result);
|
r_nodeid, result);
|
||||||
|
|
||||||
add_scan(ls, r);
|
/* A dir record rsb should never be on scan list. */
|
||||||
|
/* Try to fix this with del_scan? */
|
||||||
|
WARN_ON(!list_empty(&r->res_scan_list));
|
||||||
|
|
||||||
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -1305,7 +1306,6 @@ int dlm_master_lookup(struct dlm_ls *ls, int from_nodeid, const char *name,
|
|||||||
}
|
}
|
||||||
|
|
||||||
list_add(&r->res_slow_list, &ls->ls_slow_inactive);
|
list_add(&r->res_slow_list, &ls->ls_slow_inactive);
|
||||||
add_scan(ls, r);
|
|
||||||
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
write_unlock_bh(&ls->ls_rsbtbl_lock);
|
||||||
|
|
||||||
if (result)
|
if (result)
|
||||||
@ -1346,11 +1346,24 @@ static void deactivate_rsb(struct kref *kref)
|
|||||||
{
|
{
|
||||||
struct dlm_rsb *r = container_of(kref, struct dlm_rsb, res_ref);
|
struct dlm_rsb *r = container_of(kref, struct dlm_rsb, res_ref);
|
||||||
struct dlm_ls *ls = r->res_ls;
|
struct dlm_ls *ls = r->res_ls;
|
||||||
|
int our_nodeid = dlm_our_nodeid();
|
||||||
|
|
||||||
DLM_ASSERT(list_empty(&r->res_root_list), dlm_print_rsb(r););
|
DLM_ASSERT(list_empty(&r->res_root_list), dlm_print_rsb(r););
|
||||||
rsb_set_flag(r, RSB_INACTIVE);
|
rsb_set_flag(r, RSB_INACTIVE);
|
||||||
list_move(&r->res_slow_list, &ls->ls_slow_inactive);
|
list_move(&r->res_slow_list, &ls->ls_slow_inactive);
|
||||||
add_scan(ls, r);
|
|
||||||
|
/*
|
||||||
|
* When the rsb becomes unused:
|
||||||
|
* - If it's not a dir record for a remote master rsb,
|
||||||
|
* then it is put on the scan list to be freed.
|
||||||
|
* - If it's a dir record for a remote master rsb,
|
||||||
|
* then it is kept in the inactive state until
|
||||||
|
* receive_remove() from the master node.
|
||||||
|
*/
|
||||||
|
if (!dlm_no_directory(ls) &&
|
||||||
|
(r->res_master_nodeid != our_nodeid) &&
|
||||||
|
(dlm_dir_nodeid(r) != our_nodeid))
|
||||||
|
add_scan(ls, r);
|
||||||
|
|
||||||
if (r->res_lvbptr) {
|
if (r->res_lvbptr) {
|
||||||
dlm_free_lvb(r->res_lvbptr);
|
dlm_free_lvb(r->res_lvbptr);
|
||||||
|
Loading…
Reference in New Issue
Block a user