Blame view

fs/dlm/requestqueue.c 4.84 KB
e7fd41792   David Teigland   [DLM] The core of...
1
2
3
  /******************************************************************************
  *******************************************************************************
  **
c36258b59   David Teigland   [DLM] block dlm_r...
4
  **  Copyright (C) 2005-2007 Red Hat, Inc.  All rights reserved.
e7fd41792   David Teigland   [DLM] The core of...
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
  **
  **  This copyrighted material is made available to anyone wishing to use,
  **  modify, copy, or redistribute it subject to the terms and conditions
  **  of the GNU General Public License v.2.
  **
  *******************************************************************************
  ******************************************************************************/
  
  #include "dlm_internal.h"
  #include "member.h"
  #include "lock.h"
  #include "dir.h"
  #include "config.h"
  #include "requestqueue.h"
  
  struct rq_entry {
  	struct list_head list;
6d40c4a70   David Teigland   dlm: improve erro...
22
  	uint32_t recover_seq;
e7fd41792   David Teigland   [DLM] The core of...
23
  	int nodeid;
8b0d8e03f   Al Viro   dlm: use proper C...
24
  	struct dlm_message request;
e7fd41792   David Teigland   [DLM] The core of...
25
26
27
28
29
30
31
32
  };
  
  /*
   * Requests received while the lockspace is in recovery get added to the
   * request queue and processed when recovery is complete.  This happens when
   * the lockspace is suspended on some nodes before it is on others, or the
   * lockspace is enabled on some while still suspended on others.
   */
8b0d8e03f   Al Viro   dlm: use proper C...
33
  void dlm_add_requestqueue(struct dlm_ls *ls, int nodeid, struct dlm_message *ms)
e7fd41792   David Teigland   [DLM] The core of...
34
35
  {
  	struct rq_entry *e;
8b0d8e03f   Al Viro   dlm: use proper C...
36
  	int length = ms->m_header.h_length - sizeof(struct dlm_message);
e7fd41792   David Teigland   [DLM] The core of...
37

573c24c4a   David Teigland   dlm: always use G...
38
  	e = kmalloc(sizeof(struct rq_entry) + length, GFP_NOFS);
e7fd41792   David Teigland   [DLM] The core of...
39
  	if (!e) {
c36258b59   David Teigland   [DLM] block dlm_r...
40
41
  		log_print("dlm_add_requestqueue: out of memory len %d", length);
  		return;
e7fd41792   David Teigland   [DLM] The core of...
42
  	}
6d40c4a70   David Teigland   dlm: improve erro...
43
  	e->recover_seq = ls->ls_recover_seq & 0xFFFFFFFF;
e7fd41792   David Teigland   [DLM] The core of...
44
  	e->nodeid = nodeid;
8b0d8e03f   Al Viro   dlm: use proper C...
45
  	memcpy(&e->request, ms, ms->m_header.h_length);
e7fd41792   David Teigland   [DLM] The core of...
46

901359256   David Teigland   [DLM] Update DLM ...
47
  	mutex_lock(&ls->ls_requestqueue_mutex);
c36258b59   David Teigland   [DLM] block dlm_r...
48
  	list_add_tail(&e->list, &ls->ls_requestqueue);
901359256   David Teigland   [DLM] Update DLM ...
49
  	mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
50
  }
c36258b59   David Teigland   [DLM] block dlm_r...
51
52
53
54
55
56
57
58
59
60
  /*
   * Called by dlm_recoverd to process normal messages saved while recovery was
   * happening.  Normal locking has been enabled before this is called.  dlm_recv
   * upon receiving a message, will wait for all saved messages to be drained
   * here before processing the message it got.  If a new dlm_ls_stop() arrives
   * while we're processing these saved messages, it may block trying to suspend
   * dlm_recv if dlm_recv is waiting for us in dlm_wait_requestqueue.  In that
   * case, we don't abort since locking_stopped is still 0.  If dlm_recv is not
   * waiting for us, then this processing may be aborted due to locking_stopped.
   */
