intel_rapl_tpmi.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * intel_rapl_tpmi: Intel RAPL driver via TPMI interface
  4. *
  5. * Copyright (c) 2023, Intel Corporation.
  6. * All Rights Reserved.
  7. *
  8. */
  9. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  10. #include <linux/auxiliary_bus.h>
  11. #include <linux/intel_rapl.h>
  12. #include <linux/intel_tpmi.h>
  13. #include <linux/intel_vsec.h>
  14. #include <linux/io.h>
  15. #include <linux/module.h>
  16. #include <linux/slab.h>
  17. #define TPMI_RAPL_MAJOR_VERSION 0
  18. #define TPMI_RAPL_MINOR_VERSION 1
  19. /* 1 header + 10 registers + 5 reserved. 8 bytes for each. */
  20. #define TPMI_RAPL_DOMAIN_SIZE 128
  21. enum tpmi_rapl_domain_type {
  22. TPMI_RAPL_DOMAIN_INVALID,
  23. TPMI_RAPL_DOMAIN_SYSTEM,
  24. TPMI_RAPL_DOMAIN_PACKAGE,
  25. TPMI_RAPL_DOMAIN_RESERVED,
  26. TPMI_RAPL_DOMAIN_MEMORY,
  27. TPMI_RAPL_DOMAIN_MAX,
  28. };
  29. enum tpmi_rapl_register {
  30. TPMI_RAPL_REG_HEADER,
  31. TPMI_RAPL_REG_UNIT,
  32. TPMI_RAPL_REG_PL1,
  33. TPMI_RAPL_REG_PL2,
  34. TPMI_RAPL_REG_PL3,
  35. TPMI_RAPL_REG_PL4,
  36. TPMI_RAPL_REG_RESERVED,
  37. TPMI_RAPL_REG_ENERGY_STATUS,
  38. TPMI_RAPL_REG_PERF_STATUS,
  39. TPMI_RAPL_REG_POWER_INFO,
  40. TPMI_RAPL_REG_DOMAIN_INFO,
  41. TPMI_RAPL_REG_INTERRUPT,
  42. TPMI_RAPL_REG_MAX = 15,
  43. };
  44. struct tpmi_rapl_package {
  45. struct rapl_if_priv priv;
  46. struct oobmsm_plat_info *tpmi_info;
  47. struct rapl_package *rp;
  48. void __iomem *base;
  49. struct list_head node;
  50. };
  51. static LIST_HEAD(tpmi_rapl_packages);
  52. static DEFINE_MUTEX(tpmi_rapl_lock);
  53. static struct powercap_control_type *tpmi_control_type;
  54. static int tpmi_rapl_read_raw(int id, struct reg_action *ra, bool atomic)
  55. {
  56. if (!ra->reg.mmio)
  57. return -EINVAL;
  58. ra->value = readq(ra->reg.mmio);
  59. ra->value &= ra->mask;
  60. return 0;
  61. }
  62. static int tpmi_rapl_write_raw(int id, struct reg_action *ra)
  63. {
  64. u64 val;
  65. if (!ra->reg.mmio)
  66. return -EINVAL;
  67. val = readq(ra->reg.mmio);
  68. val &= ~ra->mask;
  69. val |= ra->value;
  70. writeq(val, ra->reg.mmio);
  71. return 0;
  72. }
  73. static struct tpmi_rapl_package *trp_alloc(int pkg_id)
  74. {
  75. struct tpmi_rapl_package *trp;
  76. int ret;
  77. mutex_lock(&tpmi_rapl_lock);
  78. if (list_empty(&tpmi_rapl_packages)) {
  79. tpmi_control_type = powercap_register_control_type(NULL, "intel-rapl", NULL);
  80. if (IS_ERR(tpmi_control_type)) {
  81. ret = PTR_ERR(tpmi_control_type);
  82. goto err_unlock;
  83. }
  84. }
  85. trp = kzalloc_obj(*trp);
  86. if (!trp) {
  87. ret = -ENOMEM;
  88. goto err_del_powercap;
  89. }
  90. list_add(&trp->node, &tpmi_rapl_packages);
  91. mutex_unlock(&tpmi_rapl_lock);
  92. return trp;
  93. err_del_powercap:
  94. if (list_empty(&tpmi_rapl_packages))
  95. powercap_unregister_control_type(tpmi_control_type);
  96. err_unlock:
  97. mutex_unlock(&tpmi_rapl_lock);
  98. return ERR_PTR(ret);
  99. }
  100. static void trp_release(struct tpmi_rapl_package *trp)
  101. {
  102. mutex_lock(&tpmi_rapl_lock);
  103. list_del(&trp->node);
  104. if (list_empty(&tpmi_rapl_packages))
  105. powercap_unregister_control_type(tpmi_control_type);
  106. kfree(trp);
  107. mutex_unlock(&tpmi_rapl_lock);
  108. }
  109. /*
  110. * Bit 0 of TPMI_RAPL_REG_DOMAIN_INFO indicates if the current package is a domain
  111. * root or not. Only domain root packages can enumerate System (Psys) Domain.
  112. */
  113. #define TPMI_RAPL_DOMAIN_ROOT BIT(0)
  114. static int parse_one_domain(struct tpmi_rapl_package *trp, u32 offset)
  115. {
  116. u8 tpmi_domain_version;
  117. enum rapl_domain_type domain_type;
  118. enum tpmi_rapl_domain_type tpmi_domain_type;
  119. enum tpmi_rapl_register reg_index;
  120. enum rapl_domain_reg_id reg_id;
  121. int tpmi_domain_size, tpmi_domain_flags;
  122. u64 tpmi_domain_header = readq(trp->base + offset);
  123. u64 tpmi_domain_info;
  124. /* Domain Parent bits are ignored for now */
  125. tpmi_domain_version = tpmi_domain_header & 0xff;
  126. tpmi_domain_type = tpmi_domain_header >> 8 & 0xff;
  127. tpmi_domain_size = tpmi_domain_header >> 16 & 0xff;
  128. tpmi_domain_flags = tpmi_domain_header >> 32 & 0xffff;
  129. if (tpmi_domain_version == TPMI_VERSION_INVALID) {
  130. pr_debug("Invalid version, other instances may be valid\n");
  131. return -ENODEV;
  132. }
  133. if (TPMI_MAJOR_VERSION(tpmi_domain_version) != TPMI_RAPL_MAJOR_VERSION) {
  134. pr_warn(FW_BUG "Unsupported major version:%ld\n",
  135. TPMI_MAJOR_VERSION(tpmi_domain_version));
  136. return -ENODEV;
  137. }
  138. if (TPMI_MINOR_VERSION(tpmi_domain_version) > TPMI_RAPL_MINOR_VERSION)
  139. pr_info("Ignore: Unsupported minor version:%ld\n",
  140. TPMI_MINOR_VERSION(tpmi_domain_version));
  141. /* Domain size: in unit of 128 Bytes */
  142. if (tpmi_domain_size != 1) {
  143. pr_warn(FW_BUG "Invalid Domain size %d\n", tpmi_domain_size);
  144. return -EINVAL;
  145. }
  146. /* Unit register and Energy Status register are mandatory for each domain */
  147. if (!(tpmi_domain_flags & BIT(TPMI_RAPL_REG_UNIT)) ||
  148. !(tpmi_domain_flags & BIT(TPMI_RAPL_REG_ENERGY_STATUS))) {
  149. pr_warn(FW_BUG "Invalid Domain flag 0x%x\n", tpmi_domain_flags);
  150. return -EINVAL;
  151. }
  152. switch (tpmi_domain_type) {
  153. case TPMI_RAPL_DOMAIN_PACKAGE:
  154. domain_type = RAPL_DOMAIN_PACKAGE;
  155. break;
  156. case TPMI_RAPL_DOMAIN_SYSTEM:
  157. if (!(tpmi_domain_flags & BIT(TPMI_RAPL_REG_DOMAIN_INFO))) {
  158. pr_warn(FW_BUG "System domain must support Domain Info register\n");
  159. return -ENODEV;
  160. }
  161. tpmi_domain_info = readq(trp->base + offset + TPMI_RAPL_REG_DOMAIN_INFO * 8);
  162. if (!(tpmi_domain_info & TPMI_RAPL_DOMAIN_ROOT))
  163. return 0;
  164. domain_type = RAPL_DOMAIN_PLATFORM;
  165. break;
  166. case TPMI_RAPL_DOMAIN_MEMORY:
  167. domain_type = RAPL_DOMAIN_DRAM;
  168. break;
  169. default:
  170. pr_warn(FW_BUG "Unsupported Domain type %d\n", tpmi_domain_type);
  171. return -EINVAL;
  172. }
  173. if (trp->priv.regs[domain_type][RAPL_DOMAIN_REG_UNIT].mmio) {
  174. pr_warn(FW_BUG "Duplicate Domain type %d\n", tpmi_domain_type);
  175. return -EINVAL;
  176. }
  177. reg_index = TPMI_RAPL_REG_HEADER;
  178. while (++reg_index != TPMI_RAPL_REG_MAX) {
  179. if (!(tpmi_domain_flags & BIT(reg_index)))
  180. continue;
  181. switch (reg_index) {
  182. case TPMI_RAPL_REG_UNIT:
  183. reg_id = RAPL_DOMAIN_REG_UNIT;
  184. break;
  185. case TPMI_RAPL_REG_PL1:
  186. reg_id = RAPL_DOMAIN_REG_LIMIT;
  187. trp->priv.limits[domain_type] |= BIT(POWER_LIMIT1);
  188. break;
  189. case TPMI_RAPL_REG_PL2:
  190. reg_id = RAPL_DOMAIN_REG_PL2;
  191. trp->priv.limits[domain_type] |= BIT(POWER_LIMIT2);
  192. break;
  193. case TPMI_RAPL_REG_PL4:
  194. reg_id = RAPL_DOMAIN_REG_PL4;
  195. trp->priv.limits[domain_type] |= BIT(POWER_LIMIT4);
  196. break;
  197. case TPMI_RAPL_REG_ENERGY_STATUS:
  198. reg_id = RAPL_DOMAIN_REG_STATUS;
  199. break;
  200. case TPMI_RAPL_REG_PERF_STATUS:
  201. reg_id = RAPL_DOMAIN_REG_PERF;
  202. break;
  203. case TPMI_RAPL_REG_POWER_INFO:
  204. reg_id = RAPL_DOMAIN_REG_INFO;
  205. break;
  206. default:
  207. continue;
  208. }
  209. trp->priv.regs[domain_type][reg_id].mmio = trp->base + offset + reg_index * 8;
  210. }
  211. return 0;
  212. }
  213. static int intel_rapl_tpmi_probe(struct auxiliary_device *auxdev,
  214. const struct auxiliary_device_id *id)
  215. {
  216. struct tpmi_rapl_package *trp;
  217. struct oobmsm_plat_info *info;
  218. struct resource *res;
  219. u32 offset;
  220. int ret;
  221. info = tpmi_get_platform_data(auxdev);
  222. if (!info)
  223. return -ENODEV;
  224. trp = trp_alloc(info->package_id);
  225. if (IS_ERR(trp))
  226. return PTR_ERR(trp);
  227. if (tpmi_get_resource_count(auxdev) > 1) {
  228. dev_err(&auxdev->dev, "does not support multiple resources\n");
  229. ret = -EINVAL;
  230. goto err;
  231. }
  232. res = tpmi_get_resource_at_index(auxdev, 0);
  233. if (!res) {
  234. dev_err(&auxdev->dev, "can't fetch device resource info\n");
  235. ret = -EIO;
  236. goto err;
  237. }
  238. trp->base = devm_ioremap_resource(&auxdev->dev, res);
  239. if (IS_ERR(trp->base)) {
  240. ret = PTR_ERR(trp->base);
  241. goto err;
  242. }
  243. for (offset = 0; offset < resource_size(res); offset += TPMI_RAPL_DOMAIN_SIZE) {
  244. ret = parse_one_domain(trp, offset);
  245. if (ret)
  246. goto err;
  247. }
  248. trp->tpmi_info = info;
  249. trp->priv.type = RAPL_IF_TPMI;
  250. trp->priv.read_raw = tpmi_rapl_read_raw;
  251. trp->priv.write_raw = tpmi_rapl_write_raw;
  252. trp->priv.control_type = tpmi_control_type;
  253. /* RAPL TPMI I/F is per physical package */
  254. trp->rp = rapl_find_package_domain(info->package_id, &trp->priv, false);
  255. if (trp->rp) {
  256. dev_err(&auxdev->dev, "Domain for Package%d already exists\n", info->package_id);
  257. ret = -EEXIST;
  258. goto err;
  259. }
  260. trp->rp = rapl_add_package(info->package_id, &trp->priv, false);
  261. if (IS_ERR(trp->rp)) {
  262. dev_err(&auxdev->dev, "Failed to add RAPL Domain for Package%d, %ld\n",
  263. info->package_id, PTR_ERR(trp->rp));
  264. ret = PTR_ERR(trp->rp);
  265. goto err;
  266. }
  267. rapl_package_add_pmu(trp->rp);
  268. auxiliary_set_drvdata(auxdev, trp);
  269. return 0;
  270. err:
  271. trp_release(trp);
  272. return ret;
  273. }
  274. static void intel_rapl_tpmi_remove(struct auxiliary_device *auxdev)
  275. {
  276. struct tpmi_rapl_package *trp = auxiliary_get_drvdata(auxdev);
  277. rapl_package_remove_pmu(trp->rp);
  278. rapl_remove_package(trp->rp);
  279. trp_release(trp);
  280. }
  281. static const struct auxiliary_device_id intel_rapl_tpmi_ids[] = {
  282. {.name = "intel_vsec.tpmi-rapl" },
  283. { }
  284. };
  285. MODULE_DEVICE_TABLE(auxiliary, intel_rapl_tpmi_ids);
  286. static struct auxiliary_driver intel_rapl_tpmi_driver = {
  287. .probe = intel_rapl_tpmi_probe,
  288. .remove = intel_rapl_tpmi_remove,
  289. .id_table = intel_rapl_tpmi_ids,
  290. };
  291. module_auxiliary_driver(intel_rapl_tpmi_driver)
  292. MODULE_IMPORT_NS("INTEL_TPMI");
  293. MODULE_DESCRIPTION("Intel RAPL TPMI Driver");
  294. MODULE_LICENSE("GPL");