sys_class_drm.c 50 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "plugin_proc.h"
  3. #define PLUGIN_PROC_MODULE_DRM_NAME "/sys/class/drm"
  4. #define CONFIG_SECTION_PLUGIN_PROC_DRM "plugin:proc:/sys/class/drm"
  5. #define AMDGPU_CHART_TYPE "amdgpu"
  6. struct amdgpu_id_struct {
  7. unsigned long long asic_id;
  8. unsigned long long pci_rev_id;
  9. const char *marketing_name;
  10. };
  11. /*
  12. * About amdgpu_ids list:
  13. * ------------------------------------------------------------------------
  14. * Copyright (C) 2023 Advanced Micro Devices, Inc. All rights reserved.
  15. *
  16. * The list is copied from:
  17. * https://raw.githubusercontent.com/Syllo/nvtop/master/src/amdgpu_ids.h
  18. *
  19. * which is modified from libdrm (MIT License):
  20. *
  21. * URL: https://gitlab.freedesktop.org/mesa/drm/-/blob/main/data/amdgpu.ids
  22. * ------------------------------------------------------------------------
  23. * **IMPORTANT**: The amdgpu_ids has to be modified after new GPU releases.
  24. * ------------------------------------------------------------------------*/
  25. static const struct amdgpu_id_struct amdgpu_ids[] = {
  26. {0x1309, 0x00, "AMD Radeon R7 Graphics"},
  27. {0x130A, 0x00, "AMD Radeon R6 Graphics"},
  28. {0x130B, 0x00, "AMD Radeon R4 Graphics"},
  29. {0x130C, 0x00, "AMD Radeon R7 Graphics"},
  30. {0x130D, 0x00, "AMD Radeon R6 Graphics"},
  31. {0x130E, 0x00, "AMD Radeon R5 Graphics"},
  32. {0x130F, 0x00, "AMD Radeon R7 Graphics"},
  33. {0x130F, 0xD4, "AMD Radeon R7 Graphics"},
  34. {0x130F, 0xD5, "AMD Radeon R7 Graphics"},
  35. {0x130F, 0xD6, "AMD Radeon R7 Graphics"},
  36. {0x130F, 0xD7, "AMD Radeon R7 Graphics"},
  37. {0x1313, 0x00, "AMD Radeon R7 Graphics"},
  38. {0x1313, 0xD4, "AMD Radeon R7 Graphics"},
  39. {0x1313, 0xD5, "AMD Radeon R7 Graphics"},
  40. {0x1313, 0xD6, "AMD Radeon R7 Graphics"},
  41. {0x1315, 0x00, "AMD Radeon R5 Graphics"},
  42. {0x1315, 0xD4, "AMD Radeon R5 Graphics"},
  43. {0x1315, 0xD5, "AMD Radeon R5 Graphics"},
  44. {0x1315, 0xD6, "AMD Radeon R5 Graphics"},
  45. {0x1315, 0xD7, "AMD Radeon R5 Graphics"},
  46. {0x1316, 0x00, "AMD Radeon R5 Graphics"},
  47. {0x1318, 0x00, "AMD Radeon R5 Graphics"},
  48. {0x131B, 0x00, "AMD Radeon R4 Graphics"},
  49. {0x131C, 0x00, "AMD Radeon R7 Graphics"},
  50. {0x131D, 0x00, "AMD Radeon R6 Graphics"},
  51. {0x15D8, 0x00, "AMD Radeon RX Vega 8 Graphics WS"},
  52. {0x15D8, 0x91, "AMD Radeon Vega 3 Graphics"},
  53. {0x15D8, 0x91, "AMD Ryzen Embedded R1606G with Radeon Vega Gfx"},
  54. {0x15D8, 0x92, "AMD Radeon Vega 3 Graphics"},
  55. {0x15D8, 0x92, "AMD Ryzen Embedded R1505G with Radeon Vega Gfx"},
  56. {0x15D8, 0x93, "AMD Radeon Vega 1 Graphics"},
  57. {0x15D8, 0xA1, "AMD Radeon Vega 10 Graphics"},
  58. {0x15D8, 0xA2, "AMD Radeon Vega 8 Graphics"},
  59. {0x15D8, 0xA3, "AMD Radeon Vega 6 Graphics"},
  60. {0x15D8, 0xA4, "AMD Radeon Vega 3 Graphics"},
  61. {0x15D8, 0xB1, "AMD Radeon Vega 10 Graphics"},
  62. {0x15D8, 0xB2, "AMD Radeon Vega 8 Graphics"},
  63. {0x15D8, 0xB3, "AMD Radeon Vega 6 Graphics"},
  64. {0x15D8, 0xB4, "AMD Radeon Vega 3 Graphics"},
  65. {0x15D8, 0xC1, "AMD Radeon Vega 10 Graphics"},
  66. {0x15D8, 0xC2, "AMD Radeon Vega 8 Graphics"},
  67. {0x15D8, 0xC3, "AMD Radeon Vega 6 Graphics"},
  68. {0x15D8, 0xC4, "AMD Radeon Vega 3 Graphics"},
  69. {0x15D8, 0xC5, "AMD Radeon Vega 3 Graphics"},
  70. {0x15D8, 0xC8, "AMD Radeon Vega 11 Graphics"},
  71. {0x15D8, 0xC9, "AMD Radeon Vega 8 Graphics"},
  72. {0x15D8, 0xCA, "AMD Radeon Vega 11 Graphics"},
  73. {0x15D8, 0xCB, "AMD Radeon Vega 8 Graphics"},
  74. {0x15D8, 0xCC, "AMD Radeon Vega 3 Graphics"},
  75. {0x15D8, 0xCE, "AMD Radeon Vega 3 Graphics"},
  76. {0x15D8, 0xCF, "AMD Ryzen Embedded R1305G with Radeon Vega Gfx"},
  77. {0x15D8, 0xD1, "AMD Radeon Vega 10 Graphics"},
  78. {0x15D8, 0xD2, "AMD Radeon Vega 8 Graphics"},
  79. {0x15D8, 0xD3, "AMD Radeon Vega 6 Graphics"},
  80. {0x15D8, 0xD4, "AMD Radeon Vega 3 Graphics"},
  81. {0x15D8, 0xD8, "AMD Radeon Vega 11 Graphics"},
  82. {0x15D8, 0xD9, "AMD Radeon Vega 8 Graphics"},
  83. {0x15D8, 0xDA, "AMD Radeon Vega 11 Graphics"},
  84. {0x15D8, 0xDB, "AMD Radeon Vega 3 Graphics"},
  85. {0x15D8, 0xDB, "AMD Radeon Vega 8 Graphics"},
  86. {0x15D8, 0xDC, "AMD Radeon Vega 3 Graphics"},
  87. {0x15D8, 0xDD, "AMD Radeon Vega 3 Graphics"},
  88. {0x15D8, 0xDE, "AMD Radeon Vega 3 Graphics"},
  89. {0x15D8, 0xDF, "AMD Radeon Vega 3 Graphics"},
  90. {0x15D8, 0xE3, "AMD Radeon Vega 3 Graphics"},
  91. {0x15D8, 0xE4, "AMD Ryzen Embedded R1102G with Radeon Vega Gfx"},
  92. {0x15DD, 0x81, "AMD Ryzen Embedded V1807B with Radeon Vega Gfx"},
  93. {0x15DD, 0x82, "AMD Ryzen Embedded V1756B with Radeon Vega Gfx"},
  94. {0x15DD, 0x83, "AMD Ryzen Embedded V1605B with Radeon Vega Gfx"},
  95. {0x15DD, 0x84, "AMD Radeon Vega 6 Graphics"},
  96. {0x15DD, 0x85, "AMD Ryzen Embedded V1202B with Radeon Vega Gfx"},
  97. {0x15DD, 0x86, "AMD Radeon Vega 11 Graphics"},
  98. {0x15DD, 0x88, "AMD Radeon Vega 8 Graphics"},
  99. {0x15DD, 0xC1, "AMD Radeon Vega 11 Graphics"},
  100. {0x15DD, 0xC2, "AMD Radeon Vega 8 Graphics"},
  101. {0x15DD, 0xC3, "AMD Radeon Vega 3 / 10 Graphics"},
  102. {0x15DD, 0xC4, "AMD Radeon Vega 8 Graphics"},
  103. {0x15DD, 0xC5, "AMD Radeon Vega 3 Graphics"},
  104. {0x15DD, 0xC6, "AMD Radeon Vega 11 Graphics"},
  105. {0x15DD, 0xC8, "AMD Radeon Vega 8 Graphics"},
  106. {0x15DD, 0xC9, "AMD Radeon Vega 11 Graphics"},
  107. {0x15DD, 0xCA, "AMD Radeon Vega 8 Graphics"},
  108. {0x15DD, 0xCB, "AMD Radeon Vega 3 Graphics"},
  109. {0x15DD, 0xCC, "AMD Radeon Vega 6 Graphics"},
  110. {0x15DD, 0xCE, "AMD Radeon Vega 3 Graphics"},
  111. {0x15DD, 0xCF, "AMD Radeon Vega 3 Graphics"},
  112. {0x15DD, 0xD0, "AMD Radeon Vega 10 Graphics"},
  113. {0x15DD, 0xD1, "AMD Radeon Vega 8 Graphics"},
  114. {0x15DD, 0xD3, "AMD Radeon Vega 11 Graphics"},
  115. {0x15DD, 0xD5, "AMD Radeon Vega 8 Graphics"},
  116. {0x15DD, 0xD6, "AMD Radeon Vega 11 Graphics"},
  117. {0x15DD, 0xD7, "AMD Radeon Vega 8 Graphics"},
  118. {0x15DD, 0xD8, "AMD Radeon Vega 3 Graphics"},
  119. {0x15DD, 0xD9, "AMD Radeon Vega 6 Graphics"},
  120. {0x15DD, 0xE1, "AMD Radeon Vega 3 Graphics"},
  121. {0x15DD, 0xE2, "AMD Radeon Vega 3 Graphics"},
  122. {0x163F, 0xAE, "AMD Custom GPU 0405"},
  123. {0x6600, 0x00, "AMD Radeon HD 8600 / 8700M"},
  124. {0x6600, 0x81, "AMD Radeon R7 M370"},
  125. {0x6601, 0x00, "AMD Radeon HD 8500M / 8700M"},
  126. {0x6604, 0x00, "AMD Radeon R7 M265 Series"},
  127. {0x6604, 0x81, "AMD Radeon R7 M350"},
  128. {0x6605, 0x00, "AMD Radeon R7 M260 Series"},
  129. {0x6605, 0x81, "AMD Radeon R7 M340"},
  130. {0x6606, 0x00, "AMD Radeon HD 8790M"},
  131. {0x6607, 0x00, "AMD Radeon R5 M240"},
  132. {0x6608, 0x00, "AMD FirePro W2100"},
  133. {0x6610, 0x00, "AMD Radeon R7 200 Series"},
  134. {0x6610, 0x81, "AMD Radeon R7 350"},
  135. {0x6610, 0x83, "AMD Radeon R5 340"},
  136. {0x6610, 0x87, "AMD Radeon R7 200 Series"},
  137. {0x6611, 0x00, "AMD Radeon R7 200 Series"},
  138. {0x6611, 0x87, "AMD Radeon R7 200 Series"},
  139. {0x6613, 0x00, "AMD Radeon R7 200 Series"},
  140. {0x6617, 0x00, "AMD Radeon R7 240 Series"},
  141. {0x6617, 0x87, "AMD Radeon R7 200 Series"},
  142. {0x6617, 0xC7, "AMD Radeon R7 240 Series"},
  143. {0x6640, 0x00, "AMD Radeon HD 8950"},
  144. {0x6640, 0x80, "AMD Radeon R9 M380"},
  145. {0x6646, 0x00, "AMD Radeon R9 M280X"},
  146. {0x6646, 0x80, "AMD Radeon R9 M385"},
  147. {0x6646, 0x80, "AMD Radeon R9 M470X"},
  148. {0x6647, 0x00, "AMD Radeon R9 M200X Series"},
  149. {0x6647, 0x80, "AMD Radeon R9 M380"},
  150. {0x6649, 0x00, "AMD FirePro W5100"},
  151. {0x6658, 0x00, "AMD Radeon R7 200 Series"},
  152. {0x665C, 0x00, "AMD Radeon HD 7700 Series"},
  153. {0x665D, 0x00, "AMD Radeon R7 200 Series"},
  154. {0x665F, 0x81, "AMD Radeon R7 360 Series"},
  155. {0x6660, 0x00, "AMD Radeon HD 8600M Series"},
  156. {0x6660, 0x81, "AMD Radeon R5 M335"},
  157. {0x6660, 0x83, "AMD Radeon R5 M330"},
  158. {0x6663, 0x00, "AMD Radeon HD 8500M Series"},
  159. {0x6663, 0x83, "AMD Radeon R5 M320"},
  160. {0x6664, 0x00, "AMD Radeon R5 M200 Series"},
  161. {0x6665, 0x00, "AMD Radeon R5 M230 Series"},
  162. {0x6665, 0x83, "AMD Radeon R5 M320"},
  163. {0x6665, 0xC3, "AMD Radeon R5 M435"},
  164. {0x6666, 0x00, "AMD Radeon R5 M200 Series"},
  165. {0x6667, 0x00, "AMD Radeon R5 M200 Series"},
  166. {0x666F, 0x00, "AMD Radeon HD 8500M"},
  167. {0x66A1, 0x02, "AMD Instinct MI60 / MI50"},
  168. {0x66A1, 0x06, "AMD Radeon Pro VII"},
  169. {0x66AF, 0xC1, "AMD Radeon VII"},
  170. {0x6780, 0x00, "AMD FirePro W9000"},
  171. {0x6784, 0x00, "ATI FirePro V (FireGL V) Graphics Adapter"},
  172. {0x6788, 0x00, "ATI FirePro V (FireGL V) Graphics Adapter"},
  173. {0x678A, 0x00, "AMD FirePro W8000"},
  174. {0x6798, 0x00, "AMD Radeon R9 200 / HD 7900 Series"},
  175. {0x6799, 0x00, "AMD Radeon HD 7900 Series"},
  176. {0x679A, 0x00, "AMD Radeon HD 7900 Series"},
  177. {0x679B, 0x00, "AMD Radeon HD 7900 Series"},
  178. {0x679E, 0x00, "AMD Radeon HD 7800 Series"},
  179. {0x67A0, 0x00, "AMD Radeon FirePro W9100"},
  180. {0x67A1, 0x00, "AMD Radeon FirePro W8100"},
  181. {0x67B0, 0x00, "AMD Radeon R9 200 Series"},
  182. {0x67B0, 0x80, "AMD Radeon R9 390 Series"},
  183. {0x67B1, 0x00, "AMD Radeon R9 200 Series"},
  184. {0x67B1, 0x80, "AMD Radeon R9 390 Series"},
  185. {0x67B9, 0x00, "AMD Radeon R9 200 Series"},
  186. {0x67C0, 0x00, "AMD Radeon Pro WX 7100 Graphics"},
  187. {0x67C0, 0x80, "AMD Radeon E9550"},
  188. {0x67C2, 0x01, "AMD Radeon Pro V7350x2"},
  189. {0x67C2, 0x02, "AMD Radeon Pro V7300X"},
  190. {0x67C4, 0x00, "AMD Radeon Pro WX 7100 Graphics"},
  191. {0x67C4, 0x80, "AMD Radeon E9560 / E9565 Graphics"},
  192. {0x67C7, 0x00, "AMD Radeon Pro WX 5100 Graphics"},
  193. {0x67C7, 0x80, "AMD Radeon E9390 Graphics"},
  194. {0x67D0, 0x01, "AMD Radeon Pro V7350x2"},
  195. {0x67D0, 0x02, "AMD Radeon Pro V7300X"},
  196. {0x67DF, 0xC0, "AMD Radeon Pro 580X"},
  197. {0x67DF, 0xC1, "AMD Radeon RX 580 Series"},
  198. {0x67DF, 0xC2, "AMD Radeon RX 570 Series"},
  199. {0x67DF, 0xC3, "AMD Radeon RX 580 Series"},
  200. {0x67DF, 0xC4, "AMD Radeon RX 480 Graphics"},
  201. {0x67DF, 0xC5, "AMD Radeon RX 470 Graphics"},
  202. {0x67DF, 0xC6, "AMD Radeon RX 570 Series"},
  203. {0x67DF, 0xC7, "AMD Radeon RX 480 Graphics"},
  204. {0x67DF, 0xCF, "AMD Radeon RX 470 Graphics"},
  205. {0x67DF, 0xD7, "AMD Radeon RX 470 Graphics"},
  206. {0x67DF, 0xE0, "AMD Radeon RX 470 Series"},
  207. {0x67DF, 0xE1, "AMD Radeon RX 590 Series"},
  208. {0x67DF, 0xE3, "AMD Radeon RX Series"},
  209. {0x67DF, 0xE7, "AMD Radeon RX 580 Series"},
  210. {0x67DF, 0xEB, "AMD Radeon Pro 580X"},
  211. {0x67DF, 0xEF, "AMD Radeon RX 570 Series"},
  212. {0x67DF, 0xF7, "AMD Radeon RX P30PH"},
  213. {0x67DF, 0xFF, "AMD Radeon RX 470 Series"},
  214. {0x67E0, 0x00, "AMD Radeon Pro WX Series"},
  215. {0x67E3, 0x00, "AMD Radeon Pro WX 4100"},
  216. {0x67E8, 0x00, "AMD Radeon Pro WX Series"},
  217. {0x67E8, 0x01, "AMD Radeon Pro WX Series"},
  218. {0x67E8, 0x80, "AMD Radeon E9260 Graphics"},
  219. {0x67EB, 0x00, "AMD Radeon Pro V5300X"},
  220. {0x67EF, 0xC0, "AMD Radeon RX Graphics"},
  221. {0x67EF, 0xC1, "AMD Radeon RX 460 Graphics"},
  222. {0x67EF, 0xC2, "AMD Radeon Pro Series"},
  223. {0x67EF, 0xC3, "AMD Radeon RX Series"},
  224. {0x67EF, 0xC5, "AMD Radeon RX 460 Graphics"},
  225. {0x67EF, 0xC7, "AMD Radeon RX Graphics"},
  226. {0x67EF, 0xCF, "AMD Radeon RX 460 Graphics"},
  227. {0x67EF, 0xE0, "AMD Radeon RX 560 Series"},
  228. {0x67EF, 0xE1, "AMD Radeon RX Series"},
  229. {0x67EF, 0xE2, "AMD Radeon RX 560X"},
  230. {0x67EF, 0xE3, "AMD Radeon RX Series"},
  231. {0x67EF, 0xE5, "AMD Radeon RX 560 Series"},
  232. {0x67EF, 0xE7, "AMD Radeon RX 560 Series"},
  233. {0x67EF, 0xEF, "AMD Radeon 550 Series"},
  234. {0x67EF, 0xFF, "AMD Radeon RX 460 Graphics"},
  235. {0x67FF, 0xC0, "AMD Radeon Pro 465"},
  236. {0x67FF, 0xC1, "AMD Radeon RX 560 Series"},
  237. {0x67FF, 0xCF, "AMD Radeon RX 560 Series"},
  238. {0x67FF, 0xEF, "AMD Radeon RX 560 Series"},
  239. {0x67FF, 0xFF, "AMD Radeon RX 550 Series"},
  240. {0x6800, 0x00, "AMD Radeon HD 7970M"},
  241. {0x6801, 0x00, "AMD Radeon HD 8970M"},
  242. {0x6806, 0x00, "AMD Radeon R9 M290X"},
  243. {0x6808, 0x00, "AMD FirePro W7000"},
  244. {0x6808, 0x00, "ATI FirePro V (FireGL V) Graphics Adapter"},
  245. {0x6809, 0x00, "ATI FirePro W5000"},
  246. {0x6810, 0x00, "AMD Radeon R9 200 Series"},
  247. {0x6810, 0x81, "AMD Radeon R9 370 Series"},
  248. {0x6811, 0x00, "AMD Radeon R9 200 Series"},
  249. {0x6811, 0x81, "AMD Radeon R7 370 Series"},
  250. {0x6818, 0x00, "AMD Radeon HD 7800 Series"},
  251. {0x6819, 0x00, "AMD Radeon HD 7800 Series"},
  252. {0x6820, 0x00, "AMD Radeon R9 M275X"},
  253. {0x6820, 0x81, "AMD Radeon R9 M375"},
  254. {0x6820, 0x83, "AMD Radeon R9 M375X"},
  255. {0x6821, 0x00, "AMD Radeon R9 M200X Series"},
  256. {0x6821, 0x83, "AMD Radeon R9 M370X"},
  257. {0x6821, 0x87, "AMD Radeon R7 M380"},
  258. {0x6822, 0x00, "AMD Radeon E8860"},
  259. {0x6823, 0x00, "AMD Radeon R9 M200X Series"},
  260. {0x6825, 0x00, "AMD Radeon HD 7800M Series"},
  261. {0x6826, 0x00, "AMD Radeon HD 7700M Series"},
  262. {0x6827, 0x00, "AMD Radeon HD 7800M Series"},
  263. {0x6828, 0x00, "AMD FirePro W600"},
  264. {0x682B, 0x00, "AMD Radeon HD 8800M Series"},
  265. {0x682B, 0x87, "AMD Radeon R9 M360"},
  266. {0x682C, 0x00, "AMD FirePro W4100"},
  267. {0x682D, 0x00, "AMD Radeon HD 7700M Series"},
  268. {0x682F, 0x00, "AMD Radeon HD 7700M Series"},
  269. {0x6830, 0x00, "AMD Radeon 7800M Series"},
  270. {0x6831, 0x00, "AMD Radeon 7700M Series"},
  271. {0x6835, 0x00, "AMD Radeon R7 Series / HD 9000 Series"},
  272. {0x6837, 0x00, "AMD Radeon HD 7700 Series"},
  273. {0x683D, 0x00, "AMD Radeon HD 7700 Series"},
  274. {0x683F, 0x00, "AMD Radeon HD 7700 Series"},
  275. {0x684C, 0x00, "ATI FirePro V (FireGL V) Graphics Adapter"},
  276. {0x6860, 0x00, "AMD Radeon Instinct MI25"},
  277. {0x6860, 0x01, "AMD Radeon Instinct MI25"},
  278. {0x6860, 0x02, "AMD Radeon Instinct MI25"},
  279. {0x6860, 0x03, "AMD Radeon Pro V340"},
  280. {0x6860, 0x04, "AMD Radeon Instinct MI25x2"},
  281. {0x6860, 0x07, "AMD Radeon Pro V320"},
  282. {0x6861, 0x00, "AMD Radeon Pro WX 9100"},
  283. {0x6862, 0x00, "AMD Radeon Pro SSG"},
  284. {0x6863, 0x00, "AMD Radeon Vega Frontier Edition"},
  285. {0x6864, 0x03, "AMD Radeon Pro V340"},
  286. {0x6864, 0x04, "AMD Radeon Instinct MI25x2"},
  287. {0x6864, 0x05, "AMD Radeon Pro V340"},
  288. {0x6868, 0x00, "AMD Radeon Pro WX 8200"},
  289. {0x686C, 0x00, "AMD Radeon Instinct MI25 MxGPU"},
  290. {0x686C, 0x01, "AMD Radeon Instinct MI25 MxGPU"},
  291. {0x686C, 0x02, "AMD Radeon Instinct MI25 MxGPU"},
  292. {0x686C, 0x03, "AMD Radeon Pro V340 MxGPU"},
  293. {0x686C, 0x04, "AMD Radeon Instinct MI25x2 MxGPU"},
  294. {0x686C, 0x05, "AMD Radeon Pro V340L MxGPU"},
  295. {0x686C, 0x06, "AMD Radeon Instinct MI25 MxGPU"},
  296. {0x687F, 0x01, "AMD Radeon RX Vega"},
  297. {0x687F, 0xC0, "AMD Radeon RX Vega"},
  298. {0x687F, 0xC1, "AMD Radeon RX Vega"},
  299. {0x687F, 0xC3, "AMD Radeon RX Vega"},
  300. {0x687F, 0xC7, "AMD Radeon RX Vega"},
  301. {0x6900, 0x00, "AMD Radeon R7 M260"},
  302. {0x6900, 0x81, "AMD Radeon R7 M360"},
  303. {0x6900, 0x83, "AMD Radeon R7 M340"},
  304. {0x6900, 0xC1, "AMD Radeon R5 M465 Series"},
  305. {0x6900, 0xC3, "AMD Radeon R5 M445 Series"},
  306. {0x6900, 0xD1, "AMD Radeon 530 Series"},
  307. {0x6900, 0xD3, "AMD Radeon 530 Series"},
  308. {0x6901, 0x00, "AMD Radeon R5 M255"},
  309. {0x6902, 0x00, "AMD Radeon Series"},
  310. {0x6907, 0x00, "AMD Radeon R5 M255"},
  311. {0x6907, 0x87, "AMD Radeon R5 M315"},
  312. {0x6920, 0x00, "AMD Radeon R9 M395X"},
  313. {0x6920, 0x01, "AMD Radeon R9 M390X"},
  314. {0x6921, 0x00, "AMD Radeon R9 M390X"},
  315. {0x6929, 0x00, "AMD FirePro S7150"},
  316. {0x6929, 0x01, "AMD FirePro S7100X"},
  317. {0x692B, 0x00, "AMD FirePro W7100"},
  318. {0x6938, 0x00, "AMD Radeon R9 200 Series"},
  319. {0x6938, 0xF0, "AMD Radeon R9 200 Series"},
  320. {0x6938, 0xF1, "AMD Radeon R9 380 Series"},
  321. {0x6939, 0x00, "AMD Radeon R9 200 Series"},
  322. {0x6939, 0xF0, "AMD Radeon R9 200 Series"},
  323. {0x6939, 0xF1, "AMD Radeon R9 380 Series"},
  324. {0x694C, 0xC0, "AMD Radeon RX Vega M GH Graphics"},
  325. {0x694E, 0xC0, "AMD Radeon RX Vega M GL Graphics"},
  326. {0x6980, 0x00, "AMD Radeon Pro WX 3100"},
  327. {0x6981, 0x00, "AMD Radeon Pro WX 3200 Series"},
  328. {0x6981, 0x01, "AMD Radeon Pro WX 3200 Series"},
  329. {0x6981, 0x10, "AMD Radeon Pro WX 3200 Series"},
  330. {0x6985, 0x00, "AMD Radeon Pro WX 3100"},
  331. {0x6986, 0x00, "AMD Radeon Pro WX 2100"},
  332. {0x6987, 0x80, "AMD Embedded Radeon E9171"},
  333. {0x6987, 0xC0, "AMD Radeon 550X Series"},
  334. {0x6987, 0xC1, "AMD Radeon RX 640"},
  335. {0x6987, 0xC3, "AMD Radeon 540X Series"},
  336. {0x6987, 0xC7, "AMD Radeon 540"},
  337. {0x6995, 0x00, "AMD Radeon Pro WX 2100"},
  338. {0x6997, 0x00, "AMD Radeon Pro WX 2100"},
  339. {0x699F, 0x81, "AMD Embedded Radeon E9170 Series"},
  340. {0x699F, 0xC0, "AMD Radeon 500 Series"},
  341. {0x699F, 0xC1, "AMD Radeon 540 Series"},
  342. {0x699F, 0xC3, "AMD Radeon 500 Series"},
  343. {0x699F, 0xC7, "AMD Radeon RX 550 / 550 Series"},
  344. {0x699F, 0xC9, "AMD Radeon 540"},
  345. {0x6FDF, 0xE7, "AMD Radeon RX 590 GME"},
  346. {0x6FDF, 0xEF, "AMD Radeon RX 580 2048SP"},
  347. {0x7300, 0xC1, "AMD FirePro S9300 x2"},
  348. {0x7300, 0xC8, "AMD Radeon R9 Fury Series"},
  349. {0x7300, 0xC9, "AMD Radeon Pro Duo"},
  350. {0x7300, 0xCA, "AMD Radeon R9 Fury Series"},
  351. {0x7300, 0xCB, "AMD Radeon R9 Fury Series"},
  352. {0x7312, 0x00, "AMD Radeon Pro W5700"},
  353. {0x731E, 0xC6, "AMD Radeon RX 5700XTB"},
  354. {0x731E, 0xC7, "AMD Radeon RX 5700B"},
  355. {0x731F, 0xC0, "AMD Radeon RX 5700 XT 50th Anniversary"},
  356. {0x731F, 0xC1, "AMD Radeon RX 5700 XT"},
  357. {0x731F, 0xC2, "AMD Radeon RX 5600M"},
  358. {0x731F, 0xC3, "AMD Radeon RX 5700M"},
  359. {0x731F, 0xC4, "AMD Radeon RX 5700"},
  360. {0x731F, 0xC5, "AMD Radeon RX 5700 XT"},
  361. {0x731F, 0xCA, "AMD Radeon RX 5600 XT"},
  362. {0x731F, 0xCB, "AMD Radeon RX 5600 OEM"},
  363. {0x7340, 0xC1, "AMD Radeon RX 5500M"},
  364. {0x7340, 0xC3, "AMD Radeon RX 5300M"},
  365. {0x7340, 0xC5, "AMD Radeon RX 5500 XT"},
  366. {0x7340, 0xC7, "AMD Radeon RX 5500"},
  367. {0x7340, 0xC9, "AMD Radeon RX 5500XTB"},
  368. {0x7340, 0xCF, "AMD Radeon RX 5300"},
  369. {0x7341, 0x00, "AMD Radeon Pro W5500"},
  370. {0x7347, 0x00, "AMD Radeon Pro W5500M"},
  371. {0x7360, 0x41, "AMD Radeon Pro 5600M"},
  372. {0x7360, 0xC3, "AMD Radeon Pro V520"},
  373. {0x738C, 0x01, "AMD Instinct MI100"},
  374. {0x73A3, 0x00, "AMD Radeon Pro W6800"},
  375. {0x73A5, 0xC0, "AMD Radeon RX 6950 XT"},
  376. {0x73AF, 0xC0, "AMD Radeon RX 6900 XT"},
  377. {0x73BF, 0xC0, "AMD Radeon RX 6900 XT"},
  378. {0x73BF, 0xC1, "AMD Radeon RX 6800 XT"},
  379. {0x73BF, 0xC3, "AMD Radeon RX 6800"},
  380. {0x73DF, 0xC0, "AMD Radeon RX 6750 XT"},
  381. {0x73DF, 0xC1, "AMD Radeon RX 6700 XT"},
  382. {0x73DF, 0xC2, "AMD Radeon RX 6800M"},
  383. {0x73DF, 0xC3, "AMD Radeon RX 6800M"},
  384. {0x73DF, 0xC5, "AMD Radeon RX 6700 XT"},
  385. {0x73DF, 0xCF, "AMD Radeon RX 6700M"},
  386. {0x73DF, 0xD7, "AMD TDC-235"},
  387. {0x73E1, 0x00, "AMD Radeon Pro W6600M"},
  388. {0x73E3, 0x00, "AMD Radeon Pro W6600"},
  389. {0x73EF, 0xC0, "AMD Radeon RX 6800S"},
  390. {0x73EF, 0xC1, "AMD Radeon RX 6650 XT"},
  391. {0x73EF, 0xC2, "AMD Radeon RX 6700S"},
  392. {0x73EF, 0xC3, "AMD Radeon RX 6650M"},
  393. {0x73EF, 0xC4, "AMD Radeon RX 6650M XT"},
  394. {0x73FF, 0xC1, "AMD Radeon RX 6600 XT"},
  395. {0x73FF, 0xC3, "AMD Radeon RX 6600M"},
  396. {0x73FF, 0xC7, "AMD Radeon RX 6600"},
  397. {0x73FF, 0xCB, "AMD Radeon RX 6600S"},
  398. {0x7408, 0x00, "AMD Instinct MI250X"},
  399. {0x740C, 0x01, "AMD Instinct MI250X / MI250"},
  400. {0x740F, 0x02, "AMD Instinct MI210"},
  401. {0x7421, 0x00, "AMD Radeon Pro W6500M"},
  402. {0x7422, 0x00, "AMD Radeon Pro W6400"},
  403. {0x7423, 0x00, "AMD Radeon Pro W6300M"},
  404. {0x7423, 0x01, "AMD Radeon Pro W6300"},
  405. {0x7424, 0x00, "AMD Radeon RX 6300"},
  406. {0x743F, 0xC1, "AMD Radeon RX 6500 XT"},
  407. {0x743F, 0xC3, "AMD Radeon RX 6500"},
  408. {0x743F, 0xC3, "AMD Radeon RX 6500M"},
  409. {0x743F, 0xC7, "AMD Radeon RX 6400"},
  410. {0x743F, 0xCF, "AMD Radeon RX 6300M"},
  411. {0x744C, 0xC8, "AMD Radeon RX 7900 XTX"},
  412. {0x744C, 0xCC, "AMD Radeon RX 7900 XT"},
  413. {0x7480, 0xC1, "AMD Radeon RX 7700S"},
  414. {0x7480, 0xC3, "AMD Radeon RX 7600S"},
  415. {0x7480, 0xC7, "AMD Radeon RX 7600M XT"},
  416. {0x7483, 0xCF, "AMD Radeon RX 7600M"},
  417. {0x9830, 0x00, "AMD Radeon HD 8400 / R3 Series"},
  418. {0x9831, 0x00, "AMD Radeon HD 8400E"},
  419. {0x9832, 0x00, "AMD Radeon HD 8330"},
  420. {0x9833, 0x00, "AMD Radeon HD 8330E"},
  421. {0x9834, 0x00, "AMD Radeon HD 8210"},
  422. {0x9835, 0x00, "AMD Radeon HD 8210E"},
  423. {0x9836, 0x00, "AMD Radeon HD 8200 / R3 Series"},
  424. {0x9837, 0x00, "AMD Radeon HD 8280E"},
  425. {0x9838, 0x00, "AMD Radeon HD 8200 / R3 series"},
  426. {0x9839, 0x00, "AMD Radeon HD 8180"},
  427. {0x983D, 0x00, "AMD Radeon HD 8250"},
  428. {0x9850, 0x00, "AMD Radeon R3 Graphics"},
  429. {0x9850, 0x03, "AMD Radeon R3 Graphics"},
  430. {0x9850, 0x40, "AMD Radeon R2 Graphics"},
  431. {0x9850, 0x45, "AMD Radeon R3 Graphics"},
  432. {0x9851, 0x00, "AMD Radeon R4 Graphics"},
  433. {0x9851, 0x01, "AMD Radeon R5E Graphics"},
  434. {0x9851, 0x05, "AMD Radeon R5 Graphics"},
  435. {0x9851, 0x06, "AMD Radeon R5E Graphics"},
  436. {0x9851, 0x40, "AMD Radeon R4 Graphics"},
  437. {0x9851, 0x45, "AMD Radeon R5 Graphics"},
  438. {0x9852, 0x00, "AMD Radeon R2 Graphics"},
  439. {0x9852, 0x40, "AMD Radeon E1 Graphics"},
  440. {0x9853, 0x00, "AMD Radeon R2 Graphics"},
  441. {0x9853, 0x01, "AMD Radeon R4E Graphics"},
  442. {0x9853, 0x03, "AMD Radeon R2 Graphics"},
  443. {0x9853, 0x05, "AMD Radeon R1E Graphics"},
  444. {0x9853, 0x06, "AMD Radeon R1E Graphics"},
  445. {0x9853, 0x07, "AMD Radeon R1E Graphics"},
  446. {0x9853, 0x08, "AMD Radeon R1E Graphics"},
  447. {0x9853, 0x40, "AMD Radeon R2 Graphics"},
  448. {0x9854, 0x00, "AMD Radeon R3 Graphics"},
  449. {0x9854, 0x01, "AMD Radeon R3E Graphics"},
  450. {0x9854, 0x02, "AMD Radeon R3 Graphics"},
  451. {0x9854, 0x05, "AMD Radeon R2 Graphics"},
  452. {0x9854, 0x06, "AMD Radeon R4 Graphics"},
  453. {0x9854, 0x07, "AMD Radeon R3 Graphics"},
  454. {0x9855, 0x02, "AMD Radeon R6 Graphics"},
  455. {0x9855, 0x05, "AMD Radeon R4 Graphics"},
  456. {0x9856, 0x00, "AMD Radeon R2 Graphics"},
  457. {0x9856, 0x01, "AMD Radeon R2E Graphics"},
  458. {0x9856, 0x02, "AMD Radeon R2 Graphics"},
  459. {0x9856, 0x05, "AMD Radeon R1E Graphics"},
  460. {0x9856, 0x06, "AMD Radeon R2 Graphics"},
  461. {0x9856, 0x07, "AMD Radeon R1E Graphics"},
  462. {0x9856, 0x08, "AMD Radeon R1E Graphics"},
  463. {0x9856, 0x13, "AMD Radeon R1E Graphics"},
  464. {0x9874, 0x81, "AMD Radeon R6 Graphics"},
  465. {0x9874, 0x84, "AMD Radeon R7 Graphics"},
  466. {0x9874, 0x85, "AMD Radeon R6 Graphics"},
  467. {0x9874, 0x87, "AMD Radeon R5 Graphics"},
  468. {0x9874, 0x88, "AMD Radeon R7E Graphics"},
  469. {0x9874, 0x89, "AMD Radeon R6E Graphics"},
  470. {0x9874, 0xC4, "AMD Radeon R7 Graphics"},
  471. {0x9874, 0xC5, "AMD Radeon R6 Graphics"},
  472. {0x9874, 0xC6, "AMD Radeon R6 Graphics"},
  473. {0x9874, 0xC7, "AMD Radeon R5 Graphics"},
  474. {0x9874, 0xC8, "AMD Radeon R7 Graphics"},
  475. {0x9874, 0xC9, "AMD Radeon R7 Graphics"},
  476. {0x9874, 0xCA, "AMD Radeon R5 Graphics"},
  477. {0x9874, 0xCB, "AMD Radeon R5 Graphics"},
  478. {0x9874, 0xCC, "AMD Radeon R7 Graphics"},
  479. {0x9874, 0xCD, "AMD Radeon R7 Graphics"},
  480. {0x9874, 0xCE, "AMD Radeon R5 Graphics"},
  481. {0x9874, 0xE1, "AMD Radeon R7 Graphics"},
  482. {0x9874, 0xE2, "AMD Radeon R7 Graphics"},
  483. {0x9874, 0xE3, "AMD Radeon R7 Graphics"},
  484. {0x9874, 0xE4, "AMD Radeon R7 Graphics"},
  485. {0x9874, 0xE5, "AMD Radeon R5 Graphics"},
  486. {0x9874, 0xE6, "AMD Radeon R5 Graphics"},
  487. {0x98E4, 0x80, "AMD Radeon R5E Graphics"},
  488. {0x98E4, 0x81, "AMD Radeon R4E Graphics"},
  489. {0x98E4, 0x83, "AMD Radeon R2E Graphics"},
  490. {0x98E4, 0x84, "AMD Radeon R2E Graphics"},
  491. {0x98E4, 0x86, "AMD Radeon R1E Graphics"},
  492. {0x98E4, 0xC0, "AMD Radeon R4 Graphics"},
  493. {0x98E4, 0xC1, "AMD Radeon R5 Graphics"},
  494. {0x98E4, 0xC2, "AMD Radeon R4 Graphics"},
  495. {0x98E4, 0xC4, "AMD Radeon R5 Graphics"},
  496. {0x98E4, 0xC6, "AMD Radeon R5 Graphics"},
  497. {0x98E4, 0xC8, "AMD Radeon R4 Graphics"},
  498. {0x98E4, 0xC9, "AMD Radeon R4 Graphics"},
  499. {0x98E4, 0xCA, "AMD Radeon R5 Graphics"},
  500. {0x98E4, 0xD0, "AMD Radeon R2 Graphics"},
  501. {0x98E4, 0xD1, "AMD Radeon R2 Graphics"},
  502. {0x98E4, 0xD2, "AMD Radeon R2 Graphics"},
  503. {0x98E4, 0xD4, "AMD Radeon R2 Graphics"},
  504. {0x98E4, 0xD9, "AMD Radeon R5 Graphics"},
  505. {0x98E4, 0xDA, "AMD Radeon R5 Graphics"},
  506. {0x98E4, 0xDB, "AMD Radeon R3 Graphics"},
  507. {0x98E4, 0xE1, "AMD Radeon R3 Graphics"},
  508. {0x98E4, 0xE2, "AMD Radeon R3 Graphics"},
  509. {0x98E4, 0xE9, "AMD Radeon R4 Graphics"},
  510. {0x98E4, 0xEA, "AMD Radeon R4 Graphics"},
  511. {0x98E4, 0xEB, "AMD Radeon R3 Graphics"},
  512. {0x98E4, 0xEC, "AMD Radeon R4 Graphics"},
  513. {0x0000, 0x00, "unknown AMD GPU"} // this must always be the last item
  514. };
  515. struct card {
  516. const char *pathname;
  517. struct amdgpu_id_struct id;
  518. /* GPU and VRAM utilizations */
  519. const char *pathname_util_gpu;
  520. RRDSET *st_util_gpu;
  521. RRDDIM *rd_util_gpu;
  522. collected_number util_gpu;
  523. const char *pathname_util_mem;
  524. RRDSET *st_util_mem;
  525. RRDDIM *rd_util_mem;
  526. collected_number util_mem;
  527. /* GPU and VRAM clock frequencies */
  528. const char *pathname_clk_gpu;
  529. procfile *ff_clk_gpu;
  530. RRDSET *st_clk_gpu;
  531. RRDDIM *rd_clk_gpu;
  532. collected_number clk_gpu;
  533. const char *pathname_clk_mem;
  534. procfile *ff_clk_mem;
  535. RRDSET *st_clk_mem;
  536. RRDDIM *rd_clk_mem;
  537. collected_number clk_mem;
  538. /* GPU memory usage */
  539. const char *pathname_mem_used_vram;
  540. const char *pathname_mem_total_vram;
  541. RRDSET *st_mem_usage_perc_vram;
  542. RRDDIM *rd_mem_used_perc_vram;
  543. RRDSET *st_mem_usage_vram;
  544. RRDDIM *rd_mem_used_vram;
  545. RRDDIM *rd_mem_free_vram;
  546. collected_number used_vram;
  547. collected_number total_vram;
  548. const char *pathname_mem_used_vis_vram;
  549. const char *pathname_mem_total_vis_vram;
  550. RRDSET *st_mem_usage_perc_vis_vram;
  551. RRDDIM *rd_mem_used_perc_vis_vram;
  552. RRDSET *st_mem_usage_vis_vram;
  553. RRDDIM *rd_mem_used_vis_vram;
  554. RRDDIM *rd_mem_free_vis_vram;
  555. collected_number used_vis_vram;
  556. collected_number total_vis_vram;
  557. const char *pathname_mem_used_gtt;
  558. const char *pathname_mem_total_gtt;
  559. RRDSET *st_mem_usage_perc_gtt;
  560. RRDDIM *rd_mem_used_perc_gtt;
  561. RRDSET *st_mem_usage_gtt;
  562. RRDDIM *rd_mem_used_gtt;
  563. RRDDIM *rd_mem_free_gtt;
  564. collected_number used_gtt;
  565. collected_number total_gtt;
  566. struct do_rrd_x *do_rrd_x_root;
  567. struct card *next;
  568. };
  569. static struct card *card_root = NULL;
  570. static void card_free(struct card *c){
  571. if(c->pathname) freez((void *) c->pathname);
  572. if(c->id.marketing_name) freez((void *) c->id.marketing_name);
  573. /* remove card from linked list */
  574. if(c == card_root) card_root = c->next;
  575. else {
  576. struct card *last;
  577. for(last = card_root; last && last->next != c; last = last->next);
  578. if(last) last->next = c->next;
  579. }
  580. freez(c);
  581. }
  582. static int check_card_is_amdgpu(const char *const pathname){
  583. int rc = -1;
  584. procfile *ff = procfile_open(pathname, " ", PROCFILE_FLAG_NO_ERROR_ON_FILE_IO);
  585. if(unlikely(!ff)){
  586. rc = -1;
  587. goto cleanup;
  588. }
  589. ff = procfile_readall(ff);
  590. if(unlikely(!ff || procfile_lines(ff) < 1 || procfile_linewords(ff, 0) < 1)){
  591. rc = -2;
  592. goto cleanup;
  593. }
  594. for(size_t l = 0; l < procfile_lines(ff); l++) {
  595. if(!strcmp(procfile_lineword(ff, l, 0), "DRIVER=amdgpu")){
  596. rc = 0;
  597. goto cleanup;
  598. }
  599. }
  600. rc = -3; // no match
  601. cleanup:
  602. procfile_close(ff);
  603. return rc;
  604. }
  605. static int read_clk_freq_file(procfile **p_ff, const char *const pathname, collected_number *num){
  606. if(unlikely(!*p_ff)){
  607. *p_ff = procfile_open(pathname, NULL, PROCFILE_FLAG_NO_ERROR_ON_FILE_IO);
  608. if(unlikely(!*p_ff)) return -2;
  609. }
  610. if(unlikely(NULL == (*p_ff = procfile_readall(*p_ff)))) return -3;
  611. for(size_t l = 0; l < procfile_lines(*p_ff) ; l++) {
  612. char *str_with_units = NULL;
  613. if((*p_ff)->lines->lines[l].words >= 3 && !strcmp(procfile_lineword((*p_ff), l, 2), "*")) //format: X: collected_number *
  614. str_with_units = procfile_lineword((*p_ff), l, 1);
  615. else if ((*p_ff)->lines->lines[l].words == 2 && !strcmp(procfile_lineword((*p_ff), l, 1), "*")) //format: collected_number *
  616. str_with_units = procfile_lineword((*p_ff), l, 0);
  617. if (str_with_units) {
  618. char *delim = strchr(str_with_units, 'M');
  619. char str_without_units[10];
  620. memcpy(str_without_units, str_with_units, delim - str_with_units);
  621. *num = str2ll(str_without_units, NULL);
  622. return 0;
  623. }
  624. }
  625. procfile_close((*p_ff));
  626. return -4;
  627. }
  628. static char *set_id(const char *const suf_1, const char *const suf_2, const char *const suf_3){
  629. static char id[RRD_ID_LENGTH_MAX + 1];
  630. snprintfz(id, RRD_ID_LENGTH_MAX, "%s_%s_%s", suf_1, suf_2, suf_3);
  631. return id;
  632. }
  633. typedef int (*do_rrd_x_func)(struct card *const c);
  634. struct do_rrd_x {
  635. do_rrd_x_func func;
  636. struct do_rrd_x *next;
  637. };
  638. static void add_do_rrd_x(struct card *const c, const do_rrd_x_func func){
  639. struct do_rrd_x *const drrd = callocz(1, sizeof(struct do_rrd_x));
  640. drrd->func = func;
  641. drrd->next = c->do_rrd_x_root;
  642. c->do_rrd_x_root = drrd;
  643. }
  644. static void rm_do_rrd_x(struct card *const c, struct do_rrd_x *const drrd){
  645. if(drrd == c->do_rrd_x_root) c->do_rrd_x_root = drrd->next;
  646. else {
  647. struct do_rrd_x *last;
  648. for(last = c->do_rrd_x_root; last && last->next != drrd; last = last->next);
  649. if(last) last->next = drrd->next;
  650. }
  651. freez(drrd);
  652. }
  653. static int do_rrd_util_gpu(struct card *const c){
  654. if(likely(!read_single_number_file(c->pathname_util_gpu, (unsigned long long *) &c->util_gpu))){
  655. rrddim_set_by_pointer(c->st_util_gpu, c->rd_util_gpu, c->util_gpu);
  656. rrdset_done(c->st_util_gpu);
  657. return 0;
  658. }
  659. else {
  660. collector_error("Cannot read util_gpu for %s: [%s]", c->pathname, c->id.marketing_name);
  661. freez((void *) c->pathname_util_gpu);
  662. rrdset_is_obsolete___safe_from_collector_thread(c->st_util_gpu);
  663. return 1;
  664. }
  665. }
  666. static int do_rrd_util_mem(struct card *const c){
  667. if(likely(!read_single_number_file(c->pathname_util_mem, (unsigned long long *) &c->util_mem))){
  668. rrddim_set_by_pointer(c->st_util_mem, c->rd_util_mem, c->util_mem);
  669. rrdset_done(c->st_util_mem);
  670. return 0;
  671. }
  672. else {
  673. collector_error("Cannot read util_mem for %s: [%s]", c->pathname, c->id.marketing_name);
  674. freez((void *) c->pathname_util_mem);
  675. rrdset_is_obsolete___safe_from_collector_thread(c->st_util_mem);
  676. return 1;
  677. }
  678. }
  679. static int do_rrd_clk_gpu(struct card *const c){
  680. if(likely(!read_clk_freq_file(&c->ff_clk_gpu, (char *) c->pathname_clk_gpu, &c->clk_gpu))){
  681. rrddim_set_by_pointer(c->st_clk_gpu, c->rd_clk_gpu, c->clk_gpu);
  682. rrdset_done(c->st_clk_gpu);
  683. return 0;
  684. }
  685. else {
  686. collector_error("Cannot read clk_gpu for %s: [%s]", c->pathname, c->id.marketing_name);
  687. freez((void *) c->pathname_clk_gpu);
  688. rrdset_is_obsolete___safe_from_collector_thread(c->st_clk_gpu);
  689. return 1;
  690. }
  691. }
  692. static int do_rrd_clk_mem(struct card *const c){
  693. if(likely(!read_clk_freq_file(&c->ff_clk_mem, (char *) c->pathname_clk_mem, &c->clk_mem))){
  694. rrddim_set_by_pointer(c->st_clk_mem, c->rd_clk_mem, c->clk_mem);
  695. rrdset_done(c->st_clk_mem);
  696. return 0;
  697. }
  698. else {
  699. collector_error("Cannot read clk_mem for %s: [%s]", c->pathname, c->id.marketing_name);
  700. freez((void *) c->pathname_clk_mem);
  701. rrdset_is_obsolete___safe_from_collector_thread(c->st_clk_mem);
  702. return 1;
  703. }
  704. }
  705. static int do_rrd_vram(struct card *const c){
  706. if(likely(!read_single_number_file(c->pathname_mem_used_vram, (unsigned long long *) &c->used_vram) &&
  707. c->total_vram)){
  708. rrddim_set_by_pointer( c->st_mem_usage_perc_vram,
  709. c->rd_mem_used_perc_vram,
  710. c->used_vram * 10000 / c->total_vram);
  711. rrdset_done(c->st_mem_usage_perc_vram);
  712. rrddim_set_by_pointer(c->st_mem_usage_vram, c->rd_mem_used_vram, c->used_vram);
  713. rrddim_set_by_pointer(c->st_mem_usage_vram, c->rd_mem_free_vram, c->total_vram - c->used_vram);
  714. rrdset_done(c->st_mem_usage_vram);
  715. return 0;
  716. }
  717. else {
  718. collector_error("Cannot read used_vram for %s: [%s]", c->pathname, c->id.marketing_name);
  719. freez((void *) c->pathname_mem_used_vram);
  720. freez((void *) c->pathname_mem_total_vram);
  721. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_perc_vram);
  722. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_vram);
  723. return 1;
  724. }
  725. }
  726. static int do_rrd_vis_vram(struct card *const c){
  727. if(likely(!read_single_number_file(c->pathname_mem_used_vis_vram, (unsigned long long *) &c->used_vis_vram) &&
  728. c->total_vis_vram)){
  729. rrddim_set_by_pointer( c->st_mem_usage_perc_vis_vram,
  730. c->rd_mem_used_perc_vis_vram,
  731. c->used_vis_vram * 10000 / c->total_vis_vram);
  732. rrdset_done(c->st_mem_usage_perc_vis_vram);
  733. rrddim_set_by_pointer(c->st_mem_usage_vis_vram, c->rd_mem_used_vis_vram, c->used_vis_vram);
  734. rrddim_set_by_pointer(c->st_mem_usage_vis_vram, c->rd_mem_free_vis_vram, c->total_vis_vram - c->used_vis_vram);
  735. rrdset_done(c->st_mem_usage_vis_vram);
  736. return 0;
  737. }
  738. else {
  739. collector_error("Cannot read used_vis_vram for %s: [%s]", c->pathname, c->id.marketing_name);
  740. freez((void *) c->pathname_mem_used_vis_vram);
  741. freez((void *) c->pathname_mem_total_vis_vram);
  742. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_perc_vis_vram);
  743. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_vis_vram);
  744. return 1;
  745. }
  746. }
  747. static int do_rrd_gtt(struct card *const c){
  748. if(likely(!read_single_number_file(c->pathname_mem_used_gtt, (unsigned long long *) &c->used_gtt) &&
  749. c->total_gtt)){
  750. rrddim_set_by_pointer( c->st_mem_usage_perc_gtt,
  751. c->rd_mem_used_perc_gtt,
  752. c->used_gtt * 10000 / c->total_gtt);
  753. rrdset_done(c->st_mem_usage_perc_gtt);
  754. rrddim_set_by_pointer(c->st_mem_usage_gtt, c->rd_mem_used_gtt, c->used_gtt);
  755. rrddim_set_by_pointer(c->st_mem_usage_gtt, c->rd_mem_free_gtt, c->total_gtt - c->used_gtt);
  756. rrdset_done(c->st_mem_usage_gtt);
  757. return 0;
  758. }
  759. else {
  760. collector_error("Cannot read used_gtt for %s: [%s]", c->pathname, c->id.marketing_name);
  761. freez((void *) c->pathname_mem_used_gtt);
  762. freez((void *) c->pathname_mem_total_gtt);
  763. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_perc_gtt);
  764. rrdset_is_obsolete___safe_from_collector_thread(c->st_mem_usage_gtt);
  765. return 1;
  766. }
  767. }
  768. int do_sys_class_drm(int update_every, usec_t dt) {
  769. (void)dt;
  770. static DIR *drm_dir = NULL;
  771. int chart_prio = NETDATA_CHART_PRIO_DRM_AMDGPU;
  772. if(unlikely(!drm_dir)) {
  773. char filename[FILENAME_MAX + 1];
  774. snprintfz(filename, FILENAME_MAX, "%s%s", netdata_configured_host_prefix, "/sys/class/drm");
  775. char *drm_dir_name = config_get(CONFIG_SECTION_PLUGIN_PROC_DRM, "directory to monitor", filename);
  776. if(unlikely(NULL == (drm_dir = opendir(drm_dir_name)))){
  777. collector_error("Cannot read directory '%s'", drm_dir_name);
  778. return 1;
  779. }
  780. struct dirent *de = NULL;
  781. while(likely(de = readdir(drm_dir))) {
  782. if( de->d_type == DT_DIR && ((de->d_name[0] == '.' && de->d_name[1] == '\0') ||
  783. (de->d_name[0] == '.' && de->d_name[1] == '.' && de->d_name[2] == '\0'))) continue;
  784. if(de->d_type == DT_LNK && !strncmp(de->d_name, "card", 4) && !strchr(de->d_name, '-')) {
  785. char filename[FILENAME_MAX + 1];
  786. snprintfz(filename, FILENAME_MAX, "%s/%s/%s", drm_dir_name, de->d_name, "device/uevent");
  787. if(check_card_is_amdgpu(filename)) continue;
  788. /* Get static info */
  789. struct card *const c = callocz(1, sizeof(struct card));
  790. snprintfz(filename, FILENAME_MAX, "%s/%s", drm_dir_name, de->d_name);
  791. c->pathname = strdupz(filename);
  792. snprintfz(filename, FILENAME_MAX, "%s/%s", c->pathname, "device/device");
  793. if(read_single_base64_or_hex_number_file(filename, &c->id.asic_id)){
  794. collector_error("Cannot read asic_id from '%s'", filename);
  795. card_free(c);
  796. continue;
  797. }
  798. snprintfz(filename, FILENAME_MAX, "%s/%s", c->pathname, "device/revision");
  799. if(read_single_base64_or_hex_number_file(filename, &c->id.pci_rev_id)){
  800. collector_error("Cannot read pci_rev_id from '%s'", filename);
  801. card_free(c);
  802. continue;
  803. }
  804. for(int i = 0; amdgpu_ids[i].asic_id; i++){
  805. if(c->id.asic_id == amdgpu_ids[i].asic_id && c->id.pci_rev_id == amdgpu_ids[i].pci_rev_id){
  806. c->id.marketing_name = strdupz(amdgpu_ids[i].marketing_name);
  807. break;
  808. }
  809. }
  810. if(!c->id.marketing_name)
  811. c->id.marketing_name = strdupz(amdgpu_ids[sizeof(amdgpu_ids)/sizeof(amdgpu_ids[0]) - 1].marketing_name);
  812. collected_number tmp_val;
  813. #define set_prop_pathname(prop_filename, prop_pathname, p_ff){ \
  814. snprintfz(filename, FILENAME_MAX, "%s/%s", c->pathname, prop_filename); \
  815. if((p_ff && !read_clk_freq_file(p_ff, filename, &tmp_val)) || \
  816. !read_single_number_file(filename, (unsigned long long *) &tmp_val)) \
  817. prop_pathname = strdupz(filename); \
  818. else \
  819. collector_info("Cannot read file '%s'", filename); \
  820. }
  821. /* Initialize GPU and VRAM utilization metrics */
  822. set_prop_pathname("device/gpu_busy_percent", c->pathname_util_gpu, NULL);
  823. if(c->pathname_util_gpu){
  824. c->st_util_gpu = rrdset_create_localhost(
  825. AMDGPU_CHART_TYPE
  826. , set_id("gpu_utilization", c->id.marketing_name, de->d_name)
  827. , NULL
  828. , "utilization"
  829. , AMDGPU_CHART_TYPE ".gpu_utilization"
  830. , "GPU utilization"
  831. , "percentage"
  832. , PLUGIN_PROC_NAME
  833. , PLUGIN_PROC_MODULE_DRM_NAME
  834. , chart_prio++
  835. , update_every
  836. , RRDSET_TYPE_LINE
  837. );
  838. rrdlabels_add(c->st_util_gpu->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  839. c->rd_util_gpu = rrddim_add(c->st_util_gpu, "utilization", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  840. add_do_rrd_x(c, do_rrd_util_gpu);
  841. }
  842. set_prop_pathname("device/mem_busy_percent", c->pathname_util_mem, NULL);
  843. if(c->pathname_util_mem){
  844. c->st_util_mem = rrdset_create_localhost(
  845. AMDGPU_CHART_TYPE
  846. , set_id("gpu_mem_utilization", c->id.marketing_name, de->d_name)
  847. , NULL
  848. , "utilization"
  849. , AMDGPU_CHART_TYPE ".gpu_mem_utilization"
  850. , "GPU memory utilization"
  851. , "percentage"
  852. , PLUGIN_PROC_NAME
  853. , PLUGIN_PROC_MODULE_DRM_NAME
  854. , chart_prio++
  855. , update_every
  856. , RRDSET_TYPE_LINE
  857. );
  858. rrdlabels_add(c->st_util_mem->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  859. c->rd_util_mem = rrddim_add(c->st_util_mem, "utilization", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  860. add_do_rrd_x(c, do_rrd_util_mem);
  861. }
  862. /* Initialize GPU and VRAM clock frequency metrics */
  863. set_prop_pathname("device/pp_dpm_sclk", c->pathname_clk_gpu, &c->ff_clk_gpu);
  864. if(c->pathname_clk_gpu){
  865. c->st_clk_gpu = rrdset_create_localhost(
  866. AMDGPU_CHART_TYPE
  867. , set_id("gpu_clk_frequency", c->id.marketing_name, de->d_name)
  868. , NULL
  869. , "frequency"
  870. , AMDGPU_CHART_TYPE ".gpu_clk_frequency"
  871. , "GPU clock frequency"
  872. , "MHz"
  873. , PLUGIN_PROC_NAME
  874. , PLUGIN_PROC_MODULE_DRM_NAME
  875. , chart_prio++
  876. , update_every
  877. , RRDSET_TYPE_LINE
  878. );
  879. rrdlabels_add(c->st_clk_gpu->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  880. c->rd_clk_gpu = rrddim_add(c->st_clk_gpu, "frequency", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  881. add_do_rrd_x(c, do_rrd_clk_gpu);
  882. }
  883. set_prop_pathname("device/pp_dpm_mclk", c->pathname_clk_mem, &c->ff_clk_mem);
  884. if(c->pathname_clk_mem){
  885. c->st_clk_mem = rrdset_create_localhost(
  886. AMDGPU_CHART_TYPE
  887. , set_id("gpu_mem_clk_frequency", c->id.marketing_name, de->d_name)
  888. , NULL
  889. , "frequency"
  890. , AMDGPU_CHART_TYPE ".gpu_mem_clk_frequency"
  891. , "GPU memory clock frequency"
  892. , "MHz"
  893. , PLUGIN_PROC_NAME
  894. , PLUGIN_PROC_MODULE_DRM_NAME
  895. , chart_prio++
  896. , update_every
  897. , RRDSET_TYPE_LINE
  898. );
  899. rrdlabels_add(c->st_clk_mem->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  900. c->rd_clk_mem = rrddim_add(c->st_clk_mem, "frequency", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  901. add_do_rrd_x(c, do_rrd_clk_mem);
  902. }
  903. /* Initialize GPU memory usage metrics */
  904. set_prop_pathname("device/mem_info_vram_used", c->pathname_mem_used_vram, NULL);
  905. set_prop_pathname("device/mem_info_vram_total", c->pathname_mem_total_vram, NULL);
  906. if(c->pathname_mem_total_vram) c->total_vram = tmp_val;
  907. if(c->pathname_mem_used_vram && c->pathname_mem_total_vram){
  908. c->st_mem_usage_perc_vram = rrdset_create_localhost(
  909. AMDGPU_CHART_TYPE
  910. , set_id("gpu_mem_vram_usage_perc", c->id.marketing_name, de->d_name)
  911. , NULL
  912. , "memory_usage"
  913. , AMDGPU_CHART_TYPE ".gpu_mem_vram_usage_perc"
  914. , "VRAM memory usage percentage"
  915. , "percentage"
  916. , PLUGIN_PROC_NAME
  917. , PLUGIN_PROC_MODULE_DRM_NAME
  918. , chart_prio++
  919. , update_every
  920. , RRDSET_TYPE_LINE
  921. );
  922. rrdlabels_add(c->st_mem_usage_perc_vram->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  923. c->rd_mem_used_perc_vram = rrddim_add(c->st_mem_usage_perc_vram, "usage", NULL, 1, 100, RRD_ALGORITHM_ABSOLUTE);
  924. c->st_mem_usage_vram = rrdset_create_localhost(
  925. AMDGPU_CHART_TYPE
  926. , set_id("gpu_mem_vram_usage", c->id.marketing_name, de->d_name)
  927. , NULL
  928. , "memory_usage"
  929. , AMDGPU_CHART_TYPE ".gpu_mem_vram_usage"
  930. , "VRAM memory usage"
  931. , "bytes"
  932. , PLUGIN_PROC_NAME
  933. , PLUGIN_PROC_MODULE_DRM_NAME
  934. , chart_prio++
  935. , update_every
  936. , RRDSET_TYPE_STACKED
  937. );
  938. rrdlabels_add(c->st_mem_usage_vram->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  939. c->rd_mem_free_vram = rrddim_add(c->st_mem_usage_vram, "free", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  940. c->rd_mem_used_vram = rrddim_add(c->st_mem_usage_vram, "used", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  941. add_do_rrd_x(c, do_rrd_vram);
  942. }
  943. set_prop_pathname("device/mem_info_vis_vram_used", c->pathname_mem_used_vis_vram, NULL);
  944. set_prop_pathname("device/mem_info_vis_vram_total", c->pathname_mem_total_vis_vram, NULL);
  945. if(c->pathname_mem_total_vis_vram) c->total_vis_vram = tmp_val;
  946. if(c->pathname_mem_used_vis_vram && c->pathname_mem_total_vis_vram){
  947. c->st_mem_usage_perc_vis_vram = rrdset_create_localhost(
  948. AMDGPU_CHART_TYPE
  949. , set_id("gpu_mem_vis_vram_usage_perc", c->id.marketing_name, de->d_name)
  950. , NULL
  951. , "memory_usage"
  952. , AMDGPU_CHART_TYPE ".gpu_mem_vis_vram_usage_perc"
  953. , "visible VRAM memory usage percentage"
  954. , "percentage"
  955. , PLUGIN_PROC_NAME
  956. , PLUGIN_PROC_MODULE_DRM_NAME
  957. , chart_prio++
  958. , update_every
  959. , RRDSET_TYPE_LINE
  960. );
  961. rrdlabels_add(c->st_mem_usage_perc_vis_vram->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  962. c->rd_mem_used_perc_vis_vram = rrddim_add(c->st_mem_usage_perc_vis_vram, "usage", NULL, 1, 100, RRD_ALGORITHM_ABSOLUTE);
  963. c->st_mem_usage_vis_vram = rrdset_create_localhost(
  964. AMDGPU_CHART_TYPE
  965. , set_id("gpu_mem_vis_vram_usage", c->id.marketing_name, de->d_name)
  966. , NULL
  967. , "memory_usage"
  968. , AMDGPU_CHART_TYPE ".gpu_mem_vis_vram_usage"
  969. , "visible VRAM memory usage"
  970. , "bytes"
  971. , PLUGIN_PROC_NAME
  972. , PLUGIN_PROC_MODULE_DRM_NAME
  973. , chart_prio++
  974. , update_every
  975. , RRDSET_TYPE_STACKED
  976. );
  977. rrdlabels_add(c->st_mem_usage_vis_vram->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  978. c->rd_mem_free_vis_vram = rrddim_add(c->st_mem_usage_vis_vram, "free", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  979. c->rd_mem_used_vis_vram = rrddim_add(c->st_mem_usage_vis_vram, "used", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  980. add_do_rrd_x(c, do_rrd_vis_vram);
  981. }
  982. set_prop_pathname("device/mem_info_gtt_used", c->pathname_mem_used_gtt, NULL);
  983. set_prop_pathname("device/mem_info_gtt_total", c->pathname_mem_total_gtt, NULL);
  984. if(c->pathname_mem_total_gtt) c->total_gtt = tmp_val;
  985. if(c->pathname_mem_used_gtt && c->pathname_mem_total_gtt){
  986. c->st_mem_usage_perc_gtt = rrdset_create_localhost(
  987. AMDGPU_CHART_TYPE
  988. , set_id("gpu_mem_gtt_usage_perc", c->id.marketing_name, de->d_name)
  989. , NULL
  990. , "memory_usage"
  991. , AMDGPU_CHART_TYPE ".gpu_mem_gtt_usage_perc"
  992. , "GTT memory usage percentage"
  993. , "percentage"
  994. , PLUGIN_PROC_NAME
  995. , PLUGIN_PROC_MODULE_DRM_NAME
  996. , chart_prio++
  997. , update_every
  998. , RRDSET_TYPE_LINE
  999. );
  1000. rrdlabels_add(c->st_mem_usage_perc_gtt->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  1001. c->rd_mem_used_perc_gtt = rrddim_add(c->st_mem_usage_perc_gtt, "usage", NULL, 1, 100, RRD_ALGORITHM_ABSOLUTE);
  1002. c->st_mem_usage_gtt = rrdset_create_localhost(
  1003. AMDGPU_CHART_TYPE
  1004. , set_id("gpu_mem_gtt_usage", c->id.marketing_name, de->d_name)
  1005. , NULL
  1006. , "memory_usage"
  1007. , AMDGPU_CHART_TYPE ".gpu_mem_gtt_usage"
  1008. , "GTT memory usage"
  1009. , "bytes"
  1010. , PLUGIN_PROC_NAME
  1011. , PLUGIN_PROC_MODULE_DRM_NAME
  1012. , chart_prio++
  1013. , update_every
  1014. , RRDSET_TYPE_STACKED
  1015. );
  1016. rrdlabels_add(c->st_mem_usage_gtt->rrdlabels, "product_name", c->id.marketing_name, RRDLABEL_SRC_AUTO);
  1017. c->rd_mem_free_gtt = rrddim_add(c->st_mem_usage_gtt, "free", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  1018. c->rd_mem_used_gtt = rrddim_add(c->st_mem_usage_gtt, "used", NULL, 1, 1, RRD_ALGORITHM_ABSOLUTE);
  1019. add_do_rrd_x(c, do_rrd_gtt);
  1020. }
  1021. c->next = card_root;
  1022. card_root = c;
  1023. }
  1024. }
  1025. }
  1026. struct card *card_cur = card_root,
  1027. *card_next;
  1028. while(card_cur){
  1029. struct do_rrd_x *do_rrd_x_cur = card_cur->do_rrd_x_root,
  1030. *do_rrd_x_next;
  1031. while(do_rrd_x_cur){
  1032. if(unlikely(do_rrd_x_cur->func(card_cur))) {
  1033. do_rrd_x_next = do_rrd_x_cur->next;
  1034. rm_do_rrd_x(card_cur, do_rrd_x_cur);
  1035. do_rrd_x_cur = do_rrd_x_next;
  1036. }
  1037. else do_rrd_x_cur = do_rrd_x_cur->next;
  1038. }
  1039. if(unlikely(!card_cur->do_rrd_x_root)){
  1040. card_next = card_cur->next;
  1041. card_free(card_cur);
  1042. card_cur = card_next;
  1043. }
  1044. else card_cur = card_cur->next;
  1045. }
  1046. return card_root ? 0 : 1;
  1047. }