e7fd41792   David Teigland   [DLM] The core of...
61
62
63
  int dlm_process_requestqueue(struct dlm_ls *ls)
  {
  	struct rq_entry *e;
4875647a0   David Teigland   dlm: fixes for no...
64
  	struct dlm_message *ms;
e7fd41792   David Teigland   [DLM] The core of...
65
  	int error = 0;
901359256   David Teigland   [DLM] Update DLM ...
66
  	mutex_lock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
67
68
69
  
  	for (;;) {
  		if (list_empty(&ls->ls_requestqueue)) {
901359256   David Teigland   [DLM] Update DLM ...
70
  			mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
71
72
73
74
  			error = 0;
  			break;
  		}
  		e = list_entry(ls->ls_requestqueue.next, struct rq_entry, list);
901359256   David Teigland   [DLM] Update DLM ...
75
  		mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
76

4875647a0   David Teigland   dlm: fixes for no...
77
78
79
80
81
82
83
  		ms = &e->request;
  
  		log_limit(ls, "dlm_process_requestqueue msg %d from %d "
  			  "lkid %x remid %x result %d seq %u",
  			  ms->m_type, ms->m_header.h_nodeid,
  			  ms->m_lkid, ms->m_remid, ms->m_result,
  			  e->recover_seq);
6d40c4a70   David Teigland   dlm: improve erro...
84
  		dlm_receive_message_saved(ls, &e->request, e->recover_seq);
e7fd41792   David Teigland   [DLM] The core of...
85

901359256   David Teigland   [DLM] Update DLM ...
86
  		mutex_lock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
87
88
89
90
91
  		list_del(&e->list);
  		kfree(e);
  
  		if (dlm_locking_stopped(ls)) {
  			log_debug(ls, "process_requestqueue abort running");
901359256   David Teigland   [DLM] Update DLM ...
92
  			mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
93
94
95
96
97
98
99
100
101
102
103
  			error = -EINTR;
  			break;
  		}
  		schedule();
  	}
  
  	return error;
  }
  
  /*
   * After recovery is done, locking is resumed and dlm_recoverd takes all the
c36258b59   David Teigland   [DLM] block dlm_r...
104
105
106
107
108
109
   * saved requests and processes them as they would have been by dlm_recv.  At
   * the same time, dlm_recv will start receiving new requests from remote nodes.
   * We want to delay dlm_recv processing new requests until dlm_recoverd has
   * finished processing the old saved requests.  We don't check for locking
   * stopped here because dlm_ls_stop won't stop locking until it's suspended us
   * (dlm_recv).
e7fd41792   David Teigland   [DLM] The core of...
110
111
112
113
114
   */
  
  void dlm_wait_requestqueue(struct dlm_ls *ls)
  {
  	for (;;) {
901359256   David Teigland   [DLM] Update DLM ...
115
  		mutex_lock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
116
117
  		if (list_empty(&ls->ls_requestqueue))
  			break;
901359256   David Teigland   [DLM] Update DLM ...
118
  		mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
119
120
  		schedule();
  	}
901359256   David Teigland   [DLM] Update DLM ...
121
  	mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
122
123
124
125
126
  }
  
  static int purge_request(struct dlm_ls *ls, struct dlm_message *ms, int nodeid)
  {
  	uint32_t type = ms->m_type;
2896ee37c   David Teigland   [DLM] fix add_req...
127
128
129
  	/* the ls is being cleaned up and freed by release_lockspace */
  	if (!ls->ls_count)
  		return 1;
e7fd41792   David Teigland   [DLM] The core of...
130
131
132
133
134
135
136
137
138
139
140
141
142
  	if (dlm_is_removed(ls, nodeid))
  		return 1;
  
  	/* directory operations are always purged because the directory is
  	   always rebuilt during recovery and the lookups resent */
  
  	if (type == DLM_MSG_REMOVE ||
  	    type == DLM_MSG_LOOKUP ||
  	    type == DLM_MSG_LOOKUP_REPLY)
  		return 1;
  
  	if (!dlm_no_directory(ls))
  		return 0;
4875647a0   David Teigland   dlm: fixes for no...
143
  	return 1;
e7fd41792   David Teigland   [DLM] The core of...
144
145
146
147
148
149
  }
  
  void dlm_purge_requestqueue(struct dlm_ls *ls)
  {
  	struct dlm_message *ms;
  	struct rq_entry *e, *safe;
901359256   David Teigland   [DLM] Update DLM ...
150
  	mutex_lock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
151
  	list_for_each_entry_safe(e, safe, &ls->ls_requestqueue, list) {
8b0d8e03f   Al Viro   dlm: use proper C...
152
  		ms =  &e->request;
e7fd41792   David Teigland   [DLM] The core of...
153
154
155
156
157
158
  
  		if (purge_request(ls, ms, e->nodeid)) {
  			list_del(&e->list);
  			kfree(e);
  		}
  	}
901359256   David Teigland   [DLM] Update DLM ...
159
  	mutex_unlock(&ls->ls_requestqueue_mutex);
e7fd41792   David Teigland   [DLM] The core of...
160
  }