compress_generate.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. //+build ignore
  2. // compression_generate.go is meant to run with go generate. It will use
  3. // go/{importer,types} to track down all the RR struct types. Then for each type
  4. // it will look to see if there are (compressible) names, if so it will add that
  5. // type to compressionLenHelperType and comressionLenSearchType which "fake" the
  6. // compression so that Len() is fast.
  7. package main
  8. import (
  9. "bytes"
  10. "fmt"
  11. "go/format"
  12. "go/importer"
  13. "go/types"
  14. "log"
  15. "os"
  16. )
  17. var packageHdr = `
  18. // Code generated by "go run compress_generate.go"; DO NOT EDIT.
  19. package dns
  20. `
  21. // getTypeStruct will take a type and the package scope, and return the
  22. // (innermost) struct if the type is considered a RR type (currently defined as
  23. // those structs beginning with a RR_Header, could be redefined as implementing
  24. // the RR interface). The bool return value indicates if embedded structs were
  25. // resolved.
  26. func getTypeStruct(t types.Type, scope *types.Scope) (*types.Struct, bool) {
  27. st, ok := t.Underlying().(*types.Struct)
  28. if !ok {
  29. return nil, false
  30. }
  31. if st.Field(0).Type() == scope.Lookup("RR_Header").Type() {
  32. return st, false
  33. }
  34. if st.Field(0).Anonymous() {
  35. st, _ := getTypeStruct(st.Field(0).Type(), scope)
  36. return st, true
  37. }
  38. return nil, false
  39. }
  40. func main() {
  41. // Import and type-check the package
  42. pkg, err := importer.Default().Import("github.com/miekg/dns")
  43. fatalIfErr(err)
  44. scope := pkg.Scope()
  45. var domainTypes []string // Types that have a domain name in them (either compressible or not).
  46. var cdomainTypes []string // Types that have a compressible domain name in them (subset of domainType)
  47. Names:
  48. for _, name := range scope.Names() {
  49. o := scope.Lookup(name)
  50. if o == nil || !o.Exported() {
  51. continue
  52. }
  53. st, _ := getTypeStruct(o.Type(), scope)
  54. if st == nil {
  55. continue
  56. }
  57. if name == "PrivateRR" {
  58. continue
  59. }
  60. if scope.Lookup("Type"+o.Name()) == nil && o.Name() != "RFC3597" {
  61. log.Fatalf("Constant Type%s does not exist.", o.Name())
  62. }
  63. for i := 1; i < st.NumFields(); i++ {
  64. if _, ok := st.Field(i).Type().(*types.Slice); ok {
  65. if st.Tag(i) == `dns:"domain-name"` {
  66. domainTypes = append(domainTypes, o.Name())
  67. continue Names
  68. }
  69. if st.Tag(i) == `dns:"cdomain-name"` {
  70. cdomainTypes = append(cdomainTypes, o.Name())
  71. domainTypes = append(domainTypes, o.Name())
  72. continue Names
  73. }
  74. continue
  75. }
  76. switch {
  77. case st.Tag(i) == `dns:"domain-name"`:
  78. domainTypes = append(domainTypes, o.Name())
  79. continue Names
  80. case st.Tag(i) == `dns:"cdomain-name"`:
  81. cdomainTypes = append(cdomainTypes, o.Name())
  82. domainTypes = append(domainTypes, o.Name())
  83. continue Names
  84. }
  85. }
  86. }
  87. b := &bytes.Buffer{}
  88. b.WriteString(packageHdr)
  89. // compressionLenHelperType - all types that have domain-name/cdomain-name can be used for compressing names
  90. fmt.Fprint(b, "func compressionLenHelperType(c map[string]int, r RR, initLen int) int {\n")
  91. fmt.Fprint(b, "currentLen := initLen\n")
  92. fmt.Fprint(b, "switch x := r.(type) {\n")
  93. for _, name := range domainTypes {
  94. o := scope.Lookup(name)
  95. st, _ := getTypeStruct(o.Type(), scope)
  96. fmt.Fprintf(b, "case *%s:\n", name)
  97. for i := 1; i < st.NumFields(); i++ {
  98. out := func(s string) {
  99. fmt.Fprintf(b, "currentLen -= len(x.%s) + 1\n", st.Field(i).Name())
  100. fmt.Fprintf(b, "currentLen += compressionLenHelper(c, x.%s, currentLen)\n", st.Field(i).Name())
  101. }
  102. if _, ok := st.Field(i).Type().(*types.Slice); ok {
  103. switch st.Tag(i) {
  104. case `dns:"domain-name"`:
  105. fallthrough
  106. case `dns:"cdomain-name"`:
  107. // For HIP we need to slice over the elements in this slice.
  108. fmt.Fprintf(b, `for i := range x.%s {
  109. currentLen -= len(x.%s[i]) + 1
  110. }
  111. `, st.Field(i).Name(), st.Field(i).Name())
  112. fmt.Fprintf(b, `for i := range x.%s {
  113. currentLen += compressionLenHelper(c, x.%s[i], currentLen)
  114. }
  115. `, st.Field(i).Name(), st.Field(i).Name())
  116. }
  117. continue
  118. }
  119. switch {
  120. case st.Tag(i) == `dns:"cdomain-name"`:
  121. fallthrough
  122. case st.Tag(i) == `dns:"domain-name"`:
  123. out(st.Field(i).Name())
  124. }
  125. }
  126. }
  127. fmt.Fprintln(b, "}\nreturn currentLen - initLen\n}\n\n")
  128. // compressionLenSearchType - search cdomain-tags types for compressible names.
  129. fmt.Fprint(b, "func compressionLenSearchType(c map[string]int, r RR) (int, bool, int) {\n")
  130. fmt.Fprint(b, "switch x := r.(type) {\n")
  131. for _, name := range cdomainTypes {
  132. o := scope.Lookup(name)
  133. st, _ := getTypeStruct(o.Type(), scope)
  134. fmt.Fprintf(b, "case *%s:\n", name)
  135. j := 1
  136. for i := 1; i < st.NumFields(); i++ {
  137. out := func(s string, j int) {
  138. fmt.Fprintf(b, "k%d, ok%d, sz%d := compressionLenSearch(c, x.%s)\n", j, j, j, st.Field(i).Name())
  139. }
  140. // There are no slice types with names that can be compressed.
  141. switch {
  142. case st.Tag(i) == `dns:"cdomain-name"`:
  143. out(st.Field(i).Name(), j)
  144. j++
  145. }
  146. }
  147. k := "k1"
  148. ok := "ok1"
  149. sz := "sz1"
  150. for i := 2; i < j; i++ {
  151. k += fmt.Sprintf(" + k%d", i)
  152. ok += fmt.Sprintf(" && ok%d", i)
  153. sz += fmt.Sprintf(" + sz%d", i)
  154. }
  155. fmt.Fprintf(b, "return %s, %s, %s\n", k, ok, sz)
  156. }
  157. fmt.Fprintln(b, "}\nreturn 0, false, 0\n}\n\n")
  158. // gofmt
  159. res, err := format.Source(b.Bytes())
  160. if err != nil {
  161. b.WriteTo(os.Stderr)
  162. log.Fatal(err)
  163. }
  164. f, err := os.Create("zcompress.go")
  165. fatalIfErr(err)
  166. defer f.Close()
  167. f.Write(res)
  168. }
  169. func fatalIfErr(err error) {
  170. if err != nil {
  171. log.Fatal(err)
  172. }
  173. }