msg.go 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206
  1. // DNS packet assembly, see RFC 1035. Converting from - Unpack() -
  2. // and to - Pack() - wire format.
  3. // All the packers and unpackers take a (msg []byte, off int)
  4. // and return (off1 int, ok bool). If they return ok==false, they
  5. // also return off1==len(msg), so that the next unpacker will
  6. // also fail. This lets us avoid checks of ok until the end of a
  7. // packing sequence.
  8. package dns
  9. //go:generate go run msg_generate.go
  10. //go:generate go run compress_generate.go
  11. import (
  12. crand "crypto/rand"
  13. "encoding/binary"
  14. "fmt"
  15. "math/big"
  16. "math/rand"
  17. "strconv"
  18. "sync"
  19. )
  20. const (
  21. maxCompressionOffset = 2 << 13 // We have 14 bits for the compression pointer
  22. maxDomainNameWireOctets = 255 // See RFC 1035 section 2.3.4
  23. )
  24. // Errors defined in this package.
  25. var (
  26. ErrAlg error = &Error{err: "bad algorithm"} // ErrAlg indicates an error with the (DNSSEC) algorithm.
  27. ErrAuth error = &Error{err: "bad authentication"} // ErrAuth indicates an error in the TSIG authentication.
  28. ErrBuf error = &Error{err: "buffer size too small"} // ErrBuf indicates that the buffer used is too small for the message.
  29. ErrConnEmpty error = &Error{err: "conn has no connection"} // ErrConnEmpty indicates a connection is being used before it is initialized.
  30. ErrExtendedRcode error = &Error{err: "bad extended rcode"} // ErrExtendedRcode ...
  31. ErrFqdn error = &Error{err: "domain must be fully qualified"} // ErrFqdn indicates that a domain name does not have a closing dot.
  32. ErrId error = &Error{err: "id mismatch"} // ErrId indicates there is a mismatch with the message's ID.
  33. ErrKeyAlg error = &Error{err: "bad key algorithm"} // ErrKeyAlg indicates that the algorithm in the key is not valid.
  34. ErrKey error = &Error{err: "bad key"}
  35. ErrKeySize error = &Error{err: "bad key size"}
  36. ErrLongDomain error = &Error{err: fmt.Sprintf("domain name exceeded %d wire-format octets", maxDomainNameWireOctets)}
  37. ErrNoSig error = &Error{err: "no signature found"}
  38. ErrPrivKey error = &Error{err: "bad private key"}
  39. ErrRcode error = &Error{err: "bad rcode"}
  40. ErrRdata error = &Error{err: "bad rdata"}
  41. ErrRRset error = &Error{err: "bad rrset"}
  42. ErrSecret error = &Error{err: "no secrets defined"}
  43. ErrShortRead error = &Error{err: "short read"}
  44. ErrSig error = &Error{err: "bad signature"} // ErrSig indicates that a signature can not be cryptographically validated.
  45. ErrSoa error = &Error{err: "no SOA"} // ErrSOA indicates that no SOA RR was seen when doing zone transfers.
  46. ErrTime error = &Error{err: "bad time"} // ErrTime indicates a timing error in TSIG authentication.
  47. ErrTruncated error = &Error{err: "failed to unpack truncated message"} // ErrTruncated indicates that we failed to unpack a truncated message. We unpacked as much as we had so Msg can still be used, if desired.
  48. )
  49. // Id by default, returns a 16 bits random number to be used as a
  50. // message id. The random provided should be good enough. This being a
  51. // variable the function can be reassigned to a custom function.
  52. // For instance, to make it return a static value:
  53. //
  54. // dns.Id = func() uint16 { return 3 }
  55. var Id = id
  56. var (
  57. idLock sync.Mutex
  58. idRand *rand.Rand
  59. )
  60. // id returns a 16 bits random number to be used as a
  61. // message id. The random provided should be good enough.
  62. func id() uint16 {
  63. idLock.Lock()
  64. if idRand == nil {
  65. // This (partially) works around
  66. // https://github.com/golang/go/issues/11833 by only
  67. // seeding idRand upon the first call to id.
  68. var seed int64
  69. var buf [8]byte
  70. if _, err := crand.Read(buf[:]); err == nil {
  71. seed = int64(binary.LittleEndian.Uint64(buf[:]))
  72. } else {
  73. seed = rand.Int63()
  74. }
  75. idRand = rand.New(rand.NewSource(seed))
  76. }
  77. // The call to idRand.Uint32 must be within the
  78. // mutex lock because *rand.Rand is not safe for
  79. // concurrent use.
  80. //
  81. // There is no added performance overhead to calling
  82. // idRand.Uint32 inside a mutex lock over just
  83. // calling rand.Uint32 as the global math/rand rng
  84. // is internally protected by a sync.Mutex.
  85. id := uint16(idRand.Uint32())
  86. idLock.Unlock()
  87. return id
  88. }
  89. // MsgHdr is a a manually-unpacked version of (id, bits).
  90. type MsgHdr struct {
  91. Id uint16
  92. Response bool
  93. Opcode int
  94. Authoritative bool
  95. Truncated bool
  96. RecursionDesired bool
  97. RecursionAvailable bool
  98. Zero bool
  99. AuthenticatedData bool
  100. CheckingDisabled bool
  101. Rcode int
  102. }
  103. // Msg contains the layout of a DNS message.
  104. type Msg struct {
  105. MsgHdr
  106. Compress bool `json:"-"` // If true, the message will be compressed when converted to wire format.
  107. Question []Question // Holds the RR(s) of the question section.
  108. Answer []RR // Holds the RR(s) of the answer section.
  109. Ns []RR // Holds the RR(s) of the authority section.
  110. Extra []RR // Holds the RR(s) of the additional section.
  111. }
  112. // ClassToString is a maps Classes to strings for each CLASS wire type.
  113. var ClassToString = map[uint16]string{
  114. ClassINET: "IN",
  115. ClassCSNET: "CS",
  116. ClassCHAOS: "CH",
  117. ClassHESIOD: "HS",
  118. ClassNONE: "NONE",
  119. ClassANY: "ANY",
  120. }
  121. // OpcodeToString maps Opcodes to strings.
  122. var OpcodeToString = map[int]string{
  123. OpcodeQuery: "QUERY",
  124. OpcodeIQuery: "IQUERY",
  125. OpcodeStatus: "STATUS",
  126. OpcodeNotify: "NOTIFY",
  127. OpcodeUpdate: "UPDATE",
  128. }
  129. // RcodeToString maps Rcodes to strings.
  130. var RcodeToString = map[int]string{
  131. RcodeSuccess: "NOERROR",
  132. RcodeFormatError: "FORMERR",
  133. RcodeServerFailure: "SERVFAIL",
  134. RcodeNameError: "NXDOMAIN",
  135. RcodeNotImplemented: "NOTIMPL",
  136. RcodeRefused: "REFUSED",
  137. RcodeYXDomain: "YXDOMAIN", // See RFC 2136
  138. RcodeYXRrset: "YXRRSET",
  139. RcodeNXRrset: "NXRRSET",
  140. RcodeNotAuth: "NOTAUTH",
  141. RcodeNotZone: "NOTZONE",
  142. RcodeBadSig: "BADSIG", // Also known as RcodeBadVers, see RFC 6891
  143. // RcodeBadVers: "BADVERS",
  144. RcodeBadKey: "BADKEY",
  145. RcodeBadTime: "BADTIME",
  146. RcodeBadMode: "BADMODE",
  147. RcodeBadName: "BADNAME",
  148. RcodeBadAlg: "BADALG",
  149. RcodeBadTrunc: "BADTRUNC",
  150. RcodeBadCookie: "BADCOOKIE",
  151. }
  152. // Domain names are a sequence of counted strings
  153. // split at the dots. They end with a zero-length string.
  154. // PackDomainName packs a domain name s into msg[off:].
  155. // If compression is wanted compress must be true and the compression
  156. // map needs to hold a mapping between domain names and offsets
  157. // pointing into msg.
  158. func PackDomainName(s string, msg []byte, off int, compression map[string]int, compress bool) (off1 int, err error) {
  159. off1, _, err = packDomainName(s, msg, off, compression, compress)
  160. return
  161. }
  162. func packDomainName(s string, msg []byte, off int, compression map[string]int, compress bool) (off1 int, labels int, err error) {
  163. // special case if msg == nil
  164. lenmsg := 256
  165. if msg != nil {
  166. lenmsg = len(msg)
  167. }
  168. ls := len(s)
  169. if ls == 0 { // Ok, for instance when dealing with update RR without any rdata.
  170. return off, 0, nil
  171. }
  172. // If not fully qualified, error out, but only if msg == nil #ugly
  173. switch {
  174. case msg == nil:
  175. if s[ls-1] != '.' {
  176. s += "."
  177. ls++
  178. }
  179. case msg != nil:
  180. if s[ls-1] != '.' {
  181. return lenmsg, 0, ErrFqdn
  182. }
  183. }
  184. // Each dot ends a segment of the name.
  185. // We trade each dot byte for a length byte.
  186. // Except for escaped dots (\.), which are normal dots.
  187. // There is also a trailing zero.
  188. // Compression
  189. nameoffset := -1
  190. pointer := -1
  191. // Emit sequence of counted strings, chopping at dots.
  192. begin := 0
  193. bs := []byte(s)
  194. roBs, bsFresh, escapedDot := s, true, false
  195. for i := 0; i < ls; i++ {
  196. if bs[i] == '\\' {
  197. for j := i; j < ls-1; j++ {
  198. bs[j] = bs[j+1]
  199. }
  200. ls--
  201. if off+1 > lenmsg {
  202. return lenmsg, labels, ErrBuf
  203. }
  204. // check for \DDD
  205. if i+2 < ls && isDigit(bs[i]) && isDigit(bs[i+1]) && isDigit(bs[i+2]) {
  206. bs[i] = dddToByte(bs[i:])
  207. for j := i + 1; j < ls-2; j++ {
  208. bs[j] = bs[j+2]
  209. }
  210. ls -= 2
  211. }
  212. escapedDot = bs[i] == '.'
  213. bsFresh = false
  214. continue
  215. }
  216. if bs[i] == '.' {
  217. if i > 0 && bs[i-1] == '.' && !escapedDot {
  218. // two dots back to back is not legal
  219. return lenmsg, labels, ErrRdata
  220. }
  221. if i-begin >= 1<<6 { // top two bits of length must be clear
  222. return lenmsg, labels, ErrRdata
  223. }
  224. // off can already (we're in a loop) be bigger than len(msg)
  225. // this happens when a name isn't fully qualified
  226. if off+1 > lenmsg {
  227. return lenmsg, labels, ErrBuf
  228. }
  229. if msg != nil {
  230. msg[off] = byte(i - begin)
  231. }
  232. offset := off
  233. off++
  234. for j := begin; j < i; j++ {
  235. if off+1 > lenmsg {
  236. return lenmsg, labels, ErrBuf
  237. }
  238. if msg != nil {
  239. msg[off] = bs[j]
  240. }
  241. off++
  242. }
  243. if compress && !bsFresh {
  244. roBs = string(bs)
  245. bsFresh = true
  246. }
  247. // Don't try to compress '.'
  248. // We should only compress when compress it true, but we should also still pick
  249. // up names that can be used for *future* compression(s).
  250. if compression != nil && roBs[begin:] != "." {
  251. if p, ok := compression[roBs[begin:]]; !ok {
  252. // Only offsets smaller than this can be used.
  253. if offset < maxCompressionOffset {
  254. compression[roBs[begin:]] = offset
  255. }
  256. } else {
  257. // The first hit is the longest matching dname
  258. // keep the pointer offset we get back and store
  259. // the offset of the current name, because that's
  260. // where we need to insert the pointer later
  261. // If compress is true, we're allowed to compress this dname
  262. if pointer == -1 && compress {
  263. pointer = p // Where to point to
  264. nameoffset = offset // Where to point from
  265. break
  266. }
  267. }
  268. }
  269. labels++
  270. begin = i + 1
  271. }
  272. escapedDot = false
  273. }
  274. // Root label is special
  275. if len(bs) == 1 && bs[0] == '.' {
  276. return off, labels, nil
  277. }
  278. // If we did compression and we find something add the pointer here
  279. if pointer != -1 {
  280. // We have two bytes (14 bits) to put the pointer in
  281. // if msg == nil, we will never do compression
  282. binary.BigEndian.PutUint16(msg[nameoffset:], uint16(pointer^0xC000))
  283. off = nameoffset + 1
  284. goto End
  285. }
  286. if msg != nil && off < len(msg) {
  287. msg[off] = 0
  288. }
  289. End:
  290. off++
  291. return off, labels, nil
  292. }
  293. // Unpack a domain name.
  294. // In addition to the simple sequences of counted strings above,
  295. // domain names are allowed to refer to strings elsewhere in the
  296. // packet, to avoid repeating common suffixes when returning
  297. // many entries in a single domain. The pointers are marked
  298. // by a length byte with the top two bits set. Ignoring those
  299. // two bits, that byte and the next give a 14 bit offset from msg[0]
  300. // where we should pick up the trail.
  301. // Note that if we jump elsewhere in the packet,
  302. // we return off1 == the offset after the first pointer we found,
  303. // which is where the next record will start.
  304. // In theory, the pointers are only allowed to jump backward.
  305. // We let them jump anywhere and stop jumping after a while.
  306. // UnpackDomainName unpacks a domain name into a string.
  307. func UnpackDomainName(msg []byte, off int) (string, int, error) {
  308. s := make([]byte, 0, 64)
  309. off1 := 0
  310. lenmsg := len(msg)
  311. maxLen := maxDomainNameWireOctets
  312. ptr := 0 // number of pointers followed
  313. Loop:
  314. for {
  315. if off >= lenmsg {
  316. return "", lenmsg, ErrBuf
  317. }
  318. c := int(msg[off])
  319. off++
  320. switch c & 0xC0 {
  321. case 0x00:
  322. if c == 0x00 {
  323. // end of name
  324. break Loop
  325. }
  326. // literal string
  327. if off+c > lenmsg {
  328. return "", lenmsg, ErrBuf
  329. }
  330. for j := off; j < off+c; j++ {
  331. switch b := msg[j]; b {
  332. case '.', '(', ')', ';', ' ', '@':
  333. fallthrough
  334. case '"', '\\':
  335. s = append(s, '\\', b)
  336. // presentation-format \X escapes add an extra byte
  337. maxLen++
  338. default:
  339. if b < 32 || b >= 127 { // unprintable, use \DDD
  340. var buf [3]byte
  341. bufs := strconv.AppendInt(buf[:0], int64(b), 10)
  342. s = append(s, '\\')
  343. for i := 0; i < 3-len(bufs); i++ {
  344. s = append(s, '0')
  345. }
  346. for _, r := range bufs {
  347. s = append(s, r)
  348. }
  349. // presentation-format \DDD escapes add 3 extra bytes
  350. maxLen += 3
  351. } else {
  352. s = append(s, b)
  353. }
  354. }
  355. }
  356. s = append(s, '.')
  357. off += c
  358. case 0xC0:
  359. // pointer to somewhere else in msg.
  360. // remember location after first ptr,
  361. // since that's how many bytes we consumed.
  362. // also, don't follow too many pointers --
  363. // maybe there's a loop.
  364. if off >= lenmsg {
  365. return "", lenmsg, ErrBuf
  366. }
  367. c1 := msg[off]
  368. off++
  369. if ptr == 0 {
  370. off1 = off
  371. }
  372. if ptr++; ptr > 10 {
  373. return "", lenmsg, &Error{err: "too many compression pointers"}
  374. }
  375. // pointer should guarantee that it advances and points forwards at least
  376. // but the condition on previous three lines guarantees that it's
  377. // at least loop-free
  378. off = (c^0xC0)<<8 | int(c1)
  379. default:
  380. // 0x80 and 0x40 are reserved
  381. return "", lenmsg, ErrRdata
  382. }
  383. }
  384. if ptr == 0 {
  385. off1 = off
  386. }
  387. if len(s) == 0 {
  388. s = []byte(".")
  389. } else if len(s) >= maxLen {
  390. // error if the name is too long, but don't throw it away
  391. return string(s), lenmsg, ErrLongDomain
  392. }
  393. return string(s), off1, nil
  394. }
  395. func packTxt(txt []string, msg []byte, offset int, tmp []byte) (int, error) {
  396. if len(txt) == 0 {
  397. if offset >= len(msg) {
  398. return offset, ErrBuf
  399. }
  400. msg[offset] = 0
  401. return offset, nil
  402. }
  403. var err error
  404. for i := range txt {
  405. if len(txt[i]) > len(tmp) {
  406. return offset, ErrBuf
  407. }
  408. offset, err = packTxtString(txt[i], msg, offset, tmp)
  409. if err != nil {
  410. return offset, err
  411. }
  412. }
  413. return offset, nil
  414. }
  415. func packTxtString(s string, msg []byte, offset int, tmp []byte) (int, error) {
  416. lenByteOffset := offset
  417. if offset >= len(msg) || len(s) > len(tmp) {
  418. return offset, ErrBuf
  419. }
  420. offset++
  421. bs := tmp[:len(s)]
  422. copy(bs, s)
  423. for i := 0; i < len(bs); i++ {
  424. if len(msg) <= offset {
  425. return offset, ErrBuf
  426. }
  427. if bs[i] == '\\' {
  428. i++
  429. if i == len(bs) {
  430. break
  431. }
  432. // check for \DDD
  433. if i+2 < len(bs) && isDigit(bs[i]) && isDigit(bs[i+1]) && isDigit(bs[i+2]) {
  434. msg[offset] = dddToByte(bs[i:])
  435. i += 2
  436. } else {
  437. msg[offset] = bs[i]
  438. }
  439. } else {
  440. msg[offset] = bs[i]
  441. }
  442. offset++
  443. }
  444. l := offset - lenByteOffset - 1
  445. if l > 255 {
  446. return offset, &Error{err: "string exceeded 255 bytes in txt"}
  447. }
  448. msg[lenByteOffset] = byte(l)
  449. return offset, nil
  450. }
  451. func packOctetString(s string, msg []byte, offset int, tmp []byte) (int, error) {
  452. if offset >= len(msg) || len(s) > len(tmp) {
  453. return offset, ErrBuf
  454. }
  455. bs := tmp[:len(s)]
  456. copy(bs, s)
  457. for i := 0; i < len(bs); i++ {
  458. if len(msg) <= offset {
  459. return offset, ErrBuf
  460. }
  461. if bs[i] == '\\' {
  462. i++
  463. if i == len(bs) {
  464. break
  465. }
  466. // check for \DDD
  467. if i+2 < len(bs) && isDigit(bs[i]) && isDigit(bs[i+1]) && isDigit(bs[i+2]) {
  468. msg[offset] = dddToByte(bs[i:])
  469. i += 2
  470. } else {
  471. msg[offset] = bs[i]
  472. }
  473. } else {
  474. msg[offset] = bs[i]
  475. }
  476. offset++
  477. }
  478. return offset, nil
  479. }
  480. func unpackTxt(msg []byte, off0 int) (ss []string, off int, err error) {
  481. off = off0
  482. var s string
  483. for off < len(msg) && err == nil {
  484. s, off, err = unpackTxtString(msg, off)
  485. if err == nil {
  486. ss = append(ss, s)
  487. }
  488. }
  489. return
  490. }
  491. func unpackTxtString(msg []byte, offset int) (string, int, error) {
  492. if offset+1 > len(msg) {
  493. return "", offset, &Error{err: "overflow unpacking txt"}
  494. }
  495. l := int(msg[offset])
  496. if offset+l+1 > len(msg) {
  497. return "", offset, &Error{err: "overflow unpacking txt"}
  498. }
  499. s := make([]byte, 0, l)
  500. for _, b := range msg[offset+1 : offset+1+l] {
  501. switch b {
  502. case '"', '\\':
  503. s = append(s, '\\', b)
  504. default:
  505. if b < 32 || b > 127 { // unprintable
  506. var buf [3]byte
  507. bufs := strconv.AppendInt(buf[:0], int64(b), 10)
  508. s = append(s, '\\')
  509. for i := 0; i < 3-len(bufs); i++ {
  510. s = append(s, '0')
  511. }
  512. for _, r := range bufs {
  513. s = append(s, r)
  514. }
  515. } else {
  516. s = append(s, b)
  517. }
  518. }
  519. }
  520. offset += 1 + l
  521. return string(s), offset, nil
  522. }
  523. // Helpers for dealing with escaped bytes
  524. func isDigit(b byte) bool { return b >= '0' && b <= '9' }
  525. func dddToByte(s []byte) byte {
  526. return byte((s[0]-'0')*100 + (s[1]-'0')*10 + (s[2] - '0'))
  527. }
  528. // Helper function for packing and unpacking
  529. func intToBytes(i *big.Int, length int) []byte {
  530. buf := i.Bytes()
  531. if len(buf) < length {
  532. b := make([]byte, length)
  533. copy(b[length-len(buf):], buf)
  534. return b
  535. }
  536. return buf
  537. }
  538. // PackRR packs a resource record rr into msg[off:].
  539. // See PackDomainName for documentation about the compression.
  540. func PackRR(rr RR, msg []byte, off int, compression map[string]int, compress bool) (off1 int, err error) {
  541. if rr == nil {
  542. return len(msg), &Error{err: "nil rr"}
  543. }
  544. off1, err = rr.pack(msg, off, compression, compress)
  545. if err != nil {
  546. return len(msg), err
  547. }
  548. // TODO(miek): Not sure if this is needed? If removed we can remove rawmsg.go as well.
  549. if rawSetRdlength(msg, off, off1) {
  550. return off1, nil
  551. }
  552. return off, ErrRdata
  553. }
  554. // UnpackRR unpacks msg[off:] into an RR.
  555. func UnpackRR(msg []byte, off int) (rr RR, off1 int, err error) {
  556. h, off, msg, err := unpackHeader(msg, off)
  557. if err != nil {
  558. return nil, len(msg), err
  559. }
  560. return UnpackRRWithHeader(h, msg, off)
  561. }
  562. // UnpackRRWithHeader unpacks the record type specific payload given an existing
  563. // RR_Header.
  564. func UnpackRRWithHeader(h RR_Header, msg []byte, off int) (rr RR, off1 int, err error) {
  565. end := off + int(h.Rdlength)
  566. if fn, known := typeToUnpack[h.Rrtype]; !known {
  567. rr, off, err = unpackRFC3597(h, msg, off)
  568. } else {
  569. rr, off, err = fn(h, msg, off)
  570. }
  571. if off != end {
  572. return &h, end, &Error{err: "bad rdlength"}
  573. }
  574. return rr, off, err
  575. }
  576. // unpackRRslice unpacks msg[off:] into an []RR.
  577. // If we cannot unpack the whole array, then it will return nil
  578. func unpackRRslice(l int, msg []byte, off int) (dst1 []RR, off1 int, err error) {
  579. var r RR
  580. // Don't pre-allocate, l may be under attacker control
  581. var dst []RR
  582. for i := 0; i < l; i++ {
  583. off1 := off
  584. r, off, err = UnpackRR(msg, off)
  585. if err != nil {
  586. off = len(msg)
  587. break
  588. }
  589. // If offset does not increase anymore, l is a lie
  590. if off1 == off {
  591. l = i
  592. break
  593. }
  594. dst = append(dst, r)
  595. }
  596. if err != nil && off == len(msg) {
  597. dst = nil
  598. }
  599. return dst, off, err
  600. }
  601. // Convert a MsgHdr to a string, with dig-like headers:
  602. //
  603. //;; opcode: QUERY, status: NOERROR, id: 48404
  604. //
  605. //;; flags: qr aa rd ra;
  606. func (h *MsgHdr) String() string {
  607. if h == nil {
  608. return "<nil> MsgHdr"
  609. }
  610. s := ";; opcode: " + OpcodeToString[h.Opcode]
  611. s += ", status: " + RcodeToString[h.Rcode]
  612. s += ", id: " + strconv.Itoa(int(h.Id)) + "\n"
  613. s += ";; flags:"
  614. if h.Response {
  615. s += " qr"
  616. }
  617. if h.Authoritative {
  618. s += " aa"
  619. }
  620. if h.Truncated {
  621. s += " tc"
  622. }
  623. if h.RecursionDesired {
  624. s += " rd"
  625. }
  626. if h.RecursionAvailable {
  627. s += " ra"
  628. }
  629. if h.Zero { // Hmm
  630. s += " z"
  631. }
  632. if h.AuthenticatedData {
  633. s += " ad"
  634. }
  635. if h.CheckingDisabled {
  636. s += " cd"
  637. }
  638. s += ";"
  639. return s
  640. }
  641. // Pack packs a Msg: it is converted to to wire format.
  642. // If the dns.Compress is true the message will be in compressed wire format.
  643. func (dns *Msg) Pack() (msg []byte, err error) {
  644. return dns.PackBuffer(nil)
  645. }
  646. // PackBuffer packs a Msg, using the given buffer buf. If buf is too small a new buffer is allocated.
  647. func (dns *Msg) PackBuffer(buf []byte) (msg []byte, err error) {
  648. var compression map[string]int
  649. if dns.Compress {
  650. compression = make(map[string]int) // Compression pointer mappings.
  651. }
  652. return dns.packBufferWithCompressionMap(buf, compression)
  653. }
  654. // packBufferWithCompressionMap packs a Msg, using the given buffer buf.
  655. func (dns *Msg) packBufferWithCompressionMap(buf []byte, compression map[string]int) (msg []byte, err error) {
  656. // We use a similar function in tsig.go's stripTsig.
  657. var dh Header
  658. if dns.Rcode < 0 || dns.Rcode > 0xFFF {
  659. return nil, ErrRcode
  660. }
  661. if dns.Rcode > 0xF {
  662. // Regular RCODE field is 4 bits
  663. opt := dns.IsEdns0()
  664. if opt == nil {
  665. return nil, ErrExtendedRcode
  666. }
  667. opt.SetExtendedRcode(uint8(dns.Rcode >> 4))
  668. }
  669. // Convert convenient Msg into wire-like Header.
  670. dh.Id = dns.Id
  671. dh.Bits = uint16(dns.Opcode)<<11 | uint16(dns.Rcode&0xF)
  672. if dns.Response {
  673. dh.Bits |= _QR
  674. }
  675. if dns.Authoritative {
  676. dh.Bits |= _AA
  677. }
  678. if dns.Truncated {
  679. dh.Bits |= _TC
  680. }
  681. if dns.RecursionDesired {
  682. dh.Bits |= _RD
  683. }
  684. if dns.RecursionAvailable {
  685. dh.Bits |= _RA
  686. }
  687. if dns.Zero {
  688. dh.Bits |= _Z
  689. }
  690. if dns.AuthenticatedData {
  691. dh.Bits |= _AD
  692. }
  693. if dns.CheckingDisabled {
  694. dh.Bits |= _CD
  695. }
  696. // Prepare variable sized arrays.
  697. question := dns.Question
  698. answer := dns.Answer
  699. ns := dns.Ns
  700. extra := dns.Extra
  701. dh.Qdcount = uint16(len(question))
  702. dh.Ancount = uint16(len(answer))
  703. dh.Nscount = uint16(len(ns))
  704. dh.Arcount = uint16(len(extra))
  705. // We need the uncompressed length here, because we first pack it and then compress it.
  706. msg = buf
  707. uncompressedLen := compressedLen(dns, false)
  708. if packLen := uncompressedLen + 1; len(msg) < packLen {
  709. msg = make([]byte, packLen)
  710. }
  711. // Pack it in: header and then the pieces.
  712. off := 0
  713. off, err = dh.pack(msg, off, compression, dns.Compress)
  714. if err != nil {
  715. return nil, err
  716. }
  717. for i := 0; i < len(question); i++ {
  718. off, err = question[i].pack(msg, off, compression, dns.Compress)
  719. if err != nil {
  720. return nil, err
  721. }
  722. }
  723. for i := 0; i < len(answer); i++ {
  724. off, err = PackRR(answer[i], msg, off, compression, dns.Compress)
  725. if err != nil {
  726. return nil, err
  727. }
  728. }
  729. for i := 0; i < len(ns); i++ {
  730. off, err = PackRR(ns[i], msg, off, compression, dns.Compress)
  731. if err != nil {
  732. return nil, err
  733. }
  734. }
  735. for i := 0; i < len(extra); i++ {
  736. off, err = PackRR(extra[i], msg, off, compression, dns.Compress)
  737. if err != nil {
  738. return nil, err
  739. }
  740. }
  741. return msg[:off], nil
  742. }
  743. // Unpack unpacks a binary message to a Msg structure.
  744. func (dns *Msg) Unpack(msg []byte) (err error) {
  745. var (
  746. dh Header
  747. off int
  748. )
  749. if dh, off, err = unpackMsgHdr(msg, off); err != nil {
  750. return err
  751. }
  752. dns.Id = dh.Id
  753. dns.Response = (dh.Bits & _QR) != 0
  754. dns.Opcode = int(dh.Bits>>11) & 0xF
  755. dns.Authoritative = (dh.Bits & _AA) != 0
  756. dns.Truncated = (dh.Bits & _TC) != 0
  757. dns.RecursionDesired = (dh.Bits & _RD) != 0
  758. dns.RecursionAvailable = (dh.Bits & _RA) != 0
  759. dns.Zero = (dh.Bits & _Z) != 0
  760. dns.AuthenticatedData = (dh.Bits & _AD) != 0
  761. dns.CheckingDisabled = (dh.Bits & _CD) != 0
  762. dns.Rcode = int(dh.Bits & 0xF)
  763. // If we are at the end of the message we should return *just* the
  764. // header. This can still be useful to the caller. 9.9.9.9 sends these
  765. // when responding with REFUSED for instance.
  766. if off == len(msg) {
  767. // reset sections before returning
  768. dns.Question, dns.Answer, dns.Ns, dns.Extra = nil, nil, nil, nil
  769. return nil
  770. }
  771. // Qdcount, Ancount, Nscount, Arcount can't be trusted, as they are
  772. // attacker controlled. This means we can't use them to pre-allocate
  773. // slices.
  774. dns.Question = nil
  775. for i := 0; i < int(dh.Qdcount); i++ {
  776. off1 := off
  777. var q Question
  778. q, off, err = unpackQuestion(msg, off)
  779. if err != nil {
  780. // Even if Truncated is set, we only will set ErrTruncated if we
  781. // actually got the questions
  782. return err
  783. }
  784. if off1 == off { // Offset does not increase anymore, dh.Qdcount is a lie!
  785. dh.Qdcount = uint16(i)
  786. break
  787. }
  788. dns.Question = append(dns.Question, q)
  789. }
  790. dns.Answer, off, err = unpackRRslice(int(dh.Ancount), msg, off)
  791. // The header counts might have been wrong so we need to update it
  792. dh.Ancount = uint16(len(dns.Answer))
  793. if err == nil {
  794. dns.Ns, off, err = unpackRRslice(int(dh.Nscount), msg, off)
  795. }
  796. // The header counts might have been wrong so we need to update it
  797. dh.Nscount = uint16(len(dns.Ns))
  798. if err == nil {
  799. dns.Extra, off, err = unpackRRslice(int(dh.Arcount), msg, off)
  800. }
  801. // The header counts might have been wrong so we need to update it
  802. dh.Arcount = uint16(len(dns.Extra))
  803. if off != len(msg) {
  804. // TODO(miek) make this an error?
  805. // use PackOpt to let people tell how detailed the error reporting should be?
  806. // println("dns: extra bytes in dns packet", off, "<", len(msg))
  807. } else if dns.Truncated {
  808. // Whether we ran into a an error or not, we want to return that it
  809. // was truncated
  810. err = ErrTruncated
  811. }
  812. return err
  813. }
  814. // Convert a complete message to a string with dig-like output.
  815. func (dns *Msg) String() string {
  816. if dns == nil {
  817. return "<nil> MsgHdr"
  818. }
  819. s := dns.MsgHdr.String() + " "
  820. s += "QUERY: " + strconv.Itoa(len(dns.Question)) + ", "
  821. s += "ANSWER: " + strconv.Itoa(len(dns.Answer)) + ", "
  822. s += "AUTHORITY: " + strconv.Itoa(len(dns.Ns)) + ", "
  823. s += "ADDITIONAL: " + strconv.Itoa(len(dns.Extra)) + "\n"
  824. if len(dns.Question) > 0 {
  825. s += "\n;; QUESTION SECTION:\n"
  826. for i := 0; i < len(dns.Question); i++ {
  827. s += dns.Question[i].String() + "\n"
  828. }
  829. }
  830. if len(dns.Answer) > 0 {
  831. s += "\n;; ANSWER SECTION:\n"
  832. for i := 0; i < len(dns.Answer); i++ {
  833. if dns.Answer[i] != nil {
  834. s += dns.Answer[i].String() + "\n"
  835. }
  836. }
  837. }
  838. if len(dns.Ns) > 0 {
  839. s += "\n;; AUTHORITY SECTION:\n"
  840. for i := 0; i < len(dns.Ns); i++ {
  841. if dns.Ns[i] != nil {
  842. s += dns.Ns[i].String() + "\n"
  843. }
  844. }
  845. }
  846. if len(dns.Extra) > 0 {
  847. s += "\n;; ADDITIONAL SECTION:\n"
  848. for i := 0; i < len(dns.Extra); i++ {
  849. if dns.Extra[i] != nil {
  850. s += dns.Extra[i].String() + "\n"
  851. }
  852. }
  853. }
  854. return s
  855. }
  856. // Len returns the message length when in (un)compressed wire format.
  857. // If dns.Compress is true compression it is taken into account. Len()
  858. // is provided to be a faster way to get the size of the resulting packet,
  859. // than packing it, measuring the size and discarding the buffer.
  860. func (dns *Msg) Len() int { return compressedLen(dns, dns.Compress) }
  861. func compressedLenWithCompressionMap(dns *Msg, compression map[string]int) int {
  862. l := 12 // Message header is always 12 bytes
  863. for _, r := range dns.Question {
  864. compressionLenHelper(compression, r.Name, l)
  865. l += r.len()
  866. }
  867. l += compressionLenSlice(l, compression, dns.Answer)
  868. l += compressionLenSlice(l, compression, dns.Ns)
  869. l += compressionLenSlice(l, compression, dns.Extra)
  870. return l
  871. }
  872. // compressedLen returns the message length when in compressed wire format
  873. // when compress is true, otherwise the uncompressed length is returned.
  874. func compressedLen(dns *Msg, compress bool) int {
  875. // We always return one more than needed.
  876. if compress {
  877. compression := map[string]int{}
  878. return compressedLenWithCompressionMap(dns, compression)
  879. }
  880. l := 12 // Message header is always 12 bytes
  881. for _, r := range dns.Question {
  882. l += r.len()
  883. }
  884. for _, r := range dns.Answer {
  885. if r != nil {
  886. l += r.len()
  887. }
  888. }
  889. for _, r := range dns.Ns {
  890. if r != nil {
  891. l += r.len()
  892. }
  893. }
  894. for _, r := range dns.Extra {
  895. if r != nil {
  896. l += r.len()
  897. }
  898. }
  899. return l
  900. }
  901. func compressionLenSlice(lenp int, c map[string]int, rs []RR) int {
  902. initLen := lenp
  903. for _, r := range rs {
  904. if r == nil {
  905. continue
  906. }
  907. // TmpLen is to track len of record at 14bits boudaries
  908. tmpLen := lenp
  909. x := r.len()
  910. // track this length, and the global length in len, while taking compression into account for both.
  911. k, ok, _ := compressionLenSearch(c, r.Header().Name)
  912. if ok {
  913. // Size of x is reduced by k, but we add 1 since k includes the '.' and label descriptor take 2 bytes
  914. // so, basically x:= x - k - 1 + 2
  915. x += 1 - k
  916. }
  917. tmpLen += compressionLenHelper(c, r.Header().Name, tmpLen)
  918. k, ok, _ = compressionLenSearchType(c, r)
  919. if ok {
  920. x += 1 - k
  921. }
  922. lenp += x
  923. tmpLen = lenp
  924. tmpLen += compressionLenHelperType(c, r, tmpLen)
  925. }
  926. return lenp - initLen
  927. }
  928. // Put the parts of the name in the compression map, return the size in bytes added in payload
  929. func compressionLenHelper(c map[string]int, s string, currentLen int) int {
  930. if currentLen > maxCompressionOffset {
  931. // We won't be able to add any label that could be re-used later anyway
  932. return 0
  933. }
  934. if _, ok := c[s]; ok {
  935. return 0
  936. }
  937. initLen := currentLen
  938. pref := ""
  939. prev := s
  940. lbs := Split(s)
  941. for j := 0; j < len(lbs); j++ {
  942. pref = s[lbs[j]:]
  943. currentLen += len(prev) - len(pref)
  944. prev = pref
  945. if _, ok := c[pref]; !ok {
  946. // If first byte label is within the first 14bits, it might be re-used later
  947. if currentLen < maxCompressionOffset {
  948. c[pref] = currentLen
  949. }
  950. } else {
  951. added := currentLen - initLen
  952. if j > 0 {
  953. // We added a new PTR
  954. added += 2
  955. }
  956. return added
  957. }
  958. }
  959. return currentLen - initLen
  960. }
  961. // Look for each part in the compression map and returns its length,
  962. // keep on searching so we get the longest match.
  963. // Will return the size of compression found, whether a match has been
  964. // found and the size of record if added in payload
  965. func compressionLenSearch(c map[string]int, s string) (int, bool, int) {
  966. off := 0
  967. end := false
  968. if s == "" { // don't bork on bogus data
  969. return 0, false, 0
  970. }
  971. fullSize := 0
  972. for {
  973. if _, ok := c[s[off:]]; ok {
  974. return len(s[off:]), true, fullSize + off
  975. }
  976. if end {
  977. break
  978. }
  979. // Each label descriptor takes 2 bytes, add it
  980. fullSize += 2
  981. off, end = NextLabel(s, off)
  982. }
  983. return 0, false, fullSize + len(s)
  984. }
  985. // Copy returns a new RR which is a deep-copy of r.
  986. func Copy(r RR) RR { r1 := r.copy(); return r1 }
  987. // Len returns the length (in octets) of the uncompressed RR in wire format.
  988. func Len(r RR) int { return r.len() }
  989. // Copy returns a new *Msg which is a deep-copy of dns.
  990. func (dns *Msg) Copy() *Msg { return dns.CopyTo(new(Msg)) }
  991. // CopyTo copies the contents to the provided message using a deep-copy and returns the copy.
  992. func (dns *Msg) CopyTo(r1 *Msg) *Msg {
  993. r1.MsgHdr = dns.MsgHdr
  994. r1.Compress = dns.Compress
  995. if len(dns.Question) > 0 {
  996. r1.Question = make([]Question, len(dns.Question))
  997. copy(r1.Question, dns.Question) // TODO(miek): Question is an immutable value, ok to do a shallow-copy
  998. }
  999. rrArr := make([]RR, len(dns.Answer)+len(dns.Ns)+len(dns.Extra))
  1000. var rri int
  1001. if len(dns.Answer) > 0 {
  1002. rrbegin := rri
  1003. for i := 0; i < len(dns.Answer); i++ {
  1004. rrArr[rri] = dns.Answer[i].copy()
  1005. rri++
  1006. }
  1007. r1.Answer = rrArr[rrbegin:rri:rri]
  1008. }
  1009. if len(dns.Ns) > 0 {
  1010. rrbegin := rri
  1011. for i := 0; i < len(dns.Ns); i++ {
  1012. rrArr[rri] = dns.Ns[i].copy()
  1013. rri++
  1014. }
  1015. r1.Ns = rrArr[rrbegin:rri:rri]
  1016. }
  1017. if len(dns.Extra) > 0 {
  1018. rrbegin := rri
  1019. for i := 0; i < len(dns.Extra); i++ {
  1020. rrArr[rri] = dns.Extra[i].copy()
  1021. rri++
  1022. }
  1023. r1.Extra = rrArr[rrbegin:rri:rri]
  1024. }
  1025. return r1
  1026. }
  1027. func (q *Question) pack(msg []byte, off int, compression map[string]int, compress bool) (int, error) {
  1028. off, err := PackDomainName(q.Name, msg, off, compression, compress)
  1029. if err != nil {
  1030. return off, err
  1031. }
  1032. off, err = packUint16(q.Qtype, msg, off)
  1033. if err != nil {
  1034. return off, err
  1035. }
  1036. off, err = packUint16(q.Qclass, msg, off)
  1037. if err != nil {
  1038. return off, err
  1039. }
  1040. return off, nil
  1041. }
  1042. func unpackQuestion(msg []byte, off int) (Question, int, error) {
  1043. var (
  1044. q Question
  1045. err error
  1046. )
  1047. q.Name, off, err = UnpackDomainName(msg, off)
  1048. if err != nil {
  1049. return q, off, err
  1050. }
  1051. if off == len(msg) {
  1052. return q, off, nil
  1053. }
  1054. q.Qtype, off, err = unpackUint16(msg, off)
  1055. if err != nil {
  1056. return q, off, err
  1057. }
  1058. if off == len(msg) {
  1059. return q, off, nil
  1060. }
  1061. q.Qclass, off, err = unpackUint16(msg, off)
  1062. if off == len(msg) {
  1063. return q, off, nil
  1064. }
  1065. return q, off, err
  1066. }
  1067. func (dh *Header) pack(msg []byte, off int, compression map[string]int, compress bool) (int, error) {
  1068. off, err := packUint16(dh.Id, msg, off)
  1069. if err != nil {
  1070. return off, err
  1071. }
  1072. off, err = packUint16(dh.Bits, msg, off)
  1073. if err != nil {
  1074. return off, err
  1075. }
  1076. off, err = packUint16(dh.Qdcount, msg, off)
  1077. if err != nil {
  1078. return off, err
  1079. }
  1080. off, err = packUint16(dh.Ancount, msg, off)
  1081. if err != nil {
  1082. return off, err
  1083. }
  1084. off, err = packUint16(dh.Nscount, msg, off)
  1085. if err != nil {
  1086. return off, err
  1087. }
  1088. off, err = packUint16(dh.Arcount, msg, off)
  1089. return off, err
  1090. }
  1091. func unpackMsgHdr(msg []byte, off int) (Header, int, error) {
  1092. var (
  1093. dh Header
  1094. err error
  1095. )
  1096. dh.Id, off, err = unpackUint16(msg, off)
  1097. if err != nil {
  1098. return dh, off, err
  1099. }
  1100. dh.Bits, off, err = unpackUint16(msg, off)
  1101. if err != nil {
  1102. return dh, off, err
  1103. }
  1104. dh.Qdcount, off, err = unpackUint16(msg, off)
  1105. if err != nil {
  1106. return dh, off, err
  1107. }
  1108. dh.Ancount, off, err = unpackUint16(msg, off)
  1109. if err != nil {
  1110. return dh, off, err
  1111. }
  1112. dh.Nscount, off, err = unpackUint16(msg, off)
  1113. if err != nil {
  1114. return dh, off, err
  1115. }
  1116. dh.Arcount, off, err = unpackUint16(msg, off)
  1117. return dh, off, err
  1118. }