esrt.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * esrt.c
  4. *
  5. * This module exports EFI System Resource Table (ESRT) entries into userspace
  6. * through the sysfs file system. The ESRT provides a read-only catalog of
  7. * system components for which the system accepts firmware upgrades via UEFI's
  8. * "Capsule Update" feature. This module allows userland utilities to evaluate
  9. * what firmware updates can be applied to this system, and potentially arrange
  10. * for those updates to occur.
  11. *
  12. * Data is currently found below /sys/firmware/efi/esrt/...
  13. */
  14. #define pr_fmt(fmt) "esrt: " fmt
  15. #include <linux/capability.h>
  16. #include <linux/device.h>
  17. #include <linux/efi.h>
  18. #include <linux/init.h>
  19. #include <linux/io.h>
  20. #include <linux/kernel.h>
  21. #include <linux/kobject.h>
  22. #include <linux/list.h>
  23. #include <linux/memblock.h>
  24. #include <linux/slab.h>
  25. #include <linux/types.h>
  26. #include <asm/io.h>
  27. #include <asm/early_ioremap.h>
  28. struct efi_system_resource_entry_v1 {
  29. efi_guid_t fw_class;
  30. u32 fw_type;
  31. u32 fw_version;
  32. u32 lowest_supported_fw_version;
  33. u32 capsule_flags;
  34. u32 last_attempt_version;
  35. u32 last_attempt_status;
  36. };
  37. /*
  38. * _count and _version are what they seem like. _max is actually just
  39. * accounting info for the firmware when creating the table; it should never
  40. * have been exposed to us. To wit, the spec says:
  41. * The maximum number of resource array entries that can be within the
  42. * table without reallocating the table, must not be zero.
  43. * Since there's no guidance about what that means in terms of memory layout,
  44. * it means nothing to us.
  45. */
  46. struct efi_system_resource_table {
  47. u32 fw_resource_count;
  48. u32 fw_resource_count_max;
  49. u64 fw_resource_version;
  50. u8 entries[];
  51. };
  52. static phys_addr_t esrt_data;
  53. static size_t esrt_data_size;
  54. static struct efi_system_resource_table *esrt;
  55. struct esre_entry {
  56. union {
  57. struct efi_system_resource_entry_v1 *esre1;
  58. } esre;
  59. struct kobject kobj;
  60. struct list_head list;
  61. };
  62. /* global list of esre_entry. */
  63. static LIST_HEAD(entry_list);
  64. /* entry attribute */
  65. struct esre_attribute {
  66. struct attribute attr;
  67. ssize_t (*show)(struct esre_entry *entry, char *buf);
  68. };
  69. static struct esre_entry *to_entry(struct kobject *kobj)
  70. {
  71. return container_of(kobj, struct esre_entry, kobj);
  72. }
  73. static struct esre_attribute *to_attr(struct attribute *attr)
  74. {
  75. return container_of(attr, struct esre_attribute, attr);
  76. }
  77. static ssize_t esre_attr_show(struct kobject *kobj,
  78. struct attribute *_attr, char *buf)
  79. {
  80. struct esre_entry *entry = to_entry(kobj);
  81. struct esre_attribute *attr = to_attr(_attr);
  82. return attr->show(entry, buf);
  83. }
  84. static const struct sysfs_ops esre_attr_ops = {
  85. .show = esre_attr_show,
  86. };
  87. /* Generic ESRT Entry ("ESRE") support. */
  88. static ssize_t fw_class_show(struct esre_entry *entry, char *buf)
  89. {
  90. char *str = buf;
  91. efi_guid_to_str(&entry->esre.esre1->fw_class, str);
  92. str += strlen(str);
  93. str += sprintf(str, "\n");
  94. return str - buf;
  95. }
  96. static struct esre_attribute esre_fw_class = __ATTR_RO_MODE(fw_class, 0400);
  97. #define esre_attr_decl(name, size, fmt) \
  98. static ssize_t name##_show(struct esre_entry *entry, char *buf) \
  99. { \
  100. return sprintf(buf, fmt "\n", \
  101. le##size##_to_cpu(entry->esre.esre1->name)); \
  102. } \
  103. \
  104. static struct esre_attribute esre_##name = __ATTR_RO_MODE(name, 0400)
  105. esre_attr_decl(fw_type, 32, "%u");
  106. esre_attr_decl(fw_version, 32, "%u");
  107. esre_attr_decl(lowest_supported_fw_version, 32, "%u");
  108. esre_attr_decl(capsule_flags, 32, "0x%x");
  109. esre_attr_decl(last_attempt_version, 32, "%u");
  110. esre_attr_decl(last_attempt_status, 32, "%u");
  111. static struct attribute *esre1_attrs[] = {
  112. &esre_fw_class.attr,
  113. &esre_fw_type.attr,
  114. &esre_fw_version.attr,
  115. &esre_lowest_supported_fw_version.attr,
  116. &esre_capsule_flags.attr,
  117. &esre_last_attempt_version.attr,
  118. &esre_last_attempt_status.attr,
  119. NULL
  120. };
  121. ATTRIBUTE_GROUPS(esre1);
  122. static void esre_release(struct kobject *kobj)
  123. {
  124. struct esre_entry *entry = to_entry(kobj);
  125. list_del(&entry->list);
  126. kfree(entry);
  127. }
  128. static const struct kobj_type esre1_ktype = {
  129. .release = esre_release,
  130. .sysfs_ops = &esre_attr_ops,
  131. .default_groups = esre1_groups,
  132. };
  133. static struct kobject *esrt_kobj;
  134. static struct kset *esrt_kset;
  135. static int esre_create_sysfs_entry(void *esre, int entry_num)
  136. {
  137. struct esre_entry *entry;
  138. entry = kzalloc_obj(*entry);
  139. if (!entry)
  140. return -ENOMEM;
  141. entry->kobj.kset = esrt_kset;
  142. if (esrt->fw_resource_version == 1) {
  143. int rc = 0;
  144. entry->esre.esre1 = esre;
  145. rc = kobject_init_and_add(&entry->kobj, &esre1_ktype, NULL,
  146. "entry%d", entry_num);
  147. if (rc) {
  148. kobject_put(&entry->kobj);
  149. return rc;
  150. }
  151. }
  152. list_add_tail(&entry->list, &entry_list);
  153. return 0;
  154. }
  155. /* support for displaying ESRT fields at the top level */
  156. #define esrt_attr_decl(name, size, fmt) \
  157. static ssize_t name##_show(struct kobject *kobj, \
  158. struct kobj_attribute *attr, char *buf)\
  159. { \
  160. return sprintf(buf, fmt "\n", le##size##_to_cpu(esrt->name)); \
  161. } \
  162. \
  163. static struct kobj_attribute esrt_##name = __ATTR_RO_MODE(name, 0400)
  164. esrt_attr_decl(fw_resource_count, 32, "%u");
  165. esrt_attr_decl(fw_resource_count_max, 32, "%u");
  166. esrt_attr_decl(fw_resource_version, 64, "%llu");
  167. static struct attribute *esrt_attrs[] = {
  168. &esrt_fw_resource_count.attr,
  169. &esrt_fw_resource_count_max.attr,
  170. &esrt_fw_resource_version.attr,
  171. NULL,
  172. };
  173. static inline int esrt_table_exists(void)
  174. {
  175. if (!efi_enabled(EFI_CONFIG_TABLES))
  176. return 0;
  177. if (efi.esrt == EFI_INVALID_TABLE_ADDR)
  178. return 0;
  179. return 1;
  180. }
  181. static umode_t esrt_attr_is_visible(struct kobject *kobj,
  182. struct attribute *attr, int n)
  183. {
  184. if (!esrt_table_exists())
  185. return 0;
  186. return attr->mode;
  187. }
  188. static const struct attribute_group esrt_attr_group = {
  189. .attrs = esrt_attrs,
  190. .is_visible = esrt_attr_is_visible,
  191. };
  192. /*
  193. * remap the table, validate it, mark it reserved and unmap it.
  194. */
  195. void __init efi_esrt_init(void)
  196. {
  197. void *va;
  198. struct efi_system_resource_table tmpesrt;
  199. size_t size, max, entry_size, entries_size;
  200. efi_memory_desc_t md;
  201. int rc;
  202. phys_addr_t end;
  203. if (!efi_enabled(EFI_MEMMAP) && !efi_enabled(EFI_PARAVIRT))
  204. return;
  205. pr_debug("esrt-init: loading.\n");
  206. if (!esrt_table_exists())
  207. return;
  208. rc = efi_mem_desc_lookup(efi.esrt, &md);
  209. if (rc < 0 ||
  210. (!(md.attribute & EFI_MEMORY_RUNTIME) &&
  211. md.type != EFI_BOOT_SERVICES_DATA &&
  212. md.type != EFI_RUNTIME_SERVICES_DATA &&
  213. md.type != EFI_ACPI_RECLAIM_MEMORY &&
  214. md.type != EFI_ACPI_MEMORY_NVS)) {
  215. pr_warn("ESRT header is not in the memory map.\n");
  216. return;
  217. }
  218. max = efi_mem_desc_end(&md) - efi.esrt;
  219. size = sizeof(*esrt);
  220. if (max < size) {
  221. pr_err("ESRT header doesn't fit on single memory map entry. (size: %zu max: %zu)\n",
  222. size, max);
  223. return;
  224. }
  225. va = early_memremap(efi.esrt, size);
  226. if (!va) {
  227. pr_err("early_memremap(%p, %zu) failed.\n", (void *)efi.esrt,
  228. size);
  229. return;
  230. }
  231. memcpy(&tmpesrt, va, sizeof(tmpesrt));
  232. early_memunmap(va, size);
  233. if (tmpesrt.fw_resource_version != 1) {
  234. pr_err("Unsupported ESRT version %lld.\n",
  235. tmpesrt.fw_resource_version);
  236. return;
  237. }
  238. entry_size = sizeof(struct efi_system_resource_entry_v1);
  239. if (tmpesrt.fw_resource_count > 0 && max - size < entry_size) {
  240. pr_err("ESRT memory map entry can only hold the header. (max: %zu size: %zu)\n",
  241. max - size, entry_size);
  242. return;
  243. }
  244. /*
  245. * The format doesn't really give us any boundary to test here,
  246. * so I'm making up 128 as the max number of individually updatable
  247. * components we support.
  248. * 128 should be pretty excessive, but there's still some chance
  249. * somebody will do that someday and we'll need to raise this.
  250. */
  251. if (tmpesrt.fw_resource_count > 128) {
  252. pr_err("ESRT says fw_resource_count has very large value %d.\n",
  253. tmpesrt.fw_resource_count);
  254. return;
  255. }
  256. /*
  257. * We know it can't be larger than N * sizeof() here, and N is limited
  258. * by the previous test to a small number, so there's no overflow.
  259. */
  260. entries_size = tmpesrt.fw_resource_count * entry_size;
  261. if (max < size + entries_size) {
  262. pr_err("ESRT does not fit on single memory map entry (size: %zu max: %zu)\n",
  263. size, max);
  264. return;
  265. }
  266. size += entries_size;
  267. esrt_data = (phys_addr_t)efi.esrt;
  268. esrt_data_size = size;
  269. end = esrt_data + size;
  270. pr_info("Reserving ESRT space from %pa to %pa.\n", &esrt_data, &end);
  271. if (md.type == EFI_BOOT_SERVICES_DATA)
  272. efi_mem_reserve(esrt_data, esrt_data_size);
  273. pr_debug("esrt-init: loaded.\n");
  274. }
  275. static int __init register_entries(void)
  276. {
  277. struct efi_system_resource_entry_v1 *v1_entries = (void *)esrt->entries;
  278. int i, rc;
  279. if (!esrt_table_exists())
  280. return 0;
  281. for (i = 0; i < le32_to_cpu(esrt->fw_resource_count); i++) {
  282. void *esre = NULL;
  283. if (esrt->fw_resource_version == 1) {
  284. esre = &v1_entries[i];
  285. } else {
  286. pr_err("Unsupported ESRT version %lld.\n",
  287. esrt->fw_resource_version);
  288. return -EINVAL;
  289. }
  290. rc = esre_create_sysfs_entry(esre, i);
  291. if (rc < 0) {
  292. pr_err("ESRT entry creation failed with error %d.\n",
  293. rc);
  294. return rc;
  295. }
  296. }
  297. return 0;
  298. }
  299. static void cleanup_entry_list(void)
  300. {
  301. struct esre_entry *entry, *next;
  302. list_for_each_entry_safe(entry, next, &entry_list, list) {
  303. kobject_put(&entry->kobj);
  304. }
  305. }
  306. static int __init esrt_sysfs_init(void)
  307. {
  308. int error;
  309. pr_debug("esrt-sysfs: loading.\n");
  310. if (!esrt_data || !esrt_data_size)
  311. return -ENOSYS;
  312. esrt = memremap(esrt_data, esrt_data_size, MEMREMAP_WB);
  313. if (!esrt) {
  314. pr_err("memremap(%pa, %zu) failed.\n", &esrt_data,
  315. esrt_data_size);
  316. return -ENOMEM;
  317. }
  318. esrt_kobj = kobject_create_and_add("esrt", efi_kobj);
  319. if (!esrt_kobj) {
  320. pr_err("Firmware table registration failed.\n");
  321. error = -ENOMEM;
  322. goto err;
  323. }
  324. error = sysfs_create_group(esrt_kobj, &esrt_attr_group);
  325. if (error) {
  326. pr_err("Sysfs attribute export failed with error %d.\n",
  327. error);
  328. goto err_remove_esrt;
  329. }
  330. esrt_kset = kset_create_and_add("entries", NULL, esrt_kobj);
  331. if (!esrt_kset) {
  332. pr_err("kset creation failed.\n");
  333. error = -ENOMEM;
  334. goto err_remove_group;
  335. }
  336. error = register_entries();
  337. if (error)
  338. goto err_cleanup_list;
  339. pr_debug("esrt-sysfs: loaded.\n");
  340. return 0;
  341. err_cleanup_list:
  342. cleanup_entry_list();
  343. kset_unregister(esrt_kset);
  344. err_remove_group:
  345. sysfs_remove_group(esrt_kobj, &esrt_attr_group);
  346. err_remove_esrt:
  347. kobject_put(esrt_kobj);
  348. err:
  349. memunmap(esrt);
  350. esrt = NULL;
  351. return error;
  352. }
  353. device_initcall(esrt_sysfs_init);
  354. /*
  355. MODULE_AUTHOR("Peter Jones <pjones@redhat.com>");
  356. MODULE_DESCRIPTION("EFI System Resource Table support");
  357. MODULE_LICENSE("GPL");
  358. */