summaryrefslogtreecommitdiffstats
path: root/lib/addr.c
diff options
context:
space:
mode:
authorThomas Graf <tgraf@suug.ch>2007-09-14 23:28:01 (GMT)
committerThomas Graf <tgraf@suug.ch>2007-09-14 23:28:01 (GMT)
commit44d362409d5469aed47d19e7908d19bd194493a4 (patch)
tree5d1e739a4566f3af796273e5c3f78ca53d234df6 /lib/addr.c
downloadlibnl-44d362409d5469aed47d19e7908d19bd194493a4.zip
libnl-44d362409d5469aed47d19e7908d19bd194493a4.tar.gz
libnl-44d362409d5469aed47d19e7908d19bd194493a4.tar.bz2
Initial import
Diffstat (limited to 'lib/addr.c')
-rw-r--r--lib/addr.c883
1 files changed, 883 insertions, 0 deletions
diff --git a/lib/addr.c b/lib/addr.c
new file mode 100644
index 0000000..7fe3781
--- /dev/null
+++ b/lib/addr.c
@@ -0,0 +1,883 @@
+/*
+ * lib/addr.c Abstract Address
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation version 2.1
+ * of the License.
+ *
+ * Copyright (c) 2003-2006 Thomas Graf <tgraf@suug.ch>
+ */
+
+/**
+ * @ingroup utils
+ * @defgroup addr Abstract Address
+ *
+ * @par 1) Transform character string to abstract address
+ * @code
+ * struct nl_addr *a = nl_addr_parse("::1", AF_UNSPEC);
+ * printf("Address family: %s\n", nl_af2str(nl_addr_get_family(a)));
+ * nl_addr_put(a);
+ * a = nl_addr_parse("11:22:33:44:55:66", AF_UNSPEC);
+ * printf("Address family: %s\n", nl_af2str(nl_addr_get_family(a)));
+ * nl_addr_put(a);
+ * @endcode
+ * @{
+ */
+
+#include <netlink-local.h>
+#include <netlink/netlink.h>
+#include <netlink/utils.h>
+#include <netlink/addr.h>
+#include <linux/socket.h>
+
+/* All this DECnet stuff is stolen from iproute2, thanks to whoever wrote
+ * this, probably Alexey. */
+static inline uint16_t dn_ntohs(uint16_t addr)
+{
+ union {
+ uint8_t byte[2];
+ uint16_t word;
+ } u = {
+ .word = addr,
+ };
+
+ return ((uint16_t) u.byte[0]) | (((uint16_t) u.byte[1]) << 8);
+}
+
+static inline int do_digit(char *str, uint16_t *addr, uint16_t scale,
+ size_t *pos, size_t len, int *started)
+{
+ uint16_t tmp = *addr / scale;
+
+ if (*pos == len)
+ return 1;
+
+ if (((tmp) > 0) || *started || (scale == 1)) {
+ *str = tmp + '0';
+ *started = 1;
+ (*pos)++;
+ *addr -= (tmp * scale);
+ }
+
+ return 0;
+}
+
+static const char *dnet_ntop(char *addrbuf, size_t addrlen, char *str,
+ size_t len)
+{
+ uint16_t addr = dn_ntohs(*(uint16_t *)addrbuf);
+ uint16_t area = addr >> 10;
+ size_t pos = 0;
+ int started = 0;
+
+ if (addrlen != 2)
+ return NULL;
+
+ addr &= 0x03ff;
+
+ if (len == 0)
+ return str;
+
+ if (do_digit(str + pos, &area, 10, &pos, len, &started))
+ return str;
+
+ if (do_digit(str + pos, &area, 1, &pos, len, &started))
+ return str;
+
+ if (pos == len)
+ return str;
+
+ *(str + pos) = '.';
+ pos++;
+ started = 0;
+
+ if (do_digit(str + pos, &addr, 1000, &pos, len, &started))
+ return str;
+
+ if (do_digit(str + pos, &addr, 100, &pos, len, &started))
+ return str;
+
+ if (do_digit(str + pos, &addr, 10, &pos, len, &started))
+ return str;
+
+ if (do_digit(str + pos, &addr, 1, &pos, len, &started))
+ return str;
+
+ if (pos == len)
+ return str;
+
+ *(str + pos) = 0;
+
+ return str;
+}
+
+static int dnet_num(const char *src, uint16_t * dst)
+{
+ int rv = 0;
+ int tmp;
+ *dst = 0;
+
+ while ((tmp = *src++) != 0) {
+ tmp -= '0';
+ if ((tmp < 0) || (tmp > 9))
+ return rv;
+
+ rv++;
+ (*dst) *= 10;
+ (*dst) += tmp;
+ }
+
+ return rv;
+}
+
+static inline int dnet_pton(const char *src, char *addrbuf)
+{
+ uint16_t area = 0;
+ uint16_t node = 0;
+ int pos;
+
+ pos = dnet_num(src, &area);
+ if ((pos == 0) || (area > 63) ||
+ ((*(src + pos) != '.') && (*(src + pos) != ',')))
+ return -EINVAL;
+
+ pos = dnet_num(src + pos + 1, &node);
+ if ((pos == 0) || (node > 1023))
+ return -EINVAL;
+
+ *(uint16_t *)addrbuf = dn_ntohs((area << 10) | node);
+
+ return 1;
+}
+
+/**
+ * @name Creating Abstract Addresses
+ * @{
+ */
+
+/**
+ * Allocate new abstract address object.
+ * @arg maxsize Maximum size of the binary address.
+ * @return Newly allocated address object or NULL
+ */
+struct nl_addr *nl_addr_alloc(size_t maxsize)
+{
+ struct nl_addr *addr;
+
+ addr = calloc(1, sizeof(*addr) + maxsize);
+ if (!addr) {
+ nl_errno(ENOMEM);
+ return NULL;
+ }
+
+ addr->a_refcnt = 1;
+ addr->a_maxsize = maxsize;
+
+ return addr;
+}
+
+/**
+ * Allocate new abstract address object based on a binary address.
+ * @arg family Address family.
+ * @arg buf Buffer containing the binary address.
+ * @arg size Length of binary address buffer.
+ * @return Newly allocated address handle or NULL
+ */
+struct nl_addr *nl_addr_build(int family, void *buf, size_t size)
+{
+ struct nl_addr *addr;
+
+ addr = nl_addr_alloc(size);
+ if (!addr)
+ return NULL;
+
+ addr->a_family = family;
+ addr->a_len = size;
+ addr->a_prefixlen = size*8;
+
+ if (size)
+ memcpy(addr->a_addr, buf, size);
+
+ return addr;
+}
+
+/**
+ * Allocate abstract address object based on a character string
+ * @arg addrstr Address represented as character string.
+ * @arg hint Address family hint or AF_UNSPEC.
+ *
+ * Regognizes the following address formats:
+ *@code
+ * Format Len Family
+ * ----------------------------------------------------------------
+ * IPv6 address format 16 AF_INET6
+ * ddd.ddd.ddd.ddd 4 AF_INET
+ * HH:HH:HH:HH:HH:HH 6 AF_LLC
+ * AA{.|,}NNNN 2 AF_DECnet
+ * HH:HH:HH:... variable AF_UNSPEC
+ * @endcode
+ *
+ * Special values:
+ * - none: All bits and length set to 0.
+ * - {default|all|any}: All bits set to 0, length based on hint or
+ * AF_INET if no hint is given.
+ *
+ * The prefix length may be appened at the end prefixed with a
+ * slash, e.g. 10.0.0.0/8.
+ *
+ * @return Newly allocated abstract address object or NULL.
+ */
+struct nl_addr *nl_addr_parse(const char *addrstr, int hint)
+{
+ int err, copy = 0, len = 0, family = AF_UNSPEC;
+ char *str, *prefix, buf[32];
+ struct nl_addr *addr = NULL; /* gcc ain't that smart */
+
+ str = strdup(addrstr);
+ if (!str) {
+ err = nl_errno(ENOMEM);
+ goto errout;
+ }
+
+ prefix = strchr(str, '/');
+ if (prefix)
+ *prefix = '\0';
+
+ if (!strcasecmp(str, "none")) {
+ family = hint;
+ goto prefix;
+ }
+
+ if (!strcasecmp(str, "default") ||
+ !strcasecmp(str, "all") ||
+ !strcasecmp(str, "any")) {
+
+ switch (hint) {
+ case AF_INET:
+ case AF_UNSPEC:
+ /* Kind of a hack, we assume that if there is
+ * no hint given the user wants to have a IPv4
+ * address given back. */
+ family = AF_INET;
+ len = 4;
+ goto prefix;
+
+ case AF_INET6:
+ family = AF_INET6;
+ len = 16;
+ goto prefix;
+
+ case AF_LLC:
+ family = AF_LLC;
+ len = 6;
+ goto prefix;
+
+ default:
+ err = nl_error(EINVAL, "Unsuported address" \
+ "family for default address");
+ goto errout;
+ }
+ }
+
+ copy = 1;
+
+ if (hint == AF_INET || hint == AF_UNSPEC) {
+ if (inet_pton(AF_INET, str, buf) > 0) {
+ family = AF_INET;
+ len = 4;
+ goto prefix;
+ }
+ if (hint == AF_INET) {
+ err = nl_error(EINVAL, "Invalid IPv4 address");
+ goto errout;
+ }
+ }
+
+ if (hint == AF_INET6 || hint == AF_UNSPEC) {
+ if (inet_pton(AF_INET6, str, buf) > 0) {
+ family = AF_INET6;
+ len = 16;
+ goto prefix;
+ }
+ if (hint == AF_INET6) {
+ err = nl_error(EINVAL, "Invalid IPv6 address");
+ goto errout;
+ }
+ }
+
+ if ((hint == AF_LLC || hint == AF_UNSPEC) && strchr(str, ':')) {
+ unsigned int a, b, c, d, e, f;
+
+ if (sscanf(str, "%02x:%02x:%02x:%02x:%02x:%02x",
+ &a, &b, &c, &d, &e, &f) == 6) {
+ family = AF_LLC;
+ len = 6;
+ buf[0] = (unsigned char) a;
+ buf[1] = (unsigned char) b;
+ buf[2] = (unsigned char) c;
+ buf[3] = (unsigned char) d;
+ buf[4] = (unsigned char) e;
+ buf[5] = (unsigned char) f;
+ goto prefix;
+ }
+
+ if (hint == AF_LLC) {
+ err = nl_error(EINVAL, "Invalid link layer address");
+ goto errout;
+ }
+ }
+
+ if ((hint == AF_DECnet || hint == AF_UNSPEC) &&
+ (strchr(str, '.') || strchr(str, ','))) {
+ if (dnet_pton(str, buf) > 0) {
+ family = AF_DECnet;
+ len = 2;
+ goto prefix;
+ }
+ if (hint == AF_DECnet) {
+ err = nl_error(EINVAL, "Invalid DECnet address");
+ goto errout;
+ }
+ }
+
+ if (hint == AF_UNSPEC && strchr(str, ':')) {
+ int i = 0;
+ char *s = str, *p;
+ for (;;) {
+ long l = strtol(s, &p, 16);
+
+ if (s == p || l > 0xff || i >= sizeof(buf)) {
+ err = -EINVAL;
+ goto errout;
+ }
+
+ buf[i++] = (unsigned char) l;
+ if (*p == '\0')
+ break;
+ s = ++p;
+ }
+
+ len = i;
+ family = AF_UNSPEC;
+ goto prefix;
+ }
+
+ err = nl_error(EINVAL, "Invalid address");
+ goto errout;
+
+prefix:
+ addr = nl_addr_alloc(len);
+ if (!addr) {
+ err = nl_errno(ENOMEM);
+ goto errout;
+ }
+
+ nl_addr_set_family(addr, family);
+
+ if (copy)
+ nl_addr_set_binary_addr(addr, buf, len);
+
+ if (prefix) {
+ char *p;
+ long pl = strtol(++prefix, &p, 0);
+ if (p == prefix) {
+ nl_addr_destroy(addr);
+ err = -EINVAL;
+ goto errout;
+ }
+ nl_addr_set_prefixlen(addr, pl);
+ } else
+ nl_addr_set_prefixlen(addr, len * 8);
+
+ err = 0;
+errout:
+ free(str);
+
+ return err ? NULL : addr;
+}
+
+/**
+ * Clone existing abstract address object.
+ * @arg addr Abstract address object.
+ * @return Newly allocated abstract address object being a duplicate of the
+ * specified address object or NULL if a failure occured.
+ */
+struct nl_addr *nl_addr_clone(struct nl_addr *addr)
+{
+ struct nl_addr *new;
+
+ new = nl_addr_build(addr->a_family, addr->a_addr, addr->a_len);
+ if (new)
+ new->a_prefixlen = addr->a_prefixlen;
+
+ return new;
+}
+
+/** @} */
+
+/**
+ * @name Destroying Abstract Addresses
+ * @{
+ */
+
+/**
+ * Destroy abstract address object.
+ * @arg addr Abstract address object.
+ */
+void nl_addr_destroy(struct nl_addr *addr)
+{
+ if (!addr)
+ return;
+
+ if (addr->a_refcnt != 1)
+ BUG();
+
+ free(addr);
+}
+
+/** @} */
+
+/**
+ * @name Managing Usage References
+ * @{
+ */
+
+struct nl_addr *nl_addr_get(struct nl_addr *addr)
+{
+ addr->a_refcnt++;
+
+ return addr;
+}
+
+void nl_addr_put(struct nl_addr *addr)
+{
+ if (!addr)
+ return;
+
+ if (addr->a_refcnt == 1)
+ nl_addr_destroy(addr);
+ else
+ addr->a_refcnt--;
+}
+
+/**
+ * Check whether an abstract address object is shared.
+ * @arg addr Abstract address object.
+ * @return Non-zero if the abstract address object is shared, otherwise 0.
+ */
+int nl_addr_shared(struct nl_addr *addr)
+{
+ return addr->a_refcnt > 1;
+}
+
+/** @} */
+
+/**
+ * @name Miscellaneous
+ * @{
+ */
+
+/**
+ * Compares two abstract address objects.
+ * @arg a A abstract address object.
+ * @arg b Another abstract address object.
+ *
+ * @return Integer less than, equal to or greather than zero if \c is found,
+ * respectively to be less than, to, or be greater than \c b.
+ */
+int nl_addr_cmp(struct nl_addr *a, struct nl_addr *b)
+{
+ int d = a->a_family - b->a_family;
+
+ if (d == 0) {
+ d = a->a_len - b->a_len;
+
+ if (a->a_len && d == 0)
+ return memcmp(a->a_addr, b->a_addr, a->a_len);
+ }
+
+ return d;
+}
+
+/**
+ * Compares the prefix of two abstract address objects.
+ * @arg a A abstract address object.
+ * @arg b Another abstract address object.
+ *
+ * @return Integer less than, equal to or greather than zero if \c is found,
+ * respectively to be less than, to, or be greater than \c b.
+ */
+int nl_addr_cmp_prefix(struct nl_addr *a, struct nl_addr *b)
+{
+ int d = a->a_family - b->a_family;
+
+ if (d == 0) {
+ int len = min(a->a_prefixlen, b->a_prefixlen);
+ int bytes = len / 8;
+
+ d = memcmp(a->a_addr, b->a_addr, bytes);
+ if (d == 0) {
+ int mask = (1UL << (len % 8)) - 1UL;
+
+ d = (a->a_addr[bytes] & mask) -
+ (b->a_addr[bytes] & mask);
+ }
+ }
+
+ return d;
+}
+
+/**
+ * Check if an address matches a certain family.
+ * @arg addr Address represented as character string.
+ * @arg family Desired address family.
+ *
+ * @return 1 if the address is of the desired address family,
+ * otherwise 0 is returned.
+ */
+int nl_addr_valid(char *addr, int family)
+{
+ int ret;
+ char buf[32];
+
+ switch (family) {
+ case AF_INET:
+ case AF_INET6:
+ ret = inet_pton(family, addr, buf);
+ if (ret <= 0)
+ return 0;
+ break;
+
+ case AF_DECnet:
+ ret = dnet_pton(addr, buf);
+ if (ret <= 0)
+ return 0;
+ break;
+
+ case AF_LLC:
+ if (sscanf(addr, "%*02x:%*02x:%*02x:%*02x:%*02x:%*02x") != 6)
+ return 0;
+ break;
+ }
+
+ return 1;
+}
+
+/**
+ * Guess address family of an abstract address object based on address size.
+ * @arg addr Abstract address object.
+ * @return Address family or AF_UNSPEC if guessing wasn't successful.
+ */
+int nl_addr_guess_family(struct nl_addr *addr)
+{
+ switch (addr->a_len) {
+ case 4:
+ return AF_INET;
+ case 6:
+ return AF_LLC;
+ case 16:
+ return AF_INET6;
+ default:
+ return AF_UNSPEC;
+ }
+}
+
+/**
+ * Fill out sockaddr structure with values from abstract address object.
+ * @arg addr Abstract address object.
+ * @arg sa Destination sockaddr structure buffer.
+ * @arg salen Length of sockaddr structure buffer.
+ *
+ * Fills out the specified sockaddr structure with the data found in the
+ * specified abstract address. The salen argument needs to be set to the
+ * size of sa but will be modified to the actual size used during before
+ * the function exits.
+ *
+ * @return 0 on success or a negative error code
+ */
+int nl_addr_fill_sockaddr(struct nl_addr *addr, struct sockaddr *sa,
+ socklen_t *salen)
+{
+ switch (addr->a_family) {
+ case AF_INET: {
+ struct sockaddr_in *sai = (struct sockaddr_in *) sa;
+
+ if (*salen < sizeof(*sai))
+ return -EINVAL;
+
+ sai->sin_family = addr->a_family;
+ memcpy(&sai->sin_addr, addr->a_addr, 4);
+ *salen = sizeof(*sai);
+ }
+ break;
+
+ case AF_INET6: {
+ struct sockaddr_in6 *sa6 = (struct sockaddr_in6 *) sa;
+
+ if (*salen < sizeof(*sa6))
+ return -EINVAL;
+
+ sa6->sin6_family = addr->a_family;
+ memcpy(&sa6->sin6_addr, addr->a_addr, 16);
+ *salen = sizeof(*sa6);
+ }
+ break;
+
+ default:
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+
+/** @} */
+
+/**
+ * @name Getting Information About Addresses
+ * @{
+ */
+
+/**
+ * Call getaddrinfo() for an abstract address object.
+ * @arg addr Abstract address object.
+ *
+ * Calls getaddrinfo() for the specified abstract address in AI_NUMERICHOST
+ * mode.
+ *
+ * @note The caller is responsible for freeing the linked list using the
+ * interface provided by getaddrinfo(3).
+ *
+ * @return A linked list of addrinfo handles or NULL with an error message
+ * associated.
+ */
+struct addrinfo *nl_addr_info(struct nl_addr *addr)
+{
+ int err;
+ struct addrinfo *res;
+ char buf[INET6_ADDRSTRLEN+5];
+ struct addrinfo hint = {
+ .ai_flags = AI_NUMERICHOST,
+ .ai_family = addr->a_family,
+ };
+
+ nl_addr2str(addr, buf, sizeof(buf));
+
+ err = getaddrinfo(buf, NULL, &hint, &res);
+ if (err != 0) {
+ nl_error(err, gai_strerror(err));
+ return NULL;
+ }
+
+ return res;
+}
+
+/**
+ * Resolve abstract address object to a name using getnameinfo().
+ * @arg addr Abstract address object.
+ * @arg host Destination buffer for host name.
+ * @arg hostlen Length of destination buffer.
+ *
+ * Resolves the abstract address to a name and writes the looked up result
+ * into the host buffer. getnameinfo() is used to perform the lookup and
+ * is put into NI_NAMEREQD mode so the function will fail if the lookup
+ * couldn't be performed.
+ *
+ * @return 0 on success or a negative error code.
+ */
+int nl_addr_resolve(struct nl_addr *addr, char *host, size_t hostlen)
+{
+ int err;
+ struct sockaddr_in6 buf;
+ socklen_t salen = sizeof(buf);
+
+ err = nl_addr_fill_sockaddr(addr, (struct sockaddr *) &buf, &salen);
+ if (err < 0)
+ return err;
+
+ return getnameinfo((struct sockaddr *) &buf, salen,
+ host, hostlen, NULL, 0, NI_NAMEREQD);
+}
+
+/** @} */
+
+/**
+ * @name Attributes
+ * @{
+ */
+
+void nl_addr_set_family(struct nl_addr *addr, int family)
+{
+ addr->a_family = family;
+}
+
+int nl_addr_get_family(struct nl_addr *addr)
+{
+ return addr->a_family;
+}
+
+/**
+ * Set binary address of abstract address object.
+ * @arg addr Abstract address object.
+ * @arg buf Buffer containing binary address.
+ * @arg len Length of buffer containing binary address.
+ */
+int nl_addr_set_binary_addr(struct nl_addr *addr, void *buf, size_t len)
+{
+ if (len > addr->a_maxsize)
+ return -ERANGE;
+
+ addr->a_len = len;
+ memcpy(addr->a_addr, buf, len);
+
+ return 0;
+}
+
+/**
+ * Get binary address of abstract address object.
+ * @arg addr Abstract address object.
+ */
+void *nl_addr_get_binary_addr(struct nl_addr *addr)
+{
+ return addr->a_addr;
+}
+
+/**
+ * Get length of binary address of abstract address object.
+ * @arg addr Abstract address object.
+ */
+unsigned int nl_addr_get_len(struct nl_addr *addr)
+{
+ return addr->a_len;
+}
+
+void nl_addr_set_prefixlen(struct nl_addr *addr, int prefixlen)
+{
+ addr->a_prefixlen = prefixlen;
+}
+
+/**
+ * Get prefix length of abstract address object.
+ * @arg addr Abstract address object.
+ */
+unsigned int nl_addr_get_prefixlen(struct nl_addr *addr)
+{
+ return addr->a_prefixlen;
+}
+
+/** @} */
+
+/**
+ * @name Translations to Strings
+ * @{
+ */
+
+/**
+ * Convert abstract address object to character string.
+ * @arg addr Abstract address object.
+ * @arg buf Destination buffer.
+ * @arg size Size of destination buffer.
+ *
+ * Converts an abstract address to a character string and stores
+ * the result in the specified destination buffer.
+ *
+ * @return Address represented in ASCII stored in destination buffer.
+ */
+char *nl_addr2str(struct nl_addr *addr, char *buf, size_t size)
+{
+ int i;
+ char tmp[16];
+
+ if (!addr->a_len) {
+ snprintf(buf, size, "none");
+ goto prefix;
+ }
+
+ switch (addr->a_family) {
+ case AF_INET:
+ inet_ntop(AF_INET, addr->a_addr, buf, size);
+ break;
+
+ case AF_INET6:
+ inet_ntop(AF_INET6, addr->a_addr, buf, size);
+ break;
+
+ case AF_DECnet:
+ dnet_ntop(addr->a_addr, addr->a_len, buf, size);
+ break;
+
+ case AF_LLC:
+ default:
+ snprintf(buf, size, "%02x",
+ (unsigned char) addr->a_addr[0]);
+ for (i = 1; i < addr->a_len; i++) {
+ snprintf(tmp, sizeof(tmp), ":%02x",
+ (unsigned char) addr->a_addr[i]);
+ strncat(buf, tmp, size - strlen(buf) - 1);
+ }
+ break;
+ }
+
+prefix:
+ if (addr->a_prefixlen != (8 * addr->a_len)) {
+ snprintf(tmp, sizeof(tmp), "/%u", addr->a_prefixlen);
+ strncat(buf, tmp, size - strlen(buf) - 1);
+ }
+
+ return buf;
+}
+
+/** @} */
+
+/**
+ * @name Address Family Transformations
+ * @{
+ */
+
+static struct trans_tbl afs[] = {
+ __ADD(AF_UNSPEC,unspec)
+ __ADD(AF_UNIX,unix)
+ __ADD(AF_LOCAL,local)
+ __ADD(AF_INET,inet)
+ __ADD(AF_AX25,ax25)
+ __ADD(AF_IPX,ipx)
+ __ADD(AF_APPLETALK,appletalk)
+ __ADD(AF_NETROM,netrom)
+ __ADD(AF_BRIDGE,bridge)
+ __ADD(AF_ATMPVC,atmpvc)
+ __ADD(AF_X25,x25)
+ __ADD(AF_INET6,inet6)
+ __ADD(AF_ROSE,rose)
+ __ADD(AF_DECnet,decnet)
+ __ADD(AF_NETBEUI,netbeui)
+ __ADD(AF_SECURITY,security)
+ __ADD(AF_KEY,key)
+ __ADD(AF_NETLINK,netlink)
+ __ADD(AF_ROUTE,route)
+ __ADD(AF_PACKET,packet)
+ __ADD(AF_ASH,ash)
+ __ADD(AF_ECONET,econet)
+ __ADD(AF_ATMSVC,atmsvc)
+ __ADD(AF_SNA,sna)
+ __ADD(AF_IRDA,irda)
+ __ADD(AF_PPPOX,pppox)
+ __ADD(AF_WANPIPE,wanpipe)
+ __ADD(AF_LLC,llc)
+ __ADD(AF_BLUETOOTH,bluetooth)
+};
+
+char *nl_af2str(int family, char *buf, size_t size)
+{
+ return __type2str(family, buf, size, afs, ARRAY_SIZE(afs));
+}
+
+int nl_str2af(const char *name)
+{
+ int fam = __str2type(name, afs, ARRAY_SIZE(afs));
+ return fam >= 0 ? fam : AF_UNSPEC;
+}
+
+/** @} */
+
+/** @} */