msg_helpers.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  1. package dns
  2. import (
  3. "encoding/base32"
  4. "encoding/base64"
  5. "encoding/binary"
  6. "encoding/hex"
  7. "net"
  8. "strings"
  9. )
  10. // helper functions called from the generated zmsg.go
  11. // These function are named after the tag to help pack/unpack, if there is no tag it is the name
  12. // of the type they pack/unpack (string, int, etc). We prefix all with unpackData or packData, so packDataA or
  13. // packDataDomainName.
  14. func unpackDataA(msg []byte, off int) (net.IP, int, error) {
  15. if off+net.IPv4len > len(msg) {
  16. return nil, len(msg), &Error{err: "overflow unpacking a"}
  17. }
  18. a := append(make(net.IP, 0, net.IPv4len), msg[off:off+net.IPv4len]...)
  19. off += net.IPv4len
  20. return a, off, nil
  21. }
  22. func packDataA(a net.IP, msg []byte, off int) (int, error) {
  23. // It must be a slice of 4, even if it is 16, we encode only the first 4
  24. if off+net.IPv4len > len(msg) {
  25. return len(msg), &Error{err: "overflow packing a"}
  26. }
  27. switch len(a) {
  28. case net.IPv4len, net.IPv6len:
  29. copy(msg[off:], a.To4())
  30. off += net.IPv4len
  31. case 0:
  32. // Allowed, for dynamic updates.
  33. default:
  34. return len(msg), &Error{err: "overflow packing a"}
  35. }
  36. return off, nil
  37. }
  38. func unpackDataAAAA(msg []byte, off int) (net.IP, int, error) {
  39. if off+net.IPv6len > len(msg) {
  40. return nil, len(msg), &Error{err: "overflow unpacking aaaa"}
  41. }
  42. aaaa := append(make(net.IP, 0, net.IPv6len), msg[off:off+net.IPv6len]...)
  43. off += net.IPv6len
  44. return aaaa, off, nil
  45. }
  46. func packDataAAAA(aaaa net.IP, msg []byte, off int) (int, error) {
  47. if off+net.IPv6len > len(msg) {
  48. return len(msg), &Error{err: "overflow packing aaaa"}
  49. }
  50. switch len(aaaa) {
  51. case net.IPv6len:
  52. copy(msg[off:], aaaa)
  53. off += net.IPv6len
  54. case 0:
  55. // Allowed, dynamic updates.
  56. default:
  57. return len(msg), &Error{err: "overflow packing aaaa"}
  58. }
  59. return off, nil
  60. }
  61. // unpackHeader unpacks an RR header, returning the offset to the end of the header and a
  62. // re-sliced msg according to the expected length of the RR.
  63. func unpackHeader(msg []byte, off int) (rr RR_Header, off1 int, truncmsg []byte, err error) {
  64. hdr := RR_Header{}
  65. if off == len(msg) {
  66. return hdr, off, msg, nil
  67. }
  68. hdr.Name, off, err = UnpackDomainName(msg, off)
  69. if err != nil {
  70. return hdr, len(msg), msg, err
  71. }
  72. hdr.Rrtype, off, err = unpackUint16(msg, off)
  73. if err != nil {
  74. return hdr, len(msg), msg, err
  75. }
  76. hdr.Class, off, err = unpackUint16(msg, off)
  77. if err != nil {
  78. return hdr, len(msg), msg, err
  79. }
  80. hdr.Ttl, off, err = unpackUint32(msg, off)
  81. if err != nil {
  82. return hdr, len(msg), msg, err
  83. }
  84. hdr.Rdlength, off, err = unpackUint16(msg, off)
  85. if err != nil {
  86. return hdr, len(msg), msg, err
  87. }
  88. msg, err = truncateMsgFromRdlength(msg, off, hdr.Rdlength)
  89. return hdr, off, msg, err
  90. }
  91. // packHeader packs an RR header, returning the offset to the end of the header.
  92. // See PackDomainName for documentation about the compression.
  93. func (hdr RR_Header) packHeader(msg []byte, off int, compression compressionMap, compress bool) (int, error) {
  94. if off == len(msg) {
  95. return off, nil
  96. }
  97. off, err := packDomainName(hdr.Name, msg, off, compression, compress)
  98. if err != nil {
  99. return len(msg), err
  100. }
  101. off, err = packUint16(hdr.Rrtype, msg, off)
  102. if err != nil {
  103. return len(msg), err
  104. }
  105. off, err = packUint16(hdr.Class, msg, off)
  106. if err != nil {
  107. return len(msg), err
  108. }
  109. off, err = packUint32(hdr.Ttl, msg, off)
  110. if err != nil {
  111. return len(msg), err
  112. }
  113. off, err = packUint16(0, msg, off) // The RDLENGTH field will be set later in packRR.
  114. if err != nil {
  115. return len(msg), err
  116. }
  117. return off, nil
  118. }
  119. // helper helper functions.
  120. // truncateMsgFromRdLength truncates msg to match the expected length of the RR.
  121. // Returns an error if msg is smaller than the expected size.
  122. func truncateMsgFromRdlength(msg []byte, off int, rdlength uint16) (truncmsg []byte, err error) {
  123. lenrd := off + int(rdlength)
  124. if lenrd > len(msg) {
  125. return msg, &Error{err: "overflowing header size"}
  126. }
  127. return msg[:lenrd], nil
  128. }
  129. var base32HexNoPadEncoding = base32.HexEncoding.WithPadding(base32.NoPadding)
  130. func fromBase32(s []byte) (buf []byte, err error) {
  131. for i, b := range s {
  132. if b >= 'a' && b <= 'z' {
  133. s[i] = b - 32
  134. }
  135. }
  136. buflen := base32HexNoPadEncoding.DecodedLen(len(s))
  137. buf = make([]byte, buflen)
  138. n, err := base32HexNoPadEncoding.Decode(buf, s)
  139. buf = buf[:n]
  140. return
  141. }
  142. func toBase32(b []byte) string {
  143. return base32HexNoPadEncoding.EncodeToString(b)
  144. }
  145. func fromBase64(s []byte) (buf []byte, err error) {
  146. buflen := base64.StdEncoding.DecodedLen(len(s))
  147. buf = make([]byte, buflen)
  148. n, err := base64.StdEncoding.Decode(buf, s)
  149. buf = buf[:n]
  150. return
  151. }
  152. func toBase64(b []byte) string { return base64.StdEncoding.EncodeToString(b) }
  153. // dynamicUpdate returns true if the Rdlength is zero.
  154. func noRdata(h RR_Header) bool { return h.Rdlength == 0 }
  155. func unpackUint8(msg []byte, off int) (i uint8, off1 int, err error) {
  156. if off+1 > len(msg) {
  157. return 0, len(msg), &Error{err: "overflow unpacking uint8"}
  158. }
  159. return msg[off], off + 1, nil
  160. }
  161. func packUint8(i uint8, msg []byte, off int) (off1 int, err error) {
  162. if off+1 > len(msg) {
  163. return len(msg), &Error{err: "overflow packing uint8"}
  164. }
  165. msg[off] = i
  166. return off + 1, nil
  167. }
  168. func unpackUint16(msg []byte, off int) (i uint16, off1 int, err error) {
  169. if off+2 > len(msg) {
  170. return 0, len(msg), &Error{err: "overflow unpacking uint16"}
  171. }
  172. return binary.BigEndian.Uint16(msg[off:]), off + 2, nil
  173. }
  174. func packUint16(i uint16, msg []byte, off int) (off1 int, err error) {
  175. if off+2 > len(msg) {
  176. return len(msg), &Error{err: "overflow packing uint16"}
  177. }
  178. binary.BigEndian.PutUint16(msg[off:], i)
  179. return off + 2, nil
  180. }
  181. func unpackUint32(msg []byte, off int) (i uint32, off1 int, err error) {
  182. if off+4 > len(msg) {
  183. return 0, len(msg), &Error{err: "overflow unpacking uint32"}
  184. }
  185. return binary.BigEndian.Uint32(msg[off:]), off + 4, nil
  186. }
  187. func packUint32(i uint32, msg []byte, off int) (off1 int, err error) {
  188. if off+4 > len(msg) {
  189. return len(msg), &Error{err: "overflow packing uint32"}
  190. }
  191. binary.BigEndian.PutUint32(msg[off:], i)
  192. return off + 4, nil
  193. }
  194. func unpackUint48(msg []byte, off int) (i uint64, off1 int, err error) {
  195. if off+6 > len(msg) {
  196. return 0, len(msg), &Error{err: "overflow unpacking uint64 as uint48"}
  197. }
  198. // Used in TSIG where the last 48 bits are occupied, so for now, assume a uint48 (6 bytes)
  199. i = uint64(msg[off])<<40 | uint64(msg[off+1])<<32 | uint64(msg[off+2])<<24 | uint64(msg[off+3])<<16 |
  200. uint64(msg[off+4])<<8 | uint64(msg[off+5])
  201. off += 6
  202. return i, off, nil
  203. }
  204. func packUint48(i uint64, msg []byte, off int) (off1 int, err error) {
  205. if off+6 > len(msg) {
  206. return len(msg), &Error{err: "overflow packing uint64 as uint48"}
  207. }
  208. msg[off] = byte(i >> 40)
  209. msg[off+1] = byte(i >> 32)
  210. msg[off+2] = byte(i >> 24)
  211. msg[off+3] = byte(i >> 16)
  212. msg[off+4] = byte(i >> 8)
  213. msg[off+5] = byte(i)
  214. off += 6
  215. return off, nil
  216. }
  217. func unpackUint64(msg []byte, off int) (i uint64, off1 int, err error) {
  218. if off+8 > len(msg) {
  219. return 0, len(msg), &Error{err: "overflow unpacking uint64"}
  220. }
  221. return binary.BigEndian.Uint64(msg[off:]), off + 8, nil
  222. }
  223. func packUint64(i uint64, msg []byte, off int) (off1 int, err error) {
  224. if off+8 > len(msg) {
  225. return len(msg), &Error{err: "overflow packing uint64"}
  226. }
  227. binary.BigEndian.PutUint64(msg[off:], i)
  228. off += 8
  229. return off, nil
  230. }
  231. func unpackString(msg []byte, off int) (string, int, error) {
  232. if off+1 > len(msg) {
  233. return "", off, &Error{err: "overflow unpacking txt"}
  234. }
  235. l := int(msg[off])
  236. if off+l+1 > len(msg) {
  237. return "", off, &Error{err: "overflow unpacking txt"}
  238. }
  239. var s strings.Builder
  240. s.Grow(l)
  241. for _, b := range msg[off+1 : off+1+l] {
  242. switch {
  243. case b == '"' || b == '\\':
  244. s.WriteByte('\\')
  245. s.WriteByte(b)
  246. case b < ' ' || b > '~': // unprintable
  247. s.WriteString(escapeByte(b))
  248. default:
  249. s.WriteByte(b)
  250. }
  251. }
  252. off += 1 + l
  253. return s.String(), off, nil
  254. }
  255. func packString(s string, msg []byte, off int) (int, error) {
  256. txtTmp := make([]byte, 256*4+1)
  257. off, err := packTxtString(s, msg, off, txtTmp)
  258. if err != nil {
  259. return len(msg), err
  260. }
  261. return off, nil
  262. }
  263. func unpackStringBase32(msg []byte, off, end int) (string, int, error) {
  264. if end > len(msg) {
  265. return "", len(msg), &Error{err: "overflow unpacking base32"}
  266. }
  267. s := toBase32(msg[off:end])
  268. return s, end, nil
  269. }
  270. func packStringBase32(s string, msg []byte, off int) (int, error) {
  271. b32, err := fromBase32([]byte(s))
  272. if err != nil {
  273. return len(msg), err
  274. }
  275. if off+len(b32) > len(msg) {
  276. return len(msg), &Error{err: "overflow packing base32"}
  277. }
  278. copy(msg[off:off+len(b32)], b32)
  279. off += len(b32)
  280. return off, nil
  281. }
  282. func unpackStringBase64(msg []byte, off, end int) (string, int, error) {
  283. // Rest of the RR is base64 encoded value, so we don't need an explicit length
  284. // to be set. Thus far all RR's that have base64 encoded fields have those as their
  285. // last one. What we do need is the end of the RR!
  286. if end > len(msg) {
  287. return "", len(msg), &Error{err: "overflow unpacking base64"}
  288. }
  289. s := toBase64(msg[off:end])
  290. return s, end, nil
  291. }
  292. func packStringBase64(s string, msg []byte, off int) (int, error) {
  293. b64, err := fromBase64([]byte(s))
  294. if err != nil {
  295. return len(msg), err
  296. }
  297. if off+len(b64) > len(msg) {
  298. return len(msg), &Error{err: "overflow packing base64"}
  299. }
  300. copy(msg[off:off+len(b64)], b64)
  301. off += len(b64)
  302. return off, nil
  303. }
  304. func unpackStringHex(msg []byte, off, end int) (string, int, error) {
  305. // Rest of the RR is hex encoded value, so we don't need an explicit length
  306. // to be set. NSEC and TSIG have hex fields with a length field.
  307. // What we do need is the end of the RR!
  308. if end > len(msg) {
  309. return "", len(msg), &Error{err: "overflow unpacking hex"}
  310. }
  311. s := hex.EncodeToString(msg[off:end])
  312. return s, end, nil
  313. }
  314. func packStringHex(s string, msg []byte, off int) (int, error) {
  315. h, err := hex.DecodeString(s)
  316. if err != nil {
  317. return len(msg), err
  318. }
  319. if off+len(h) > len(msg) {
  320. return len(msg), &Error{err: "overflow packing hex"}
  321. }
  322. copy(msg[off:off+len(h)], h)
  323. off += len(h)
  324. return off, nil
  325. }
  326. func unpackStringAny(msg []byte, off, end int) (string, int, error) {
  327. if end > len(msg) {
  328. return "", len(msg), &Error{err: "overflow unpacking anything"}
  329. }
  330. return string(msg[off:end]), end, nil
  331. }
  332. func packStringAny(s string, msg []byte, off int) (int, error) {
  333. if off+len(s) > len(msg) {
  334. return len(msg), &Error{err: "overflow packing anything"}
  335. }
  336. copy(msg[off:off+len(s)], s)
  337. off += len(s)
  338. return off, nil
  339. }
  340. func unpackStringTxt(msg []byte, off int) ([]string, int, error) {
  341. txt, off, err := unpackTxt(msg, off)
  342. if err != nil {
  343. return nil, len(msg), err
  344. }
  345. return txt, off, nil
  346. }
  347. func packStringTxt(s []string, msg []byte, off int) (int, error) {
  348. txtTmp := make([]byte, 256*4+1) // If the whole string consists out of \DDD we need this many.
  349. off, err := packTxt(s, msg, off, txtTmp)
  350. if err != nil {
  351. return len(msg), err
  352. }
  353. return off, nil
  354. }
  355. func unpackDataOpt(msg []byte, off int) ([]EDNS0, int, error) {
  356. var edns []EDNS0
  357. Option:
  358. var code uint16
  359. if off+4 > len(msg) {
  360. return nil, len(msg), &Error{err: "overflow unpacking opt"}
  361. }
  362. code = binary.BigEndian.Uint16(msg[off:])
  363. off += 2
  364. optlen := binary.BigEndian.Uint16(msg[off:])
  365. off += 2
  366. if off+int(optlen) > len(msg) {
  367. return nil, len(msg), &Error{err: "overflow unpacking opt"}
  368. }
  369. switch code {
  370. case EDNS0NSID:
  371. e := new(EDNS0_NSID)
  372. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  373. return nil, len(msg), err
  374. }
  375. edns = append(edns, e)
  376. off += int(optlen)
  377. case EDNS0SUBNET:
  378. e := new(EDNS0_SUBNET)
  379. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  380. return nil, len(msg), err
  381. }
  382. edns = append(edns, e)
  383. off += int(optlen)
  384. case EDNS0COOKIE:
  385. e := new(EDNS0_COOKIE)
  386. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  387. return nil, len(msg), err
  388. }
  389. edns = append(edns, e)
  390. off += int(optlen)
  391. case EDNS0UL:
  392. e := new(EDNS0_UL)
  393. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  394. return nil, len(msg), err
  395. }
  396. edns = append(edns, e)
  397. off += int(optlen)
  398. case EDNS0LLQ:
  399. e := new(EDNS0_LLQ)
  400. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  401. return nil, len(msg), err
  402. }
  403. edns = append(edns, e)
  404. off += int(optlen)
  405. case EDNS0DAU:
  406. e := new(EDNS0_DAU)
  407. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  408. return nil, len(msg), err
  409. }
  410. edns = append(edns, e)
  411. off += int(optlen)
  412. case EDNS0DHU:
  413. e := new(EDNS0_DHU)
  414. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  415. return nil, len(msg), err
  416. }
  417. edns = append(edns, e)
  418. off += int(optlen)
  419. case EDNS0N3U:
  420. e := new(EDNS0_N3U)
  421. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  422. return nil, len(msg), err
  423. }
  424. edns = append(edns, e)
  425. off += int(optlen)
  426. case EDNS0PADDING:
  427. e := new(EDNS0_PADDING)
  428. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  429. return nil, len(msg), err
  430. }
  431. edns = append(edns, e)
  432. off += int(optlen)
  433. default:
  434. e := new(EDNS0_LOCAL)
  435. e.Code = code
  436. if err := e.unpack(msg[off : off+int(optlen)]); err != nil {
  437. return nil, len(msg), err
  438. }
  439. edns = append(edns, e)
  440. off += int(optlen)
  441. }
  442. if off < len(msg) {
  443. goto Option
  444. }
  445. return edns, off, nil
  446. }
  447. func packDataOpt(options []EDNS0, msg []byte, off int) (int, error) {
  448. for _, el := range options {
  449. b, err := el.pack()
  450. if err != nil || off+3 > len(msg) {
  451. return len(msg), &Error{err: "overflow packing opt"}
  452. }
  453. binary.BigEndian.PutUint16(msg[off:], el.Option()) // Option code
  454. binary.BigEndian.PutUint16(msg[off+2:], uint16(len(b))) // Length
  455. off += 4
  456. if off+len(b) > len(msg) {
  457. copy(msg[off:], b)
  458. off = len(msg)
  459. continue
  460. }
  461. // Actual data
  462. copy(msg[off:off+len(b)], b)
  463. off += len(b)
  464. }
  465. return off, nil
  466. }
  467. func unpackStringOctet(msg []byte, off int) (string, int, error) {
  468. s := string(msg[off:])
  469. return s, len(msg), nil
  470. }
  471. func packStringOctet(s string, msg []byte, off int) (int, error) {
  472. txtTmp := make([]byte, 256*4+1)
  473. off, err := packOctetString(s, msg, off, txtTmp)
  474. if err != nil {
  475. return len(msg), err
  476. }
  477. return off, nil
  478. }
  479. func unpackDataNsec(msg []byte, off int) ([]uint16, int, error) {
  480. var nsec []uint16
  481. length, window, lastwindow := 0, 0, -1
  482. for off < len(msg) {
  483. if off+2 > len(msg) {
  484. return nsec, len(msg), &Error{err: "overflow unpacking nsecx"}
  485. }
  486. window = int(msg[off])
  487. length = int(msg[off+1])
  488. off += 2
  489. if window <= lastwindow {
  490. // RFC 4034: Blocks are present in the NSEC RR RDATA in
  491. // increasing numerical order.
  492. return nsec, len(msg), &Error{err: "out of order NSEC block"}
  493. }
  494. if length == 0 {
  495. // RFC 4034: Blocks with no types present MUST NOT be included.
  496. return nsec, len(msg), &Error{err: "empty NSEC block"}
  497. }
  498. if length > 32 {
  499. return nsec, len(msg), &Error{err: "NSEC block too long"}
  500. }
  501. if off+length > len(msg) {
  502. return nsec, len(msg), &Error{err: "overflowing NSEC block"}
  503. }
  504. // Walk the bytes in the window and extract the type bits
  505. for j := 0; j < length; j++ {
  506. b := msg[off+j]
  507. // Check the bits one by one, and set the type
  508. if b&0x80 == 0x80 {
  509. nsec = append(nsec, uint16(window*256+j*8+0))
  510. }
  511. if b&0x40 == 0x40 {
  512. nsec = append(nsec, uint16(window*256+j*8+1))
  513. }
  514. if b&0x20 == 0x20 {
  515. nsec = append(nsec, uint16(window*256+j*8+2))
  516. }
  517. if b&0x10 == 0x10 {
  518. nsec = append(nsec, uint16(window*256+j*8+3))
  519. }
  520. if b&0x8 == 0x8 {
  521. nsec = append(nsec, uint16(window*256+j*8+4))
  522. }
  523. if b&0x4 == 0x4 {
  524. nsec = append(nsec, uint16(window*256+j*8+5))
  525. }
  526. if b&0x2 == 0x2 {
  527. nsec = append(nsec, uint16(window*256+j*8+6))
  528. }
  529. if b&0x1 == 0x1 {
  530. nsec = append(nsec, uint16(window*256+j*8+7))
  531. }
  532. }
  533. off += length
  534. lastwindow = window
  535. }
  536. return nsec, off, nil
  537. }
  538. func packDataNsec(bitmap []uint16, msg []byte, off int) (int, error) {
  539. if len(bitmap) == 0 {
  540. return off, nil
  541. }
  542. var lastwindow, lastlength uint16
  543. for j := 0; j < len(bitmap); j++ {
  544. t := bitmap[j]
  545. window := t / 256
  546. length := (t-window*256)/8 + 1
  547. if window > lastwindow && lastlength != 0 { // New window, jump to the new offset
  548. off += int(lastlength) + 2
  549. lastlength = 0
  550. }
  551. if window < lastwindow || length < lastlength {
  552. return len(msg), &Error{err: "nsec bits out of order"}
  553. }
  554. if off+2+int(length) > len(msg) {
  555. return len(msg), &Error{err: "overflow packing nsec"}
  556. }
  557. // Setting the window #
  558. msg[off] = byte(window)
  559. // Setting the octets length
  560. msg[off+1] = byte(length)
  561. // Setting the bit value for the type in the right octet
  562. msg[off+1+int(length)] |= byte(1 << (7 - t%8))
  563. lastwindow, lastlength = window, length
  564. }
  565. off += int(lastlength) + 2
  566. return off, nil
  567. }
  568. func unpackDataDomainNames(msg []byte, off, end int) ([]string, int, error) {
  569. var (
  570. servers []string
  571. s string
  572. err error
  573. )
  574. if end > len(msg) {
  575. return nil, len(msg), &Error{err: "overflow unpacking domain names"}
  576. }
  577. for off < end {
  578. s, off, err = UnpackDomainName(msg, off)
  579. if err != nil {
  580. return servers, len(msg), err
  581. }
  582. servers = append(servers, s)
  583. }
  584. return servers, off, nil
  585. }
  586. func packDataDomainNames(names []string, msg []byte, off int, compression compressionMap, compress bool) (int, error) {
  587. var err error
  588. for j := 0; j < len(names); j++ {
  589. off, err = packDomainName(names[j], msg, off, compression, compress)
  590. if err != nil {
  591. return len(msg), err
  592. }
  593. }
  594. return off, nil
  595. }