type_map.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. // +build !gccgo
  2. package reflect2
  3. import (
  4. "reflect"
  5. "runtime"
  6. "strings"
  7. "sync"
  8. "unsafe"
  9. )
  10. // typelinks1 for 1.5 ~ 1.6
  11. //go:linkname typelinks1 reflect.typelinks
  12. func typelinks1() [][]unsafe.Pointer
  13. // typelinks2 for 1.7 ~
  14. //go:linkname typelinks2 reflect.typelinks
  15. func typelinks2() (sections []unsafe.Pointer, offset [][]int32)
  16. // initOnce guards initialization of types and packages
  17. var initOnce sync.Once
  18. var types map[string]reflect.Type
  19. var packages map[string]map[string]reflect.Type
  20. // discoverTypes initializes types and packages
  21. func discoverTypes() {
  22. types = make(map[string]reflect.Type)
  23. packages = make(map[string]map[string]reflect.Type)
  24. ver := runtime.Version()
  25. if ver == "go1.5" || strings.HasPrefix(ver, "go1.5.") {
  26. loadGo15Types()
  27. } else if ver == "go1.6" || strings.HasPrefix(ver, "go1.6.") {
  28. loadGo15Types()
  29. } else {
  30. loadGo17Types()
  31. }
  32. }
  33. func loadGo15Types() {
  34. var obj interface{} = reflect.TypeOf(0)
  35. typePtrss := typelinks1()
  36. for _, typePtrs := range typePtrss {
  37. for _, typePtr := range typePtrs {
  38. (*emptyInterface)(unsafe.Pointer(&obj)).word = typePtr
  39. typ := obj.(reflect.Type)
  40. if typ.Kind() == reflect.Ptr && typ.Elem().Kind() == reflect.Struct {
  41. loadedType := typ.Elem()
  42. pkgTypes := packages[loadedType.PkgPath()]
  43. if pkgTypes == nil {
  44. pkgTypes = map[string]reflect.Type{}
  45. packages[loadedType.PkgPath()] = pkgTypes
  46. }
  47. types[loadedType.String()] = loadedType
  48. pkgTypes[loadedType.Name()] = loadedType
  49. }
  50. if typ.Kind() == reflect.Slice && typ.Elem().Kind() == reflect.Ptr &&
  51. typ.Elem().Elem().Kind() == reflect.Struct {
  52. loadedType := typ.Elem().Elem()
  53. pkgTypes := packages[loadedType.PkgPath()]
  54. if pkgTypes == nil {
  55. pkgTypes = map[string]reflect.Type{}
  56. packages[loadedType.PkgPath()] = pkgTypes
  57. }
  58. types[loadedType.String()] = loadedType
  59. pkgTypes[loadedType.Name()] = loadedType
  60. }
  61. }
  62. }
  63. }
  64. func loadGo17Types() {
  65. var obj interface{} = reflect.TypeOf(0)
  66. sections, offset := typelinks2()
  67. for i, offs := range offset {
  68. rodata := sections[i]
  69. for _, off := range offs {
  70. (*emptyInterface)(unsafe.Pointer(&obj)).word = resolveTypeOff(unsafe.Pointer(rodata), off)
  71. typ := obj.(reflect.Type)
  72. if typ.Kind() == reflect.Ptr && typ.Elem().Kind() == reflect.Struct {
  73. loadedType := typ.Elem()
  74. pkgTypes := packages[loadedType.PkgPath()]
  75. if pkgTypes == nil {
  76. pkgTypes = map[string]reflect.Type{}
  77. packages[loadedType.PkgPath()] = pkgTypes
  78. }
  79. types[loadedType.String()] = loadedType
  80. pkgTypes[loadedType.Name()] = loadedType
  81. }
  82. }
  83. }
  84. }
  85. type emptyInterface struct {
  86. typ unsafe.Pointer
  87. word unsafe.Pointer
  88. }
  89. // TypeByName return the type by its name, just like Class.forName in java
  90. func TypeByName(typeName string) Type {
  91. initOnce.Do(discoverTypes)
  92. return Type2(types[typeName])
  93. }
  94. // TypeByPackageName return the type by its package and name
  95. func TypeByPackageName(pkgPath string, name string) Type {
  96. initOnce.Do(discoverTypes)
  97. pkgTypes := packages[pkgPath]
  98. if pkgTypes == nil {
  99. return nil
  100. }
  101. return Type2(pkgTypes[name])
  102. }