addr.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898
  1. /*
  2. * lib/addr.c Abstract Address
  3. *
  4. * This library is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Lesser General Public
  6. * License as published by the Free Software Foundation version 2.1
  7. * of the License.
  8. *
  9. * Copyright (c) 2003-2006 Thomas Graf <tgraf@suug.ch>
  10. */
  11. /**
  12. * @ingroup utils
  13. * @defgroup addr Abstract Address
  14. *
  15. * @par 1) Transform character string to abstract address
  16. * @code
  17. * struct nl_addr *a = nl_addr_parse("::1", AF_UNSPEC);
  18. * printf("Address family: %s\n", nl_af2str(nl_addr_get_family(a)));
  19. * nl_addr_put(a);
  20. * a = nl_addr_parse("11:22:33:44:55:66", AF_UNSPEC);
  21. * printf("Address family: %s\n", nl_af2str(nl_addr_get_family(a)));
  22. * nl_addr_put(a);
  23. * @endcode
  24. * @{
  25. */
  26. #include <netlink-local.h>
  27. #include <netlink/netlink.h>
  28. #include <netlink/utils.h>
  29. #include <netlink/addr.h>
  30. #include <linux/socket.h>
  31. /* All this DECnet stuff is stolen from iproute2, thanks to whoever wrote
  32. * this, probably Alexey. */
  33. static inline uint16_t dn_ntohs(uint16_t addr)
  34. {
  35. union {
  36. uint8_t byte[2];
  37. uint16_t word;
  38. } u = {
  39. .word = addr,
  40. };
  41. return ((uint16_t) u.byte[0]) | (((uint16_t) u.byte[1]) << 8);
  42. }
  43. static inline int do_digit(char *str, uint16_t *addr, uint16_t scale,
  44. size_t *pos, size_t len, int *started)
  45. {
  46. uint16_t tmp = *addr / scale;
  47. if (*pos == len)
  48. return 1;
  49. if (((tmp) > 0) || *started || (scale == 1)) {
  50. *str = tmp + '0';
  51. *started = 1;
  52. (*pos)++;
  53. *addr -= (tmp * scale);
  54. }
  55. return 0;
  56. }
  57. static const char *dnet_ntop(char *addrbuf, size_t addrlen, char *str,
  58. size_t len)
  59. {
  60. uint16_t addr = dn_ntohs(*(uint16_t *)addrbuf);
  61. uint16_t area = addr >> 10;
  62. size_t pos = 0;
  63. int started = 0;
  64. if (addrlen != 2)
  65. return NULL;
  66. addr &= 0x03ff;
  67. if (len == 0)
  68. return str;
  69. if (do_digit(str + pos, &area, 10, &pos, len, &started))
  70. return str;
  71. if (do_digit(str + pos, &area, 1, &pos, len, &started))
  72. return str;
  73. if (pos == len)
  74. return str;
  75. *(str + pos) = '.';
  76. pos++;
  77. started = 0;
  78. if (do_digit(str + pos, &addr, 1000, &pos, len, &started))
  79. return str;
  80. if (do_digit(str + pos, &addr, 100, &pos, len, &started))
  81. return str;
  82. if (do_digit(str + pos, &addr, 10, &pos, len, &started))
  83. return str;
  84. if (do_digit(str + pos, &addr, 1, &pos, len, &started))
  85. return str;
  86. if (pos == len)
  87. return str;
  88. *(str + pos) = 0;
  89. return str;
  90. }
  91. static int dnet_num(const char *src, uint16_t * dst)
  92. {
  93. int rv = 0;
  94. int tmp;
  95. *dst = 0;
  96. while ((tmp = *src++) != 0) {
  97. tmp -= '0';
  98. if ((tmp < 0) || (tmp > 9))
  99. return rv;
  100. rv++;
  101. (*dst) *= 10;
  102. (*dst) += tmp;
  103. }
  104. return rv;
  105. }
  106. static inline int dnet_pton(const char *src, char *addrbuf)
  107. {
  108. uint16_t area = 0;
  109. uint16_t node = 0;
  110. int pos;
  111. pos = dnet_num(src, &area);
  112. if ((pos == 0) || (area > 63) ||
  113. ((*(src + pos) != '.') && (*(src + pos) != ',')))
  114. return -EINVAL;
  115. pos = dnet_num(src + pos + 1, &node);
  116. if ((pos == 0) || (node > 1023))
  117. return -EINVAL;
  118. *(uint16_t *)addrbuf = dn_ntohs((area << 10) | node);
  119. return 1;
  120. }
  121. /**
  122. * @name Creating Abstract Addresses
  123. * @{
  124. */
  125. /**
  126. * Allocate new abstract address object.
  127. * @arg maxsize Maximum size of the binary address.
  128. * @return Newly allocated address object or NULL
  129. */
  130. struct nl_addr *nl_addr_alloc(size_t maxsize)
  131. {
  132. struct nl_addr *addr;
  133. addr = calloc(1, sizeof(*addr) + maxsize);
  134. if (!addr) {
  135. nl_errno(ENOMEM);
  136. return NULL;
  137. }
  138. addr->a_refcnt = 1;
  139. addr->a_maxsize = maxsize;
  140. return addr;
  141. }
  142. /**
  143. * Allocate new abstract address object based on a binary address.
  144. * @arg family Address family.
  145. * @arg buf Buffer containing the binary address.
  146. * @arg size Length of binary address buffer.
  147. * @return Newly allocated address handle or NULL
  148. */
  149. struct nl_addr *nl_addr_build(int family, void *buf, size_t size)
  150. {
  151. struct nl_addr *addr;
  152. addr = nl_addr_alloc(size);
  153. if (!addr)
  154. return NULL;
  155. addr->a_family = family;
  156. addr->a_len = size;
  157. addr->a_prefixlen = size*8;
  158. if (size)
  159. memcpy(addr->a_addr, buf, size);
  160. return addr;
  161. }
  162. /**
  163. * Allocate abstract address object based on a character string
  164. * @arg addrstr Address represented as character string.
  165. * @arg hint Address family hint or AF_UNSPEC.
  166. *
  167. * Regognizes the following address formats:
  168. *@code
  169. * Format Len Family
  170. * ----------------------------------------------------------------
  171. * IPv6 address format 16 AF_INET6
  172. * ddd.ddd.ddd.ddd 4 AF_INET
  173. * HH:HH:HH:HH:HH:HH 6 AF_LLC
  174. * AA{.|,}NNNN 2 AF_DECnet
  175. * HH:HH:HH:... variable AF_UNSPEC
  176. * @endcode
  177. *
  178. * Special values:
  179. * - none: All bits and length set to 0.
  180. * - {default|all|any}: All bits set to 0, length based on hint or
  181. * AF_INET if no hint is given.
  182. *
  183. * The prefix length may be appened at the end prefixed with a
  184. * slash, e.g. 10.0.0.0/8.
  185. *
  186. * @return Newly allocated abstract address object or NULL.
  187. */
  188. struct nl_addr *nl_addr_parse(const char *addrstr, int hint)
  189. {
  190. int err, copy = 0, len = 0, family = AF_UNSPEC;
  191. char *str, *prefix, buf[32];
  192. struct nl_addr *addr = NULL; /* gcc ain't that smart */
  193. str = strdup(addrstr);
  194. if (!str) {
  195. err = nl_errno(ENOMEM);
  196. goto errout;
  197. }
  198. prefix = strchr(str, '/');
  199. if (prefix)
  200. *prefix = '\0';
  201. if (!strcasecmp(str, "none")) {
  202. family = hint;
  203. goto prefix;
  204. }
  205. if (!strcasecmp(str, "default") ||
  206. !strcasecmp(str, "all") ||
  207. !strcasecmp(str, "any")) {
  208. switch (hint) {
  209. case AF_INET:
  210. case AF_UNSPEC:
  211. /* Kind of a hack, we assume that if there is
  212. * no hint given the user wants to have a IPv4
  213. * address given back. */
  214. family = AF_INET;
  215. len = 4;
  216. goto prefix;
  217. case AF_INET6:
  218. family = AF_INET6;
  219. len = 16;
  220. goto prefix;
  221. case AF_LLC:
  222. family = AF_LLC;
  223. len = 6;
  224. goto prefix;
  225. default:
  226. err = nl_error(EINVAL, "Unsuported address" \
  227. "family for default address");
  228. goto errout;
  229. }
  230. }
  231. copy = 1;
  232. if (hint == AF_INET || hint == AF_UNSPEC) {
  233. if (inet_pton(AF_INET, str, buf) > 0) {
  234. family = AF_INET;
  235. len = 4;
  236. goto prefix;
  237. }
  238. if (hint == AF_INET) {
  239. err = nl_error(EINVAL, "Invalid IPv4 address");
  240. goto errout;
  241. }
  242. }
  243. if (hint == AF_INET6 || hint == AF_UNSPEC) {
  244. if (inet_pton(AF_INET6, str, buf) > 0) {
  245. family = AF_INET6;
  246. len = 16;
  247. goto prefix;
  248. }
  249. if (hint == AF_INET6) {
  250. err = nl_error(EINVAL, "Invalid IPv6 address");
  251. goto errout;
  252. }
  253. }
  254. if ((hint == AF_LLC || hint == AF_UNSPEC) && strchr(str, ':')) {
  255. unsigned int a, b, c, d, e, f;
  256. if (sscanf(str, "%02x:%02x:%02x:%02x:%02x:%02x",
  257. &a, &b, &c, &d, &e, &f) == 6) {
  258. family = AF_LLC;
  259. len = 6;
  260. buf[0] = (unsigned char) a;
  261. buf[1] = (unsigned char) b;
  262. buf[2] = (unsigned char) c;
  263. buf[3] = (unsigned char) d;
  264. buf[4] = (unsigned char) e;
  265. buf[5] = (unsigned char) f;
  266. goto prefix;
  267. }
  268. if (hint == AF_LLC) {
  269. err = nl_error(EINVAL, "Invalid link layer address");
  270. goto errout;
  271. }
  272. }
  273. if ((hint == AF_DECnet || hint == AF_UNSPEC) &&
  274. (strchr(str, '.') || strchr(str, ','))) {
  275. if (dnet_pton(str, buf) > 0) {
  276. family = AF_DECnet;
  277. len = 2;
  278. goto prefix;
  279. }
  280. if (hint == AF_DECnet) {
  281. err = nl_error(EINVAL, "Invalid DECnet address");
  282. goto errout;
  283. }
  284. }
  285. if (hint == AF_UNSPEC && strchr(str, ':')) {
  286. int i = 0;
  287. char *s = str, *p;
  288. for (;;) {
  289. long l = strtol(s, &p, 16);
  290. if (s == p || l > 0xff || i >= sizeof(buf)) {
  291. err = -EINVAL;
  292. goto errout;
  293. }
  294. buf[i++] = (unsigned char) l;
  295. if (*p == '\0')
  296. break;
  297. s = ++p;
  298. }
  299. len = i;
  300. family = AF_UNSPEC;
  301. goto prefix;
  302. }
  303. err = nl_error(EINVAL, "Invalid address");
  304. goto errout;
  305. prefix:
  306. addr = nl_addr_alloc(len);
  307. if (!addr) {
  308. err = nl_errno(ENOMEM);
  309. goto errout;
  310. }
  311. nl_addr_set_family(addr, family);
  312. if (copy)
  313. nl_addr_set_binary_addr(addr, buf, len);
  314. if (prefix) {
  315. char *p;
  316. long pl = strtol(++prefix, &p, 0);
  317. if (p == prefix) {
  318. nl_addr_destroy(addr);
  319. err = -EINVAL;
  320. goto errout;
  321. }
  322. nl_addr_set_prefixlen(addr, pl);
  323. } else
  324. nl_addr_set_prefixlen(addr, len * 8);
  325. err = 0;
  326. errout:
  327. free(str);
  328. return err ? NULL : addr;
  329. }
  330. /**
  331. * Clone existing abstract address object.
  332. * @arg addr Abstract address object.
  333. * @return Newly allocated abstract address object being a duplicate of the
  334. * specified address object or NULL if a failure occured.
  335. */
  336. struct nl_addr *nl_addr_clone(struct nl_addr *addr)
  337. {
  338. struct nl_addr *new;
  339. new = nl_addr_build(addr->a_family, addr->a_addr, addr->a_len);
  340. if (new)
  341. new->a_prefixlen = addr->a_prefixlen;
  342. return new;
  343. }
  344. /** @} */
  345. /**
  346. * @name Destroying Abstract Addresses
  347. * @{
  348. */
  349. /**
  350. * Destroy abstract address object.
  351. * @arg addr Abstract address object.
  352. */
  353. void nl_addr_destroy(struct nl_addr *addr)
  354. {
  355. if (!addr)
  356. return;
  357. if (addr->a_refcnt != 1)
  358. BUG();
  359. free(addr);
  360. }
  361. /** @} */
  362. /**
  363. * @name Managing Usage References
  364. * @{
  365. */
  366. struct nl_addr *nl_addr_get(struct nl_addr *addr)
  367. {
  368. addr->a_refcnt++;
  369. return addr;
  370. }
  371. void nl_addr_put(struct nl_addr *addr)
  372. {
  373. if (!addr)
  374. return;
  375. if (addr->a_refcnt == 1)
  376. nl_addr_destroy(addr);
  377. else
  378. addr->a_refcnt--;
  379. }
  380. /**
  381. * Check whether an abstract address object is shared.
  382. * @arg addr Abstract address object.
  383. * @return Non-zero if the abstract address object is shared, otherwise 0.
  384. */
  385. int nl_addr_shared(struct nl_addr *addr)
  386. {
  387. return addr->a_refcnt > 1;
  388. }
  389. /** @} */
  390. /**
  391. * @name Miscellaneous
  392. * @{
  393. */
  394. /**
  395. * Compares two abstract address objects.
  396. * @arg a A abstract address object.
  397. * @arg b Another abstract address object.
  398. *
  399. * @return Integer less than, equal to or greather than zero if \c is found,
  400. * respectively to be less than, to, or be greater than \c b.
  401. */
  402. int nl_addr_cmp(struct nl_addr *a, struct nl_addr *b)
  403. {
  404. int d = a->a_family - b->a_family;
  405. if (d == 0) {
  406. d = a->a_len - b->a_len;
  407. if (a->a_len && d == 0)
  408. return memcmp(a->a_addr, b->a_addr, a->a_len);
  409. }
  410. return d;
  411. }
  412. /**
  413. * Compares the prefix of two abstract address objects.
  414. * @arg a A abstract address object.
  415. * @arg b Another abstract address object.
  416. *
  417. * @return Integer less than, equal to or greather than zero if \c is found,
  418. * respectively to be less than, to, or be greater than \c b.
  419. */
  420. int nl_addr_cmp_prefix(struct nl_addr *a, struct nl_addr *b)
  421. {
  422. int d = a->a_family - b->a_family;
  423. if (d == 0) {
  424. int len = min(a->a_prefixlen, b->a_prefixlen);
  425. int bytes = len / 8;
  426. d = memcmp(a->a_addr, b->a_addr, bytes);
  427. if (d == 0) {
  428. int mask = (1UL << (len % 8)) - 1UL;
  429. d = (a->a_addr[bytes] & mask) -
  430. (b->a_addr[bytes] & mask);
  431. }
  432. }
  433. return d;
  434. }
  435. /**
  436. * Returns true if the address consists of all zeros
  437. * @arg addr Address to look at.
  438. */
  439. int nl_addr_iszero(struct nl_addr *addr)
  440. {
  441. int i;
  442. for (i = 0; i < addr->a_len; i++)
  443. if (addr->a_addr[i])
  444. return 0;
  445. return 1;
  446. }
  447. /**
  448. * Check if an address matches a certain family.
  449. * @arg addr Address represented as character string.
  450. * @arg family Desired address family.
  451. *
  452. * @return 1 if the address is of the desired address family,
  453. * otherwise 0 is returned.
  454. */
  455. int nl_addr_valid(char *addr, int family)
  456. {
  457. int ret;
  458. char buf[32];
  459. switch (family) {
  460. case AF_INET:
  461. case AF_INET6:
  462. ret = inet_pton(family, addr, buf);
  463. if (ret <= 0)
  464. return 0;
  465. break;
  466. case AF_DECnet:
  467. ret = dnet_pton(addr, buf);
  468. if (ret <= 0)
  469. return 0;
  470. break;
  471. case AF_LLC:
  472. if (sscanf(addr, "%*02x:%*02x:%*02x:%*02x:%*02x:%*02x") != 6)
  473. return 0;
  474. break;
  475. }
  476. return 1;
  477. }
  478. /**
  479. * Guess address family of an abstract address object based on address size.
  480. * @arg addr Abstract address object.
  481. * @return Address family or AF_UNSPEC if guessing wasn't successful.
  482. */
  483. int nl_addr_guess_family(struct nl_addr *addr)
  484. {
  485. switch (addr->a_len) {
  486. case 4:
  487. return AF_INET;
  488. case 6:
  489. return AF_LLC;
  490. case 16:
  491. return AF_INET6;
  492. default:
  493. return AF_UNSPEC;
  494. }
  495. }
  496. /**
  497. * Fill out sockaddr structure with values from abstract address object.
  498. * @arg addr Abstract address object.
  499. * @arg sa Destination sockaddr structure buffer.
  500. * @arg salen Length of sockaddr structure buffer.
  501. *
  502. * Fills out the specified sockaddr structure with the data found in the
  503. * specified abstract address. The salen argument needs to be set to the
  504. * size of sa but will be modified to the actual size used during before
  505. * the function exits.
  506. *
  507. * @return 0 on success or a negative error code
  508. */
  509. int nl_addr_fill_sockaddr(struct nl_addr *addr, struct sockaddr *sa,
  510. socklen_t *salen)
  511. {
  512. switch (addr->a_family) {
  513. case AF_INET: {
  514. struct sockaddr_in *sai = (struct sockaddr_in *) sa;
  515. if (*salen < sizeof(*sai))
  516. return -EINVAL;
  517. sai->sin_family = addr->a_family;
  518. memcpy(&sai->sin_addr, addr->a_addr, 4);
  519. *salen = sizeof(*sai);
  520. }
  521. break;
  522. case AF_INET6: {
  523. struct sockaddr_in6 *sa6 = (struct sockaddr_in6 *) sa;
  524. if (*salen < sizeof(*sa6))
  525. return -EINVAL;
  526. sa6->sin6_family = addr->a_family;
  527. memcpy(&sa6->sin6_addr, addr->a_addr, 16);
  528. *salen = sizeof(*sa6);
  529. }
  530. break;
  531. default:
  532. return -EINVAL;
  533. }
  534. return 0;
  535. }
  536. /** @} */
  537. /**
  538. * @name Getting Information About Addresses
  539. * @{
  540. */
  541. /**
  542. * Call getaddrinfo() for an abstract address object.
  543. * @arg addr Abstract address object.
  544. *
  545. * Calls getaddrinfo() for the specified abstract address in AI_NUMERICHOST
  546. * mode.
  547. *
  548. * @note The caller is responsible for freeing the linked list using the
  549. * interface provided by getaddrinfo(3).
  550. *
  551. * @return A linked list of addrinfo handles or NULL with an error message
  552. * associated.
  553. */
  554. struct addrinfo *nl_addr_info(struct nl_addr *addr)
  555. {
  556. int err;
  557. struct addrinfo *res;
  558. char buf[INET6_ADDRSTRLEN+5];
  559. struct addrinfo hint = {
  560. .ai_flags = AI_NUMERICHOST,
  561. .ai_family = addr->a_family,
  562. };
  563. nl_addr2str(addr, buf, sizeof(buf));
  564. err = getaddrinfo(buf, NULL, &hint, &res);
  565. if (err != 0) {
  566. nl_error(err, gai_strerror(err));
  567. return NULL;
  568. }
  569. return res;
  570. }
  571. /**
  572. * Resolve abstract address object to a name using getnameinfo().
  573. * @arg addr Abstract address object.
  574. * @arg host Destination buffer for host name.
  575. * @arg hostlen Length of destination buffer.
  576. *
  577. * Resolves the abstract address to a name and writes the looked up result
  578. * into the host buffer. getnameinfo() is used to perform the lookup and
  579. * is put into NI_NAMEREQD mode so the function will fail if the lookup
  580. * couldn't be performed.
  581. *
  582. * @return 0 on success or a negative error code.
  583. */
  584. int nl_addr_resolve(struct nl_addr *addr, char *host, size_t hostlen)
  585. {
  586. int err;
  587. struct sockaddr_in6 buf;
  588. socklen_t salen = sizeof(buf);
  589. err = nl_addr_fill_sockaddr(addr, (struct sockaddr *) &buf, &salen);
  590. if (err < 0)
  591. return err;
  592. return getnameinfo((struct sockaddr *) &buf, salen,
  593. host, hostlen, NULL, 0, NI_NAMEREQD);
  594. }
  595. /** @} */
  596. /**
  597. * @name Attributes
  598. * @{
  599. */
  600. void nl_addr_set_family(struct nl_addr *addr, int family)
  601. {
  602. addr->a_family = family;
  603. }
  604. int nl_addr_get_family(struct nl_addr *addr)
  605. {
  606. return addr->a_family;
  607. }
  608. /**
  609. * Set binary address of abstract address object.
  610. * @arg addr Abstract address object.
  611. * @arg buf Buffer containing binary address.
  612. * @arg len Length of buffer containing binary address.
  613. */
  614. int nl_addr_set_binary_addr(struct nl_addr *addr, void *buf, size_t len)
  615. {
  616. if (len > addr->a_maxsize)
  617. return -ERANGE;
  618. addr->a_len = len;
  619. memcpy(addr->a_addr, buf, len);
  620. return 0;
  621. }
  622. /**
  623. * Get binary address of abstract address object.
  624. * @arg addr Abstract address object.
  625. */
  626. void *nl_addr_get_binary_addr(struct nl_addr *addr)
  627. {
  628. return addr->a_addr;
  629. }
  630. /**
  631. * Get length of binary address of abstract address object.
  632. * @arg addr Abstract address object.
  633. */
  634. unsigned int nl_addr_get_len(struct nl_addr *addr)
  635. {
  636. return addr->a_len;
  637. }
  638. void nl_addr_set_prefixlen(struct nl_addr *addr, int prefixlen)
  639. {
  640. addr->a_prefixlen = prefixlen;
  641. }
  642. /**
  643. * Get prefix length of abstract address object.
  644. * @arg addr Abstract address object.
  645. */
  646. unsigned int nl_addr_get_prefixlen(struct nl_addr *addr)
  647. {
  648. return addr->a_prefixlen;
  649. }
  650. /** @} */
  651. /**
  652. * @name Translations to Strings
  653. * @{
  654. */
  655. /**
  656. * Convert abstract address object to character string.
  657. * @arg addr Abstract address object.
  658. * @arg buf Destination buffer.
  659. * @arg size Size of destination buffer.
  660. *
  661. * Converts an abstract address to a character string and stores
  662. * the result in the specified destination buffer.
  663. *
  664. * @return Address represented in ASCII stored in destination buffer.
  665. */
  666. char *nl_addr2str(struct nl_addr *addr, char *buf, size_t size)
  667. {
  668. int i;
  669. char tmp[16];
  670. if (!addr->a_len) {
  671. snprintf(buf, size, "none");
  672. goto prefix;
  673. }
  674. switch (addr->a_family) {
  675. case AF_INET:
  676. inet_ntop(AF_INET, addr->a_addr, buf, size);
  677. break;
  678. case AF_INET6:
  679. inet_ntop(AF_INET6, addr->a_addr, buf, size);
  680. break;
  681. case AF_DECnet:
  682. dnet_ntop(addr->a_addr, addr->a_len, buf, size);
  683. break;
  684. case AF_LLC:
  685. default:
  686. snprintf(buf, size, "%02x",
  687. (unsigned char) addr->a_addr[0]);
  688. for (i = 1; i < addr->a_len; i++) {
  689. snprintf(tmp, sizeof(tmp), ":%02x",
  690. (unsigned char) addr->a_addr[i]);
  691. strncat(buf, tmp, size - strlen(buf) - 1);
  692. }
  693. break;
  694. }
  695. prefix:
  696. if (addr->a_prefixlen != (8 * addr->a_len)) {
  697. snprintf(tmp, sizeof(tmp), "/%u", addr->a_prefixlen);
  698. strncat(buf, tmp, size - strlen(buf) - 1);
  699. }
  700. return buf;
  701. }
  702. /** @} */
  703. /**
  704. * @name Address Family Transformations
  705. * @{
  706. */
  707. static struct trans_tbl afs[] = {
  708. __ADD(AF_UNSPEC,unspec)
  709. __ADD(AF_UNIX,unix)
  710. __ADD(AF_LOCAL,local)
  711. __ADD(AF_INET,inet)
  712. __ADD(AF_AX25,ax25)
  713. __ADD(AF_IPX,ipx)
  714. __ADD(AF_APPLETALK,appletalk)
  715. __ADD(AF_NETROM,netrom)
  716. __ADD(AF_BRIDGE,bridge)
  717. __ADD(AF_ATMPVC,atmpvc)
  718. __ADD(AF_X25,x25)
  719. __ADD(AF_INET6,inet6)
  720. __ADD(AF_ROSE,rose)
  721. __ADD(AF_DECnet,decnet)
  722. __ADD(AF_NETBEUI,netbeui)
  723. __ADD(AF_SECURITY,security)
  724. __ADD(AF_KEY,key)
  725. __ADD(AF_NETLINK,netlink)
  726. __ADD(AF_ROUTE,route)
  727. __ADD(AF_PACKET,packet)
  728. __ADD(AF_ASH,ash)
  729. __ADD(AF_ECONET,econet)
  730. __ADD(AF_ATMSVC,atmsvc)
  731. __ADD(AF_SNA,sna)
  732. __ADD(AF_IRDA,irda)
  733. __ADD(AF_PPPOX,pppox)
  734. __ADD(AF_WANPIPE,wanpipe)
  735. __ADD(AF_LLC,llc)
  736. __ADD(AF_BLUETOOTH,bluetooth)
  737. };
  738. char *nl_af2str(int family, char *buf, size_t size)
  739. {
  740. return __type2str(family, buf, size, afs, ARRAY_SIZE(afs));
  741. }
  742. int nl_str2af(const char *name)
  743. {
  744. int fam = __str2type(name, afs, ARRAY_SIZE(afs));
  745. return fam >= 0 ? fam : AF_UNSPEC;
  746. }
  747. /** @} */
  748. /** @} */