type_map.go 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. // +build !gccgo
  2. package reflect2
  3. import (
  4. "reflect"
  5. "sync"
  6. "unsafe"
  7. )
  8. // typelinks2 for 1.7 ~
  9. //go:linkname typelinks2 reflect.typelinks
  10. func typelinks2() (sections []unsafe.Pointer, offset [][]int32)
  11. // initOnce guards initialization of types and packages
  12. var initOnce sync.Once
  13. var types map[string]reflect.Type
  14. var packages map[string]map[string]reflect.Type
  15. // discoverTypes initializes types and packages
  16. func discoverTypes() {
  17. types = make(map[string]reflect.Type)
  18. packages = make(map[string]map[string]reflect.Type)
  19. loadGoTypes()
  20. }
  21. func loadGoTypes() {
  22. var obj interface{} = reflect.TypeOf(0)
  23. sections, offset := typelinks2()
  24. for i, offs := range offset {
  25. rodata := sections[i]
  26. for _, off := range offs {
  27. (*emptyInterface)(unsafe.Pointer(&obj)).word = resolveTypeOff(unsafe.Pointer(rodata), off)
  28. typ := obj.(reflect.Type)
  29. if typ.Kind() == reflect.Ptr && typ.Elem().Kind() == reflect.Struct {
  30. loadedType := typ.Elem()
  31. pkgTypes := packages[loadedType.PkgPath()]
  32. if pkgTypes == nil {
  33. pkgTypes = map[string]reflect.Type{}
  34. packages[loadedType.PkgPath()] = pkgTypes
  35. }
  36. types[loadedType.String()] = loadedType
  37. pkgTypes[loadedType.Name()] = loadedType
  38. }
  39. }
  40. }
  41. }
  42. type emptyInterface struct {
  43. typ unsafe.Pointer
  44. word unsafe.Pointer
  45. }
  46. // TypeByName return the type by its name, just like Class.forName in java
  47. func TypeByName(typeName string) Type {
  48. initOnce.Do(discoverTypes)
  49. return Type2(types[typeName])
  50. }
  51. // TypeByPackageName return the type by its package and name
  52. func TypeByPackageName(pkgPath string, name string) Type {
  53. initOnce.Do(discoverTypes)
  54. pkgTypes := packages[pkgPath]
  55. if pkgTypes == nil {
  56. return nil
  57. }
  58. return Type2(pkgTypes[name])
  59. }