go-fieldtrack.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. /* go-fieldtrack.c -- structure field data analysis.
  2. Copyright 2012 The Go Authors. All rights reserved.
  3. Use of this source code is governed by a BSD-style
  4. license that can be found in the LICENSE file. */
  5. #include "runtime.h"
  6. #include "go-type.h"
  7. #include "map.h"
  8. /* The compiler will track fields that have the tag go:"track". Any
  9. function that refers to such a field will call this function with a
  10. string
  11. fieldtrack "package.type.field"
  12. This function does not actually do anything. Instead, we gather
  13. the field tracking information by looking for strings of that form
  14. in the read-only data section. This is, of course, a horrible
  15. hack, but it's good enough for now. We can improve it, e.g., by a
  16. linker plugin, if this turns out to be useful. */
  17. void
  18. __go_fieldtrack (byte *p __attribute__ ((unused)))
  19. {
  20. }
  21. /* A runtime function to add all the tracked fields to a
  22. map[string]bool. */
  23. extern const char _etext[] __attribute__ ((weak));
  24. extern const char __etext[] __attribute__ ((weak));
  25. extern const char __data_start[] __attribute__ ((weak));
  26. extern const char _edata[] __attribute__ ((weak));
  27. extern const char __edata[] __attribute__ ((weak));
  28. extern const char __bss_start[] __attribute__ ((weak));
  29. void runtime_Fieldtrack (struct __go_map *) __asm__ (GOSYM_PREFIX "runtime.Fieldtrack");
  30. void
  31. runtime_Fieldtrack (struct __go_map *m)
  32. {
  33. const char *p;
  34. const char *pend;
  35. const char *prefix;
  36. size_t prefix_len;
  37. p = __data_start;
  38. if (p == NULL)
  39. p = __etext;
  40. if (p == NULL)
  41. p = _etext;
  42. if (p == NULL)
  43. return;
  44. pend = __edata;
  45. if (pend == NULL)
  46. pend = _edata;
  47. if (pend == NULL)
  48. pend = __bss_start;
  49. if (pend == NULL)
  50. return;
  51. prefix = "fieldtrack ";
  52. prefix_len = __builtin_strlen (prefix);
  53. while (p < pend)
  54. {
  55. const char *q1;
  56. const char *q2;
  57. q1 = __builtin_memchr (p + prefix_len, '"', pend - (p + prefix_len));
  58. if (q1 == NULL)
  59. break;
  60. if (__builtin_memcmp (q1 - prefix_len, prefix, prefix_len) != 0)
  61. {
  62. p = q1 + 1;
  63. continue;
  64. }
  65. q1++;
  66. q2 = __builtin_memchr (q1, '"', pend - q1);
  67. if (q2 == NULL)
  68. break;
  69. if (__builtin_memchr (q1, '\0', q2 - q1) == NULL)
  70. {
  71. String s;
  72. void *v;
  73. _Bool *pb;
  74. s.str = (const byte *) q1;
  75. s.len = q2 - q1;
  76. v = __go_map_index (m, &s, 1);
  77. pb = (_Bool *) v;
  78. *pb = 1;
  79. }
  80. p = q2;
  81. }
  82. }