/contrib/bind9/lib/dns/rdata/generic/mx_15.c

https://bitbucket.org/freebsd/freebsd-head/ · C · 324 lines · 234 code · 72 blank · 18 comment · 56 complexity · 98ebf72bb07c6a5723aaabbd70b01c4c MD5 · raw file

  1. /*
  2. * Copyright (C) 2004, 2005, 2007, 2009 Internet Systems Consortium, Inc. ("ISC")
  3. * Copyright (C) 1998-2001, 2003 Internet Software Consortium.
  4. *
  5. * Permission to use, copy, modify, and/or distribute this software for any
  6. * purpose with or without fee is hereby granted, provided that the above
  7. * copyright notice and this permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  10. * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  11. * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  12. * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  13. * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  14. * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  15. * PERFORMANCE OF THIS SOFTWARE.
  16. */
  17. /* $Id: mx_15.c,v 1.58 2009/12/04 22:06:37 tbox Exp $ */
  18. /* reviewed: Wed Mar 15 18:05:46 PST 2000 by brister */
  19. #ifndef RDATA_GENERIC_MX_15_C
  20. #define RDATA_GENERIC_MX_15_C
  21. #include <string.h>
  22. #include <isc/net.h>
  23. #define RRTYPE_MX_ATTRIBUTES (0)
  24. static isc_boolean_t
  25. check_mx(isc_token_t *token) {
  26. char tmp[sizeof("xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:123.123.123.123.")];
  27. struct in_addr addr;
  28. struct in6_addr addr6;
  29. if (strlcpy(tmp, DNS_AS_STR(*token), sizeof(tmp)) >= sizeof(tmp))
  30. return (ISC_TRUE);
  31. if (tmp[strlen(tmp) - 1] == '.')
  32. tmp[strlen(tmp) - 1] = '\0';
  33. if (inet_aton(tmp, &addr) == 1 ||
  34. inet_pton(AF_INET6, tmp, &addr6) == 1)
  35. return (ISC_FALSE);
  36. return (ISC_TRUE);
  37. }
  38. static inline isc_result_t
  39. fromtext_mx(ARGS_FROMTEXT) {
  40. isc_token_t token;
  41. dns_name_t name;
  42. isc_buffer_t buffer;
  43. isc_boolean_t ok;
  44. REQUIRE(type == 15);
  45. UNUSED(type);
  46. UNUSED(rdclass);
  47. UNUSED(callbacks);
  48. RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_number,
  49. ISC_FALSE));
  50. if (token.value.as_ulong > 0xffffU)
  51. RETTOK(ISC_R_RANGE);
  52. RETERR(uint16_tobuffer(token.value.as_ulong, target));
  53. RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
  54. ISC_FALSE));
  55. ok = ISC_TRUE;
  56. if ((options & DNS_RDATA_CHECKMX) != 0)
  57. ok = check_mx(&token);
  58. if (!ok && (options & DNS_RDATA_CHECKMXFAIL) != 0)
  59. RETTOK(DNS_R_MXISADDRESS);
  60. if (!ok && callbacks != NULL)
  61. warn_badmx(&token, lexer, callbacks);
  62. dns_name_init(&name, NULL);
  63. buffer_fromregion(&buffer, &token.value.as_region);
  64. origin = (origin != NULL) ? origin : dns_rootname;
  65. RETTOK(dns_name_fromtext(&name, &buffer, origin, options, target));
  66. ok = ISC_TRUE;
  67. if ((options & DNS_RDATA_CHECKNAMES) != 0)
  68. ok = dns_name_ishostname(&name, ISC_FALSE);
  69. if (!ok && (options & DNS_RDATA_CHECKNAMESFAIL) != 0)
  70. RETTOK(DNS_R_BADNAME);
  71. if (!ok && callbacks != NULL)
  72. warn_badname(&name, lexer, callbacks);
  73. return (ISC_R_SUCCESS);
  74. }
  75. static inline isc_result_t
  76. totext_mx(ARGS_TOTEXT) {
  77. isc_region_t region;
  78. dns_name_t name;
  79. dns_name_t prefix;
  80. isc_boolean_t sub;
  81. char buf[sizeof("64000")];
  82. unsigned short num;
  83. REQUIRE(rdata->type == 15);
  84. REQUIRE(rdata->length != 0);
  85. dns_name_init(&name, NULL);
  86. dns_name_init(&prefix, NULL);
  87. dns_rdata_toregion(rdata, &region);
  88. num = uint16_fromregion(&region);
  89. isc_region_consume(&region, 2);
  90. sprintf(buf, "%u", num);
  91. RETERR(str_totext(buf, target));
  92. RETERR(str_totext(" ", target));
  93. dns_name_fromregion(&name, &region);
  94. sub = name_prefix(&name, tctx->origin, &prefix);
  95. return (dns_name_totext(&prefix, sub, target));
  96. }
  97. static inline isc_result_t
  98. fromwire_mx(ARGS_FROMWIRE) {
  99. dns_name_t name;
  100. isc_region_t sregion;
  101. REQUIRE(type == 15);
  102. UNUSED(type);
  103. UNUSED(rdclass);
  104. dns_decompress_setmethods(dctx, DNS_COMPRESS_GLOBAL14);
  105. dns_name_init(&name, NULL);
  106. isc_buffer_activeregion(source, &sregion);
  107. if (sregion.length < 2)
  108. return (ISC_R_UNEXPECTEDEND);
  109. RETERR(mem_tobuffer(target, sregion.base, 2));
  110. isc_buffer_forward(source, 2);
  111. return (dns_name_fromwire(&name, source, dctx, options, target));
  112. }
  113. static inline isc_result_t
  114. towire_mx(ARGS_TOWIRE) {
  115. dns_name_t name;
  116. dns_offsets_t offsets;
  117. isc_region_t region;
  118. REQUIRE(rdata->type == 15);
  119. REQUIRE(rdata->length != 0);
  120. dns_compress_setmethods(cctx, DNS_COMPRESS_GLOBAL14);
  121. dns_rdata_toregion(rdata, &region);
  122. RETERR(mem_tobuffer(target, region.base, 2));
  123. isc_region_consume(&region, 2);
  124. dns_name_init(&name, offsets);
  125. dns_name_fromregion(&name, &region);
  126. return (dns_name_towire(&name, cctx, target));
  127. }
  128. static inline int
  129. compare_mx(ARGS_COMPARE) {
  130. dns_name_t name1;
  131. dns_name_t name2;
  132. isc_region_t region1;
  133. isc_region_t region2;
  134. int order;
  135. REQUIRE(rdata1->type == rdata2->type);
  136. REQUIRE(rdata1->rdclass == rdata2->rdclass);
  137. REQUIRE(rdata1->type == 15);
  138. REQUIRE(rdata1->length != 0);
  139. REQUIRE(rdata2->length != 0);
  140. order = memcmp(rdata1->data, rdata2->data, 2);
  141. if (order != 0)
  142. return (order < 0 ? -1 : 1);
  143. dns_name_init(&name1, NULL);
  144. dns_name_init(&name2, NULL);
  145. dns_rdata_toregion(rdata1, &region1);
  146. dns_rdata_toregion(rdata2, &region2);
  147. isc_region_consume(&region1, 2);
  148. isc_region_consume(&region2, 2);
  149. dns_name_fromregion(&name1, &region1);
  150. dns_name_fromregion(&name2, &region2);
  151. return (dns_name_rdatacompare(&name1, &name2));
  152. }
  153. static inline isc_result_t
  154. fromstruct_mx(ARGS_FROMSTRUCT) {
  155. dns_rdata_mx_t *mx = source;
  156. isc_region_t region;
  157. REQUIRE(type == 15);
  158. REQUIRE(source != NULL);
  159. REQUIRE(mx->common.rdtype == type);
  160. REQUIRE(mx->common.rdclass == rdclass);
  161. UNUSED(type);
  162. UNUSED(rdclass);
  163. RETERR(uint16_tobuffer(mx->pref, target));
  164. dns_name_toregion(&mx->mx, &region);
  165. return (isc_buffer_copyregion(target, &region));
  166. }
  167. static inline isc_result_t
  168. tostruct_mx(ARGS_TOSTRUCT) {
  169. isc_region_t region;
  170. dns_rdata_mx_t *mx = target;
  171. dns_name_t name;
  172. REQUIRE(rdata->type == 15);
  173. REQUIRE(target != NULL);
  174. REQUIRE(rdata->length != 0);
  175. mx->common.rdclass = rdata->rdclass;
  176. mx->common.rdtype = rdata->type;
  177. ISC_LINK_INIT(&mx->common, link);
  178. dns_name_init(&name, NULL);
  179. dns_rdata_toregion(rdata, &region);
  180. mx->pref = uint16_fromregion(&region);
  181. isc_region_consume(&region, 2);
  182. dns_name_fromregion(&name, &region);
  183. dns_name_init(&mx->mx, NULL);
  184. RETERR(name_duporclone(&name, mctx, &mx->mx));
  185. mx->mctx = mctx;
  186. return (ISC_R_SUCCESS);
  187. }
  188. static inline void
  189. freestruct_mx(ARGS_FREESTRUCT) {
  190. dns_rdata_mx_t *mx = source;
  191. REQUIRE(source != NULL);
  192. REQUIRE(mx->common.rdtype == 15);
  193. if (mx->mctx == NULL)
  194. return;
  195. dns_name_free(&mx->mx, mx->mctx);
  196. mx->mctx = NULL;
  197. }
  198. static inline isc_result_t
  199. additionaldata_mx(ARGS_ADDLDATA) {
  200. dns_name_t name;
  201. dns_offsets_t offsets;
  202. isc_region_t region;
  203. REQUIRE(rdata->type == 15);
  204. dns_name_init(&name, offsets);
  205. dns_rdata_toregion(rdata, &region);
  206. isc_region_consume(&region, 2);
  207. dns_name_fromregion(&name, &region);
  208. return ((add)(arg, &name, dns_rdatatype_a));
  209. }
  210. static inline isc_result_t
  211. digest_mx(ARGS_DIGEST) {
  212. isc_region_t r1, r2;
  213. dns_name_t name;
  214. REQUIRE(rdata->type == 15);
  215. dns_rdata_toregion(rdata, &r1);
  216. r2 = r1;
  217. isc_region_consume(&r2, 2);
  218. r1.length = 2;
  219. RETERR((digest)(arg, &r1));
  220. dns_name_init(&name, NULL);
  221. dns_name_fromregion(&name, &r2);
  222. return (dns_name_digest(&name, digest, arg));
  223. }
  224. static inline isc_boolean_t
  225. checkowner_mx(ARGS_CHECKOWNER) {
  226. REQUIRE(type == 15);
  227. UNUSED(type);
  228. UNUSED(rdclass);
  229. return (dns_name_ishostname(name, wildcard));
  230. }
  231. static inline isc_boolean_t
  232. checknames_mx(ARGS_CHECKNAMES) {
  233. isc_region_t region;
  234. dns_name_t name;
  235. REQUIRE(rdata->type == 15);
  236. UNUSED(owner);
  237. dns_rdata_toregion(rdata, &region);
  238. isc_region_consume(&region, 2);
  239. dns_name_init(&name, NULL);
  240. dns_name_fromregion(&name, &region);
  241. if (!dns_name_ishostname(&name, ISC_FALSE)) {
  242. if (bad != NULL)
  243. dns_name_clone(&name, bad);
  244. return (ISC_FALSE);
  245. }
  246. return (ISC_TRUE);
  247. }
  248. static inline int
  249. casecompare_mx(ARGS_COMPARE) {
  250. return (compare_mx(rdata1, rdata2));
  251. }
  252. #endif /* RDATA_GENERIC_MX_15_C */