runtime-info.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. package main
  2. import (
  3. "bufio"
  4. "bytes"
  5. "fmt"
  6. "log"
  7. "runtime"
  8. "strings"
  9. )
  10. // reused by GoRuntinesStatus()
  11. var buffer []byte
  12. var GStatus map[string]string = map[string]string{
  13. "idle": "I",
  14. "runnable": "r",
  15. "running": "R",
  16. "syscall": "s",
  17. "waiting": "W",
  18. "dead": "D",
  19. "copystack": "C",
  20. "preempted": "P",
  21. "sleep": "S",
  22. "select": "s", // missing
  23. "chan receive": "<", // missing
  24. "chan send": ">", // missing
  25. }
  26. var output *bytes.Buffer
  27. // This uses the most memory. There has to be a better way.
  28. const DEBUG_THIS bool = false
  29. // Get Go Routines and map the Status. See GStatus.
  30. func GoRoutinesStatus() []byte {
  31. if output == nil {
  32. log.Println("new bytes.Buffer{}")
  33. output = &bytes.Buffer{}
  34. }
  35. output.Reset()
  36. // Map status to letter.
  37. /*
  38. if GStatus == nil {
  39. GStatus = make(map[string]string)
  40. GStatus = map[string]string{
  41. "idle": "I",
  42. "runnable": "r",
  43. "running": "R",
  44. "syscall": "s",
  45. "waiting": "W",
  46. "dead": "D",
  47. "copystack": "C",
  48. "preempted": "P",
  49. "sleep": "S",
  50. "select": "s", // missing
  51. "chan receive": "<", // missing
  52. "chan send": ">", // missing
  53. }
  54. }
  55. */
  56. // 2K works, we don't have to grow any more.
  57. if buffer == nil {
  58. buffer = make([]byte, 2048)
  59. }
  60. read := runtime.Stack(buffer, true)
  61. if read == cap(buffer) {
  62. log.Printf("Increasing buffer from (%d bytes)\n", cap(buffer))
  63. buffer = make([]byte, cap(buffer)+1024)
  64. return GoRoutinesStatus()
  65. }
  66. var buff []byte = buffer[:read]
  67. // fmt.Println(string(buffer[0:read]))
  68. // This is horribly inefficient (memory allocation wise)
  69. var pos, lineend int
  70. var slice []byte
  71. var space, spaceend int
  72. var status []byte
  73. var count int
  74. for pos != -1 {
  75. if DEBUG_THIS {
  76. log.Printf("POS: %d (%d)\n", pos, len(buff))
  77. }
  78. lineend = bytes.Index(buff[pos:], []byte("\n"))
  79. if DEBUG_THIS {
  80. log.Printf("LineEnd: %d\n", lineend)
  81. }
  82. if lineend != -1 {
  83. lineend += pos
  84. slice = buff[pos:lineend]
  85. pos = lineend + 1
  86. } else {
  87. slice = buffer[pos:]
  88. pos = -1
  89. }
  90. // Process line
  91. if bytes.HasPrefix(slice, []byte("goroutine ")) {
  92. if DEBUG_THIS {
  93. log.Printf("GoRoutine %d [%s]\n", count, slice)
  94. count++
  95. }
  96. // Found a gorutine line
  97. space = bytes.Index(slice, []byte{'['})
  98. spaceend = bytes.Index(slice[space+1:], []byte{']'})
  99. if spaceend == -1 {
  100. status = slice[space+1:]
  101. } else {
  102. spaceend += space + 1
  103. // log.Printf("space %d, spaceend %d\n", space, spaceend)
  104. status = slice[space+1 : spaceend]
  105. }
  106. // Ok, status looks like what we're looking for here!
  107. if DEBUG_THIS {
  108. log.Printf("Status: [%s]\n", status)
  109. }
  110. space = bytes.Index(status, []byte{','})
  111. if space != -1 {
  112. status = status[:space]
  113. // log.Printf("Status now: [%s]\n", status)
  114. }
  115. rstatus, ok := GStatus[string(status)]
  116. if ok {
  117. output.WriteString(rstatus)
  118. } else {
  119. log.Printf("** Status [%s] not found.\n", status)
  120. }
  121. }
  122. }
  123. if false {
  124. var reader = bytes.NewReader(buffer[0:read])
  125. var scanner = bufio.NewScanner(reader)
  126. for scanner.Scan() {
  127. var text = scanner.Text()
  128. if strings.HasPrefix(text, "goroutine ") {
  129. // fmt.Println(text)
  130. // goroutine 20 [runnable]:
  131. // goroutine 1 [select, 1 minutes]:
  132. // goroutine 17 [chan receive]:
  133. // Get ID and get Status.
  134. parts := strings.SplitN(text, " ", 3)
  135. /*
  136. gid, err := strconv.Atoi(parts[1])
  137. if err != nil {
  138. continue
  139. }
  140. */
  141. status := parts[2][1 : len(parts[2])-2]
  142. if strings.Contains(status, ",") {
  143. status = strings.Split(status, ",")[0]
  144. }
  145. rstatus, ok := GStatus[status]
  146. if ok {
  147. output.WriteString(rstatus)
  148. } else {
  149. log.Printf("Status %s not found.\n[%s]\n", rstatus, text)
  150. }
  151. }
  152. }
  153. }
  154. return output.Bytes()
  155. }
  156. // Return a nicely formatted string representing memory usage.
  157. func ReprMem(size uint64, output *bytes.Buffer) {
  158. if output == nil {
  159. return
  160. }
  161. output.Reset()
  162. var value float64 = float64(size)
  163. var units byte = ' '
  164. if value > 1024 {
  165. // In KB
  166. units = 'K'
  167. value /= 1024
  168. }
  169. if value > 1024 {
  170. // In MB
  171. units = 'M'
  172. value /= 1024
  173. }
  174. if value > 1024 {
  175. // Im GB
  176. units = 'G'
  177. value /= 1024
  178. }
  179. fmt.Fprintf(output, "%0.2f", value)
  180. if output.Len() > 4 {
  181. output.Reset()
  182. fmt.Fprintf(output, "%0.1f", value)
  183. }
  184. if output.Len() > 4 {
  185. output.Reset()
  186. fmt.Fprintf(output, "%0.0f", value)
  187. }
  188. if units != ' ' {
  189. output.WriteByte(units)
  190. }
  191. }
  192. // Return array of memory usage information
  193. func Memory(result *map[string]*bytes.Buffer) {
  194. var memstats runtime.MemStats
  195. runtime.ReadMemStats(&memstats)
  196. if (*result)["Sys"] == nil {
  197. (*result)["Sys"] = &bytes.Buffer{}
  198. }
  199. ReprMem(memstats.Sys, (*result)["Sys"])
  200. // ReprMem(memstats.HeapAlloc, (*result)["Heap"])
  201. /*
  202. if (*result)["HeapSys"] == nil {
  203. (*result)["HeapSys"] = &bytes.Buffer{}
  204. }
  205. */
  206. if (*result)["Heap"] == nil {
  207. (*result)["Heap"] = &bytes.Buffer{}
  208. }
  209. ReprMem(memstats.HeapAlloc, (*result)["Heap"])
  210. if (*result)["HeapSys"] == nil {
  211. (*result)["HeapSys"] = &bytes.Buffer{}
  212. }
  213. ReprMem(memstats.HeapSys, (*result)["MeapSys"])
  214. if (*result)["StackSys"] == nil {
  215. (*result)["StackSys"] = &bytes.Buffer{}
  216. }
  217. ReprMem(memstats.StackSys, (*result)["StackSys"])
  218. // Don't scale the Number of GCs, it is good as-is.
  219. if (*result)["NumGC"] == nil {
  220. (*result)["NumGC"] = &bytes.Buffer{}
  221. }
  222. (*result)["NumGC"].Reset()
  223. fmt.Fprintf((*result)["NumGC"], "%d", memstats.NumGC)
  224. // ReprMem(uint64(memstats.NumGC), (*result)["NumGC"])
  225. /*
  226. (*result)["HeapInUse"] = ReprMem(memstats.HeapInuse)
  227. (*result)["HeapSys"] = ReprMem(memstats.HeapSys)
  228. (*result)["Sys"] = ReprMem(memstats.Sys)
  229. (*result)["StackInUse"] = ReprMem(memstats.StackInuse)
  230. (*result)["StackSys"] = ReprMem(memstats.StackSys)
  231. (*result)["GCSys"] = ReprMem(memstats.GCSys)
  232. */
  233. }