vmcore_info.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * crash.c - kernel crash support code.
  4. * Copyright (C) 2002-2004 Eric Biederman <ebiederm@xmission.com>
  5. */
  6. #include <linux/buildid.h>
  7. #include <linux/init.h>
  8. #include <linux/utsname.h>
  9. #include <linux/vmalloc.h>
  10. #include <linux/sizes.h>
  11. #include <linux/kexec.h>
  12. #include <linux/memory.h>
  13. #include <linux/cpuhotplug.h>
  14. #include <linux/memblock.h>
  15. #include <linux/kmemleak.h>
  16. #include <asm/page.h>
  17. #include <asm/sections.h>
  18. #include <crypto/sha1.h>
  19. #include "kallsyms_internal.h"
  20. #include "kexec_internal.h"
  21. /* vmcoreinfo stuff */
  22. unsigned char *vmcoreinfo_data;
  23. size_t vmcoreinfo_size;
  24. u32 *vmcoreinfo_note;
  25. /* trusted vmcoreinfo, e.g. we can make a copy in the crash memory */
  26. static unsigned char *vmcoreinfo_data_safecopy;
  27. struct hwerr_info {
  28. atomic_t count;
  29. time64_t timestamp;
  30. };
  31. /*
  32. * The hwerr_data[] array is declared with global scope so that it remains
  33. * accessible to vmcoreinfo even when Link Time Optimization (LTO) is enabled.
  34. */
  35. struct hwerr_info hwerr_data[HWERR_RECOV_MAX];
  36. Elf_Word *append_elf_note(Elf_Word *buf, char *name, unsigned int type,
  37. void *data, size_t data_len)
  38. {
  39. struct elf_note *note = (struct elf_note *)buf;
  40. note->n_namesz = strlen(name) + 1;
  41. note->n_descsz = data_len;
  42. note->n_type = type;
  43. buf += DIV_ROUND_UP(sizeof(*note), sizeof(Elf_Word));
  44. memcpy(buf, name, note->n_namesz);
  45. buf += DIV_ROUND_UP(note->n_namesz, sizeof(Elf_Word));
  46. memcpy(buf, data, data_len);
  47. buf += DIV_ROUND_UP(data_len, sizeof(Elf_Word));
  48. return buf;
  49. }
  50. void final_note(Elf_Word *buf)
  51. {
  52. memset(buf, 0, sizeof(struct elf_note));
  53. }
  54. static void update_vmcoreinfo_note(void)
  55. {
  56. u32 *buf = vmcoreinfo_note;
  57. if (!vmcoreinfo_size)
  58. return;
  59. buf = append_elf_note(buf, VMCOREINFO_NOTE_NAME, 0, vmcoreinfo_data,
  60. vmcoreinfo_size);
  61. final_note(buf);
  62. }
  63. void crash_update_vmcoreinfo_safecopy(void *ptr)
  64. {
  65. if (ptr)
  66. memcpy(ptr, vmcoreinfo_data, vmcoreinfo_size);
  67. vmcoreinfo_data_safecopy = ptr;
  68. }
  69. void crash_save_vmcoreinfo(void)
  70. {
  71. if (!vmcoreinfo_note)
  72. return;
  73. /* Use the safe copy to generate vmcoreinfo note if have */
  74. if (vmcoreinfo_data_safecopy)
  75. vmcoreinfo_data = vmcoreinfo_data_safecopy;
  76. vmcoreinfo_append_str("CRASHTIME=%lld\n", ktime_get_real_seconds());
  77. update_vmcoreinfo_note();
  78. }
  79. void vmcoreinfo_append_str(const char *fmt, ...)
  80. {
  81. va_list args;
  82. char buf[0x50];
  83. size_t r;
  84. va_start(args, fmt);
  85. r = vscnprintf(buf, sizeof(buf), fmt, args);
  86. va_end(args);
  87. r = min(r, (size_t)VMCOREINFO_BYTES - vmcoreinfo_size);
  88. memcpy(&vmcoreinfo_data[vmcoreinfo_size], buf, r);
  89. vmcoreinfo_size += r;
  90. WARN_ONCE(vmcoreinfo_size == VMCOREINFO_BYTES,
  91. "vmcoreinfo data exceeds allocated size, truncating");
  92. }
  93. /*
  94. * provide an empty default implementation here -- architecture
  95. * code may override this
  96. */
  97. void __weak arch_crash_save_vmcoreinfo(void)
  98. {}
  99. phys_addr_t __weak paddr_vmcoreinfo_note(void)
  100. {
  101. return __pa(vmcoreinfo_note);
  102. }
  103. EXPORT_SYMBOL(paddr_vmcoreinfo_note);
  104. void hwerr_log_error_type(enum hwerr_error_type src)
  105. {
  106. if (src < 0 || src >= HWERR_RECOV_MAX)
  107. return;
  108. atomic_inc(&hwerr_data[src].count);
  109. WRITE_ONCE(hwerr_data[src].timestamp, ktime_get_real_seconds());
  110. }
  111. EXPORT_SYMBOL_GPL(hwerr_log_error_type);
  112. static int __init crash_save_vmcoreinfo_init(void)
  113. {
  114. int order;
  115. order = get_order(VMCOREINFO_BYTES);
  116. vmcoreinfo_data = (unsigned char *)__get_free_pages(GFP_KERNEL | __GFP_ZERO, order);
  117. if (!vmcoreinfo_data) {
  118. pr_warn("Memory allocation for vmcoreinfo_data failed\n");
  119. return -ENOMEM;
  120. }
  121. vmcoreinfo_note = alloc_pages_exact(VMCOREINFO_NOTE_SIZE,
  122. GFP_KERNEL | __GFP_ZERO);
  123. if (!vmcoreinfo_note) {
  124. free_pages((unsigned long)vmcoreinfo_data, order);
  125. vmcoreinfo_data = NULL;
  126. pr_warn("Memory allocation for vmcoreinfo_note failed\n");
  127. return -ENOMEM;
  128. }
  129. VMCOREINFO_OSRELEASE(init_uts_ns.name.release);
  130. VMCOREINFO_BUILD_ID();
  131. VMCOREINFO_PAGESIZE(PAGE_SIZE);
  132. VMCOREINFO_SYMBOL(init_uts_ns);
  133. VMCOREINFO_OFFSET(uts_namespace, name);
  134. VMCOREINFO_SYMBOL(node_online_map);
  135. #ifdef CONFIG_MMU
  136. VMCOREINFO_SYMBOL_ARRAY(swapper_pg_dir);
  137. #endif
  138. VMCOREINFO_SYMBOL(_stext);
  139. vmcoreinfo_append_str("NUMBER(VMALLOC_START)=0x%lx\n", (unsigned long) VMALLOC_START);
  140. #ifndef CONFIG_NUMA
  141. VMCOREINFO_SYMBOL(mem_map);
  142. VMCOREINFO_SYMBOL(contig_page_data);
  143. #endif
  144. #ifdef CONFIG_SPARSEMEM_VMEMMAP
  145. VMCOREINFO_SYMBOL_ARRAY(vmemmap);
  146. #endif
  147. #ifdef CONFIG_SPARSEMEM
  148. VMCOREINFO_SYMBOL_ARRAY(mem_section);
  149. VMCOREINFO_LENGTH(mem_section, NR_SECTION_ROOTS);
  150. VMCOREINFO_STRUCT_SIZE(mem_section);
  151. VMCOREINFO_OFFSET(mem_section, section_mem_map);
  152. VMCOREINFO_NUMBER(SECTION_SIZE_BITS);
  153. VMCOREINFO_NUMBER(MAX_PHYSMEM_BITS);
  154. #endif
  155. VMCOREINFO_STRUCT_SIZE(page);
  156. VMCOREINFO_STRUCT_SIZE(pglist_data);
  157. VMCOREINFO_STRUCT_SIZE(zone);
  158. VMCOREINFO_STRUCT_SIZE(free_area);
  159. VMCOREINFO_STRUCT_SIZE(list_head);
  160. VMCOREINFO_SIZE(nodemask_t);
  161. VMCOREINFO_OFFSET(page, flags);
  162. VMCOREINFO_OFFSET(page, _refcount);
  163. VMCOREINFO_OFFSET(page, mapping);
  164. VMCOREINFO_OFFSET(page, lru);
  165. VMCOREINFO_OFFSET(page, _mapcount);
  166. VMCOREINFO_OFFSET(page, private);
  167. VMCOREINFO_OFFSET(page, compound_head);
  168. VMCOREINFO_OFFSET(pglist_data, node_zones);
  169. VMCOREINFO_OFFSET(pglist_data, nr_zones);
  170. #ifdef CONFIG_FLATMEM
  171. VMCOREINFO_OFFSET(pglist_data, node_mem_map);
  172. #endif
  173. VMCOREINFO_OFFSET(pglist_data, node_start_pfn);
  174. VMCOREINFO_OFFSET(pglist_data, node_spanned_pages);
  175. VMCOREINFO_OFFSET(pglist_data, node_id);
  176. VMCOREINFO_OFFSET(zone, free_area);
  177. VMCOREINFO_OFFSET(zone, vm_stat);
  178. VMCOREINFO_OFFSET(zone, spanned_pages);
  179. VMCOREINFO_OFFSET(free_area, free_list);
  180. VMCOREINFO_OFFSET(list_head, next);
  181. VMCOREINFO_OFFSET(list_head, prev);
  182. VMCOREINFO_LENGTH(zone.free_area, NR_PAGE_ORDERS);
  183. log_buf_vmcoreinfo_setup();
  184. VMCOREINFO_LENGTH(free_area.free_list, MIGRATE_TYPES);
  185. VMCOREINFO_NUMBER(NR_FREE_PAGES);
  186. VMCOREINFO_NUMBER(PG_lru);
  187. VMCOREINFO_NUMBER(PG_private);
  188. VMCOREINFO_NUMBER(PG_swapcache);
  189. VMCOREINFO_NUMBER(PG_swapbacked);
  190. #define PAGE_SLAB_MAPCOUNT_VALUE (PGTY_slab << 24)
  191. VMCOREINFO_NUMBER(PAGE_SLAB_MAPCOUNT_VALUE);
  192. #ifdef CONFIG_MEMORY_FAILURE
  193. VMCOREINFO_NUMBER(PG_hwpoison);
  194. #endif
  195. VMCOREINFO_NUMBER(PG_head_mask);
  196. #define PAGE_BUDDY_MAPCOUNT_VALUE (PGTY_buddy << 24)
  197. VMCOREINFO_NUMBER(PAGE_BUDDY_MAPCOUNT_VALUE);
  198. #define PAGE_HUGETLB_MAPCOUNT_VALUE (PGTY_hugetlb << 24)
  199. VMCOREINFO_NUMBER(PAGE_HUGETLB_MAPCOUNT_VALUE);
  200. #define PAGE_OFFLINE_MAPCOUNT_VALUE (PGTY_offline << 24)
  201. VMCOREINFO_NUMBER(PAGE_OFFLINE_MAPCOUNT_VALUE);
  202. #ifdef CONFIG_UNACCEPTED_MEMORY
  203. #define PAGE_UNACCEPTED_MAPCOUNT_VALUE (PGTY_unaccepted << 24)
  204. VMCOREINFO_NUMBER(PAGE_UNACCEPTED_MAPCOUNT_VALUE);
  205. #endif
  206. #ifdef CONFIG_KALLSYMS
  207. VMCOREINFO_SYMBOL(kallsyms_names);
  208. VMCOREINFO_SYMBOL(kallsyms_num_syms);
  209. VMCOREINFO_SYMBOL(kallsyms_token_table);
  210. VMCOREINFO_SYMBOL(kallsyms_token_index);
  211. VMCOREINFO_SYMBOL(kallsyms_offsets);
  212. #endif /* CONFIG_KALLSYMS */
  213. arch_crash_save_vmcoreinfo();
  214. update_vmcoreinfo_note();
  215. return 0;
  216. }
  217. subsys_initcall(crash_save_vmcoreinfo_init);