aboutsummaryrefslogtreecommitdiffstats
path: root/fs/lockd
diff options
context:
space:
mode:
Diffstat (limited to 'fs/lockd')
-rw-r--r--fs/lockd/host.c86
1 files changed, 56 insertions, 30 deletions
diff --git a/fs/lockd/host.c b/fs/lockd/host.c
index be8f19d53183..3c4dc33c1bea 100644
--- a/fs/lockd/host.c
+++ b/fs/lockd/host.c
@@ -38,6 +38,17 @@ static struct nsm_handle *nsm_find(const struct sockaddr *sap,
38 const size_t hostname_len, 38 const size_t hostname_len,
39 const int create); 39 const int create);
40 40
41struct nlm_lookup_host_info {
42 const int server; /* search for server|client */
43 const struct sockaddr_in *sin; /* address to search for */
44 const unsigned short protocol; /* transport to search for*/
45 const u32 version; /* NLM version to search for */
46 const char *hostname; /* remote's hostname */
47 const size_t hostname_len; /* it's length */
48 const struct sockaddr_in *src_sin; /* our address (optional) */
49 const size_t src_len; /* it's length */
50};
51
41/* 52/*
42 * Hash function must work well on big- and little-endian platforms 53 * Hash function must work well on big- and little-endian platforms
43 */ 54 */
@@ -121,23 +132,13 @@ static void nlm_display_address(const struct sockaddr *sap,
121/* 132/*
122 * Common host lookup routine for server & client 133 * Common host lookup routine for server & client
123 */ 134 */
124static struct nlm_host *nlm_lookup_host(int server, 135static struct nlm_host *nlm_lookup_host(struct nlm_lookup_host_info *ni)
125 const struct sockaddr_in *sin,
126 int proto, u32 version,
127 const char *hostname,
128 unsigned int hostname_len,
129 const struct sockaddr_in *ssin)
130{ 136{
131 struct hlist_head *chain; 137 struct hlist_head *chain;
132 struct hlist_node *pos; 138 struct hlist_node *pos;
133 struct nlm_host *host; 139 struct nlm_host *host;
134 struct nsm_handle *nsm = NULL; 140 struct nsm_handle *nsm = NULL;
135 141
136 dprintk("lockd: nlm_lookup_host(proto=%d, vers=%u,"
137 " my role is %s, hostname=%.*s)\n",
138 proto, version, server ? "server" : "client",
139 hostname_len, hostname ? hostname : "<none>");
140
141 mutex_lock(&nlm_host_mutex); 142 mutex_lock(&nlm_host_mutex);
142 143
143 if (time_after_eq(jiffies, next_gc)) 144 if (time_after_eq(jiffies, next_gc))
@@ -150,22 +151,23 @@ static struct nlm_host *nlm_lookup_host(int server,
150 * different NLM rpc_clients into one single nlm_host object. 151 * different NLM rpc_clients into one single nlm_host object.
151 * This would allow us to have one nlm_host per address. 152 * This would allow us to have one nlm_host per address.
152 */ 153 */
153 chain = &nlm_hosts[nlm_hash_address((struct sockaddr *)sin)]; 154 chain = &nlm_hosts[nlm_hash_address((struct sockaddr *)ni->sin)];
154 hlist_for_each_entry(host, pos, chain, h_hash) { 155 hlist_for_each_entry(host, pos, chain, h_hash) {
155 if (!nlm_cmp_addr(nlm_addr(host), (struct sockaddr *)sin)) 156 if (!nlm_cmp_addr(nlm_addr(host), (struct sockaddr *)ni->sin))
156 continue; 157 continue;
157 158
158 /* See if we have an NSM handle for this client */ 159 /* See if we have an NSM handle for this client */
159 if (!nsm) 160 if (!nsm)
160 nsm = host->h_nsmhandle; 161 nsm = host->h_nsmhandle;
161 162
162 if (host->h_proto != proto) 163 if (host->h_proto != ni->protocol)
163 continue; 164 continue;
164 if (host->h_version != version) 165 if (host->h_version != ni->version)
165 continue; 166 continue;
166 if (host->h_server != server) 167 if (host->h_server != ni->server)
167 continue; 168 continue;
168 if (!nlm_cmp_addr(nlm_srcaddr(host), (struct sockaddr *)ssin)) 169 if (!nlm_cmp_addr(nlm_srcaddr(host),
170 (struct sockaddr *)ni->src_sin))
169 continue; 171 continue;
170 172
171 /* Move to head of hash chain. */ 173 /* Move to head of hash chain. */
@@ -186,8 +188,9 @@ static struct nlm_host *nlm_lookup_host(int server,
186 atomic_inc(&nsm->sm_count); 188 atomic_inc(&nsm->sm_count);
187 else { 189 else {
188 host = NULL; 190 host = NULL;
189 nsm = nsm_find((struct sockaddr *)sin, sizeof(*sin), 191 nsm = nsm_find((struct sockaddr *)ni->sin,
190 hostname, hostname_len, 1); 192 sizeof(struct sockaddr_in),
193 ni->hostname, ni->hostname_len, 1);
191 if (!nsm) { 194 if (!nsm) {
192 dprintk("lockd: nlm_lookup_host failed; " 195 dprintk("lockd: nlm_lookup_host failed; "
193 "no nsm handle\n"); 196 "no nsm handle\n");
@@ -202,12 +205,12 @@ static struct nlm_host *nlm_lookup_host(int server,
202 goto out; 205 goto out;
203 } 206 }
204 host->h_name = nsm->sm_name; 207 host->h_name = nsm->sm_name;
205 memcpy(nlm_addr(host), sin, sizeof(*sin)); 208 memcpy(nlm_addr(host), ni->sin, sizeof(struct sockaddr_in));
206 host->h_addrlen = sizeof(*sin); 209 host->h_addrlen = sizeof(struct sockaddr_in);
207 nlm_clear_port(nlm_addr(host)); 210 nlm_clear_port(nlm_addr(host));
208 memcpy(nlm_srcaddr(host), ssin, sizeof(*ssin)); 211 memcpy(nlm_srcaddr(host), ni->src_sin, sizeof(struct sockaddr_in));
209 host->h_version = version; 212 host->h_version = ni->version;
210 host->h_proto = proto; 213 host->h_proto = ni->protocol;
211 host->h_rpcclnt = NULL; 214 host->h_rpcclnt = NULL;
212 mutex_init(&host->h_mutex); 215 mutex_init(&host->h_mutex);
213 host->h_nextrebind = jiffies + NLM_HOST_REBIND; 216 host->h_nextrebind = jiffies + NLM_HOST_REBIND;
@@ -218,7 +221,7 @@ static struct nlm_host *nlm_lookup_host(int server,
218 host->h_state = 0; /* pseudo NSM state */ 221 host->h_state = 0; /* pseudo NSM state */
219 host->h_nsmstate = 0; /* real NSM state */ 222 host->h_nsmstate = 0; /* real NSM state */
220 host->h_nsmhandle = nsm; 223 host->h_nsmhandle = nsm;
221 host->h_server = server; 224 host->h_server = ni->server;
222 hlist_add_head(&host->h_hash, chain); 225 hlist_add_head(&host->h_hash, chain);
223 INIT_LIST_HEAD(&host->h_lockowners); 226 INIT_LIST_HEAD(&host->h_lockowners);
224 spin_lock_init(&host->h_lock); 227 spin_lock_init(&host->h_lock);
@@ -273,9 +276,21 @@ struct nlm_host *nlmclnt_lookup_host(const struct sockaddr_in *sin,
273 const struct sockaddr_in source = { 276 const struct sockaddr_in source = {
274 .sin_family = AF_UNSPEC, 277 .sin_family = AF_UNSPEC,
275 }; 278 };
279 struct nlm_lookup_host_info ni = {
280 .server = 0,
281 .sin = sin,
282 .protocol = proto,
283 .version = version,
284 .hostname = hostname,
285 .hostname_len = hostname_len,
286 .src_sin = &source,
287 };
276 288
277 return nlm_lookup_host(0, sin, proto, version, 289 dprintk("lockd: %s(host='%s', vers=%u, proto=%s)\n", __func__,
278 hostname, hostname_len, &source); 290 (hostname ? hostname : "<none>"), version,
291 (proto == IPPROTO_UDP ? "udp" : "tcp"));
292
293 return nlm_lookup_host(&ni);
279} 294}
280 295
281/* 296/*
@@ -289,10 +304,21 @@ nlmsvc_lookup_host(struct svc_rqst *rqstp,
289 .sin_family = AF_INET, 304 .sin_family = AF_INET,
290 .sin_addr = rqstp->rq_daddr.addr, 305 .sin_addr = rqstp->rq_daddr.addr,
291 }; 306 };
307 struct nlm_lookup_host_info ni = {
308 .server = 1,
309 .sin = svc_addr_in(rqstp),
310 .protocol = rqstp->rq_prot,
311 .version = rqstp->rq_vers,
312 .hostname = hostname,
313 .hostname_len = hostname_len,
314 .src_sin = &source,
315 };
316
317 dprintk("lockd: %s(host='%*s', vers=%u, proto=%s)\n", __func__,
318 (int)hostname_len, hostname, rqstp->rq_vers,
319 (rqstp->rq_prot == IPPROTO_UDP ? "udp" : "tcp"));
292 320
293 return nlm_lookup_host(1, svc_addr_in(rqstp), 321 return nlm_lookup_host(&ni);
294 rqstp->rq_prot, rqstp->rq_vers,
295 hostname, hostname_len, &source);
296} 322}
297 323
298/* 324/*