aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorTrond Myklebust <Trond.Myklebust@netapp.com>2009-03-28 16:50:58 -0400
committerTrond Myklebust <Trond.Myklebust@netapp.com>2009-03-28 16:50:58 -0400
commitd188262d60e879800d3a0a768ecbb216d6ef0d40 (patch)
tree4416aba693ff03551987addb3de626920b212f5d /fs
parentf738f5170367b367e38b2d75a413e7b3c52d46a5 (diff)
parent9f4c899c0d90e1b51b6864834f3877b47c161a0e (diff)
Merge commit '9f4c899c0d90e1b51b6864834f3877b47c161a0e' into devel
Diffstat (limited to 'fs')
-rw-r--r--fs/nfs/client.c68
1 files changed, 39 insertions, 29 deletions
diff --git a/fs/nfs/client.c b/fs/nfs/client.c
index 06654b831d19..574158ae2398 100644
--- a/fs/nfs/client.c
+++ b/fs/nfs/client.c
@@ -255,6 +255,32 @@ static int nfs_sockaddr_match_ipaddr(const struct sockaddr *sa1,
255 } 255 }
256 return 0; 256 return 0;
257} 257}
258
259/*
260 * Test if two ip6 socket addresses refer to the same socket by
261 * comparing relevant fields. The padding bytes specifically, are not
262 * compared. sin6_flowinfo is not compared because it only affects QoS
263 * and sin6_scope_id is only compared if the address is "link local"
264 * because "link local" addresses need only be unique to a specific
265 * link. Conversely, ordinary unicast addresses might have different
266 * sin6_scope_id.
267 *
268 * The caller should ensure both socket addresses are AF_INET6.
269 */
270static int nfs_sockaddr_cmp_ip6(const struct sockaddr *sa1,
271 const struct sockaddr *sa2)
272{
273 const struct sockaddr_in6 *saddr1 = (const struct sockaddr_in6 *)sa1;
274 const struct sockaddr_in6 *saddr2 = (const struct sockaddr_in6 *)sa2;
275
276 if (!ipv6_addr_equal(&saddr1->sin6_addr,
277 &saddr1->sin6_addr))
278 return 0;
279 if (ipv6_addr_scope(&saddr1->sin6_addr) == IPV6_ADDR_SCOPE_LINKLOCAL &&
280 saddr1->sin6_scope_id != saddr2->sin6_scope_id)
281 return 0;
282 return saddr1->sin6_port == saddr2->sin6_port;
283}
258#else 284#else
259static int nfs_sockaddr_match_ipaddr4(const struct sockaddr_in *sa1, 285static int nfs_sockaddr_match_ipaddr4(const struct sockaddr_in *sa1,
260 const struct sockaddr_in *sa2) 286 const struct sockaddr_in *sa2)
@@ -270,6 +296,12 @@ static int nfs_sockaddr_match_ipaddr(const struct sockaddr *sa1,
270 return nfs_sockaddr_match_ipaddr4((const struct sockaddr_in *)sa1, 296 return nfs_sockaddr_match_ipaddr4((const struct sockaddr_in *)sa1,
271 (const struct sockaddr_in *)sa2); 297 (const struct sockaddr_in *)sa2);
272} 298}
299
300static int nfs_sockaddr_cmp_ip6(const struct sockaddr * sa1,
301 const struct sockaddr * sa2)
302{
303 return 0;
304}
273#endif 305#endif
274 306
275/* 307/*
@@ -279,38 +311,18 @@ static int nfs_sockaddr_match_ipaddr(const struct sockaddr *sa1,
279 * 311 *
280 * The caller should ensure both socket addresses are AF_INET. 312 * The caller should ensure both socket addresses are AF_INET.
281 */ 313 */
282static int nfs_sockaddr_cmp_ip4(const struct sockaddr_in * saddr1, 314static int nfs_sockaddr_cmp_ip4(const struct sockaddr *sa1,
283 const struct sockaddr_in * saddr2) 315 const struct sockaddr *sa2)
284{ 316{
317 const struct sockaddr_in *saddr1 = (const struct sockaddr_in *)sa1;
318 const struct sockaddr_in *saddr2 = (const struct sockaddr_in *)sa2;
319
285 if (saddr1->sin_addr.s_addr != saddr2->sin_addr.s_addr) 320 if (saddr1->sin_addr.s_addr != saddr2->sin_addr.s_addr)
286 return 0; 321 return 0;
287 return saddr1->sin_port == saddr2->sin_port; 322 return saddr1->sin_port == saddr2->sin_port;
288} 323}
289 324
290/* 325/*
291 * Test if two ip6 socket addresses refer to the same socket by
292 * comparing relevant fields. The padding bytes specifically, are not
293 * compared. sin6_flowinfo is not compared because it only affects QoS
294 * and sin6_scope_id is only compared if the address is "link local"
295 * because "link local" addresses need only be unique to a specific
296 * link. Conversely, ordinary unicast addresses might have different
297 * sin6_scope_id.
298 *
299 * The caller should ensure both socket addresses are AF_INET6.
300 */
301static int nfs_sockaddr_cmp_ip6 (const struct sockaddr_in6 * saddr1,
302 const struct sockaddr_in6 * saddr2)
303{
304 if (!ipv6_addr_equal(&saddr1->sin6_addr,
305 &saddr1->sin6_addr))
306 return 0;
307 if (ipv6_addr_scope(&saddr1->sin6_addr) == IPV6_ADDR_SCOPE_LINKLOCAL &&
308 saddr1->sin6_scope_id != saddr2->sin6_scope_id)
309 return 0;
310 return saddr1->sin6_port == saddr2->sin6_port;
311}
312
313/*
314 * Test if two socket addresses represent the same actual socket, 326 * Test if two socket addresses represent the same actual socket,
315 * by comparing (only) relevant fields. 327 * by comparing (only) relevant fields.
316 */ 328 */
@@ -322,11 +334,9 @@ static int nfs_sockaddr_cmp(const struct sockaddr *sa1,
322 334
323 switch (sa1->sa_family) { 335 switch (sa1->sa_family) {
324 case AF_INET: 336 case AF_INET:
325 return nfs_sockaddr_cmp_ip4((const struct sockaddr_in *) sa1, 337 return nfs_sockaddr_cmp_ip4(sa1, sa2);
326 (const struct sockaddr_in *) sa2);
327 case AF_INET6: 338 case AF_INET6:
328 return nfs_sockaddr_cmp_ip6((const struct sockaddr_in6 *) sa1, 339 return nfs_sockaddr_cmp_ip6(sa1, sa2);
329 (const struct sockaddr_in6 *) sa2);
330 } 340 }
331 return 0; 341 return 0;
332} 342}