-/* $FreeBSD: src/sys/netinet6/scope6.c,v 1.1.2.2 2001/07/03 11:01:55 ume Exp $ */
+/* $FreeBSD: src/sys/netinet6/scope6.c,v 1.3 2002/03/25 10:12:51 ume Exp $ */
/* $KAME: scope6.c,v 1.10 2000/07/24 13:29:31 itojun Exp $ */
/*
#include <netinet6/in6_var.h>
#include <netinet6/scope6_var.h>
+extern lck_mtx_t *scope6_mutex;
+
struct scope6_id {
/*
* 16 is correspondent to 4bit multicast scope field.
*/
u_int32_t s6id_list[16];
};
-static size_t if_indexlim = 8;
+static size_t if_scope_indexlim = 8;
struct scope6_id *scope6_ids = NULL;
-void
-scope6_ifattach(ifp)
- struct ifnet *ifp;
+int
+scope6_ifattach(
+ struct ifnet *ifp)
{
- int s = splnet();
-
/*
* We have some arrays that should be indexed by if_index.
* since if_index will grow dynamically, they should grow too.
*/
- if (scope6_ids == NULL || if_index >= if_indexlim) {
+ lck_mtx_lock(scope6_mutex);
+ if (scope6_ids == NULL || if_index >= if_scope_indexlim) {
size_t n;
caddr_t q;
+ int newlim = if_scope_indexlim;
- while (if_index >= if_indexlim)
- if_indexlim <<= 1;
+ while (if_index >= newlim)
+ newlim <<= 1;
/* grow scope index array */
- n = if_indexlim * sizeof(struct scope6_id);
+ n = newlim * sizeof(struct scope6_id);
/* XXX: need new malloc type? */
q = (caddr_t)_MALLOC(n, M_IFADDR, M_WAITOK);
+ if (q == NULL) {
+ lck_mtx_unlock(scope6_mutex);
+ return ENOBUFS;
+ }
+ if_scope_indexlim = newlim;
bzero(q, n);
if (scope6_ids) {
bcopy((caddr_t)scope6_ids, q, n/2);
/* don't initialize if called twice */
if (SID.s6id_list[IPV6_ADDR_SCOPE_LINKLOCAL]) {
- splx(s);
- return;
+ lck_mtx_unlock(scope6_mutex);
+ return 0;
}
/*
SID.s6id_list[IPV6_ADDR_SCOPE_ORGLOCAL] = 1;
#endif
#undef SID
+ lck_mtx_unlock(scope6_mutex);
- splx(s);
+ return 0;
}
int
-scope6_set(ifp, idlist)
- struct ifnet *ifp;
- u_int32_t *idlist;
+scope6_set(
+ struct ifnet *ifp,
+ u_int32_t *idlist)
{
- int i, s;
+ int i;
int error = 0;
if (scope6_ids == NULL) /* paranoid? */
* interface addresses, routing table entries, PCB entries...
*/
- s = splnet();
-
+ lck_mtx_lock(scope6_mutex);
for (i = 0; i < 16; i++) {
if (idlist[i] &&
idlist[i] != scope6_ids[ifp->if_index].s6id_list[i]) {
* IDs, but we check the consistency for
* safety in later use.
*/
- splx(s);
+ lck_mtx_unlock(scope6_mutex);
return(EINVAL);
}
scope6_ids[ifp->if_index].s6id_list[i] = idlist[i];
}
}
- splx(s);
+ lck_mtx_unlock(scope6_mutex);
return(error);
}
int
-scope6_get(ifp, idlist)
- struct ifnet *ifp;
- u_int32_t *idlist;
+scope6_get(
+ struct ifnet *ifp,
+ u_int32_t *idlist)
{
if (scope6_ids == NULL) /* paranoid? */
return(EINVAL);
+ lck_mtx_lock(scope6_mutex);
bcopy(scope6_ids[ifp->if_index].s6id_list, idlist,
sizeof(scope6_ids[ifp->if_index].s6id_list));
+ lck_mtx_unlock(scope6_mutex);
return(0);
}
}
}
- if (bcmp(&in6addr_loopback, addr, sizeof(addr) - 1) == 0) {
+ if (bcmp(&in6addr_loopback, addr, sizeof(*addr) - 1) == 0) {
if (addr->s6_addr8[15] == 1) /* loopback */
return IPV6_ADDR_SCOPE_NODELOCAL;
if (addr->s6_addr8[15] == 0) /* unspecified */
}
int
-in6_addr2scopeid(ifp, addr)
- struct ifnet *ifp; /* must not be NULL */
- struct in6_addr *addr; /* must not be NULL */
+in6_addr2scopeid(
+ struct ifnet *ifp, /* must not be NULL */
+ struct in6_addr *addr) /* must not be NULL */
{
int scope = in6_addrscope(addr);
+ int index = ifp->if_index;
+ int retid = 0;
if (scope6_ids == NULL) /* paranoid? */
return(0); /* XXX */
- if (ifp->if_index >= if_indexlim)
+
+ lck_mtx_lock(scope6_mutex);
+ if (index >= if_scope_indexlim) {
+ lck_mtx_unlock(scope6_mutex);
return(0); /* XXX */
+ }
-#define SID scope6_ids[ifp->if_index]
+#define SID scope6_ids[index]
switch(scope) {
case IPV6_ADDR_SCOPE_NODELOCAL:
- return(-1); /* XXX: is this an appropriate value? */
-
+ retid = -1; /* XXX: is this an appropriate value? */
+ break;
case IPV6_ADDR_SCOPE_LINKLOCAL:
- return(SID.s6id_list[IPV6_ADDR_SCOPE_LINKLOCAL]);
-
+ retid=SID.s6id_list[IPV6_ADDR_SCOPE_LINKLOCAL];
+ break;
case IPV6_ADDR_SCOPE_SITELOCAL:
- return(SID.s6id_list[IPV6_ADDR_SCOPE_SITELOCAL]);
-
+ retid=SID.s6id_list[IPV6_ADDR_SCOPE_SITELOCAL];
+ break;
case IPV6_ADDR_SCOPE_ORGLOCAL:
- return(SID.s6id_list[IPV6_ADDR_SCOPE_ORGLOCAL]);
-
+ retid=SID.s6id_list[IPV6_ADDR_SCOPE_ORGLOCAL];
+ break;
default:
- return(0); /* XXX: treat as global. */
+ break; /* XXX: value 0, treat as global. */
}
#undef SID
+
+ lck_mtx_unlock(scope6_mutex);
+ return retid;
}
void
-scope6_setdefault(ifp)
- struct ifnet *ifp; /* note that this might be NULL */
+scope6_setdefault(
+ struct ifnet *ifp) /* note that this might be NULL */
{
/*
* Currently, this function just set the default "link" according to
* We might eventually have to separate the notion of "link" from
* "interface" and provide a user interface to set the default.
*/
+ lck_mtx_lock(scope6_mutex);
if (ifp) {
scope6_ids[0].s6id_list[IPV6_ADDR_SCOPE_LINKLOCAL] =
ifp->if_index;
}
else
scope6_ids[0].s6id_list[IPV6_ADDR_SCOPE_LINKLOCAL] = 0;
+ lck_mtx_unlock(scope6_mutex);
}
int
-scope6_get_default(idlist)
- u_int32_t *idlist;
+scope6_get_default(
+ u_int32_t *idlist)
{
if (scope6_ids == NULL) /* paranoid? */
return(EINVAL);
+ lck_mtx_lock(scope6_mutex);
bcopy(scope6_ids[0].s6id_list, idlist,
sizeof(scope6_ids[0].s6id_list));
+ lck_mtx_unlock(scope6_mutex);
return(0);
}
u_int32_t
-scope6_addr2default(addr)
- struct in6_addr *addr;
+scope6_addr2default(
+ struct in6_addr *addr)
{
- return(scope6_ids[0].s6id_list[in6_addrscope(addr)]);
+ u_int32_t id = 0;
+ int index = in6_addrscope(addr);
+ lck_mtx_lock(scope6_mutex);
+ id = scope6_ids[0].s6id_list[index];
+ lck_mtx_unlock(scope6_mutex);
+ return (id);
}