etnaviv_dump.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. /*
  2. * Copyright (C) 2015 Etnaviv Project
  3. *
  4. * This program is free software; you can redistribute it and/or modify it
  5. * under the terms of the GNU General Public License version 2 as published by
  6. * the Free Software Foundation.
  7. *
  8. * This program is distributed in the hope that it will be useful, but WITHOUT
  9. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  11. * more details.
  12. *
  13. * You should have received a copy of the GNU General Public License along with
  14. * this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include <linux/devcoredump.h>
  17. #include "etnaviv_cmdbuf.h"
  18. #include "etnaviv_dump.h"
  19. #include "etnaviv_gem.h"
  20. #include "etnaviv_gpu.h"
  21. #include "etnaviv_mmu.h"
  22. #include "etnaviv_sched.h"
  23. #include "state.xml.h"
  24. #include "state_hi.xml.h"
  25. static bool etnaviv_dump_core = true;
  26. module_param_named(dump_core, etnaviv_dump_core, bool, 0600);
  27. struct core_dump_iterator {
  28. void *start;
  29. struct etnaviv_dump_object_header *hdr;
  30. void *data;
  31. };
  32. static const unsigned short etnaviv_dump_registers[] = {
  33. VIVS_HI_AXI_STATUS,
  34. VIVS_HI_CLOCK_CONTROL,
  35. VIVS_HI_IDLE_STATE,
  36. VIVS_HI_AXI_CONFIG,
  37. VIVS_HI_INTR_ENBL,
  38. VIVS_HI_CHIP_IDENTITY,
  39. VIVS_HI_CHIP_FEATURE,
  40. VIVS_HI_CHIP_MODEL,
  41. VIVS_HI_CHIP_REV,
  42. VIVS_HI_CHIP_DATE,
  43. VIVS_HI_CHIP_TIME,
  44. VIVS_HI_CHIP_MINOR_FEATURE_0,
  45. VIVS_HI_CACHE_CONTROL,
  46. VIVS_HI_AXI_CONTROL,
  47. VIVS_PM_POWER_CONTROLS,
  48. VIVS_PM_MODULE_CONTROLS,
  49. VIVS_PM_MODULE_STATUS,
  50. VIVS_PM_PULSE_EATER,
  51. VIVS_MC_MMU_FE_PAGE_TABLE,
  52. VIVS_MC_MMU_TX_PAGE_TABLE,
  53. VIVS_MC_MMU_PE_PAGE_TABLE,
  54. VIVS_MC_MMU_PEZ_PAGE_TABLE,
  55. VIVS_MC_MMU_RA_PAGE_TABLE,
  56. VIVS_MC_DEBUG_MEMORY,
  57. VIVS_MC_MEMORY_BASE_ADDR_RA,
  58. VIVS_MC_MEMORY_BASE_ADDR_FE,
  59. VIVS_MC_MEMORY_BASE_ADDR_TX,
  60. VIVS_MC_MEMORY_BASE_ADDR_PEZ,
  61. VIVS_MC_MEMORY_BASE_ADDR_PE,
  62. VIVS_MC_MEMORY_TIMING_CONTROL,
  63. VIVS_MC_BUS_CONFIG,
  64. VIVS_FE_DMA_STATUS,
  65. VIVS_FE_DMA_DEBUG_STATE,
  66. VIVS_FE_DMA_ADDRESS,
  67. VIVS_FE_DMA_LOW,
  68. VIVS_FE_DMA_HIGH,
  69. VIVS_FE_AUTO_FLUSH,
  70. };
  71. static void etnaviv_core_dump_header(struct core_dump_iterator *iter,
  72. u32 type, void *data_end)
  73. {
  74. struct etnaviv_dump_object_header *hdr = iter->hdr;
  75. hdr->magic = cpu_to_le32(ETDUMP_MAGIC);
  76. hdr->type = cpu_to_le32(type);
  77. hdr->file_offset = cpu_to_le32(iter->data - iter->start);
  78. hdr->file_size = cpu_to_le32(data_end - iter->data);
  79. iter->hdr++;
  80. iter->data += hdr->file_size;
  81. }
  82. static void etnaviv_core_dump_registers(struct core_dump_iterator *iter,
  83. struct etnaviv_gpu *gpu)
  84. {
  85. struct etnaviv_dump_registers *reg = iter->data;
  86. unsigned int i;
  87. for (i = 0; i < ARRAY_SIZE(etnaviv_dump_registers); i++, reg++) {
  88. reg->reg = etnaviv_dump_registers[i];
  89. reg->value = gpu_read(gpu, etnaviv_dump_registers[i]);
  90. }
  91. etnaviv_core_dump_header(iter, ETDUMP_BUF_REG, reg);
  92. }
  93. static void etnaviv_core_dump_mmu(struct core_dump_iterator *iter,
  94. struct etnaviv_gpu *gpu, size_t mmu_size)
  95. {
  96. etnaviv_iommu_dump(gpu->mmu, iter->data);
  97. etnaviv_core_dump_header(iter, ETDUMP_BUF_MMU, iter->data + mmu_size);
  98. }
  99. static void etnaviv_core_dump_mem(struct core_dump_iterator *iter, u32 type,
  100. void *ptr, size_t size, u64 iova)
  101. {
  102. memcpy(iter->data, ptr, size);
  103. iter->hdr->iova = cpu_to_le64(iova);
  104. etnaviv_core_dump_header(iter, type, iter->data + size);
  105. }
  106. void etnaviv_core_dump(struct etnaviv_gpu *gpu)
  107. {
  108. struct core_dump_iterator iter;
  109. struct etnaviv_vram_mapping *vram;
  110. struct etnaviv_gem_object *obj;
  111. struct etnaviv_gem_submit *submit;
  112. struct drm_sched_job *s_job;
  113. unsigned int n_obj, n_bomap_pages;
  114. size_t file_size, mmu_size;
  115. __le64 *bomap, *bomap_start;
  116. /* Only catch the first event, or when manually re-armed */
  117. if (!etnaviv_dump_core)
  118. return;
  119. etnaviv_dump_core = false;
  120. mmu_size = etnaviv_iommu_dump_size(gpu->mmu);
  121. /* We always dump registers, mmu, ring and end marker */
  122. n_obj = 4;
  123. n_bomap_pages = 0;
  124. file_size = ARRAY_SIZE(etnaviv_dump_registers) *
  125. sizeof(struct etnaviv_dump_registers) +
  126. mmu_size + gpu->buffer.size;
  127. /* Add in the active command buffers */
  128. spin_lock(&gpu->sched.job_list_lock);
  129. list_for_each_entry(s_job, &gpu->sched.ring_mirror_list, node) {
  130. submit = to_etnaviv_submit(s_job);
  131. file_size += submit->cmdbuf.size;
  132. n_obj++;
  133. }
  134. spin_unlock(&gpu->sched.job_list_lock);
  135. /* Add in the active buffer objects */
  136. list_for_each_entry(vram, &gpu->mmu->mappings, mmu_node) {
  137. if (!vram->use)
  138. continue;
  139. obj = vram->object;
  140. file_size += obj->base.size;
  141. n_bomap_pages += obj->base.size >> PAGE_SHIFT;
  142. n_obj++;
  143. }
  144. /* If we have any buffer objects, add a bomap object */
  145. if (n_bomap_pages) {
  146. file_size += n_bomap_pages * sizeof(__le64);
  147. n_obj++;
  148. }
  149. /* Add the size of the headers */
  150. file_size += sizeof(*iter.hdr) * n_obj;
  151. /* Allocate the file in vmalloc memory, it's likely to be big */
  152. iter.start = __vmalloc(file_size, GFP_KERNEL | __GFP_NOWARN | __GFP_NORETRY,
  153. PAGE_KERNEL);
  154. if (!iter.start) {
  155. dev_warn(gpu->dev, "failed to allocate devcoredump file\n");
  156. return;
  157. }
  158. /* Point the data member after the headers */
  159. iter.hdr = iter.start;
  160. iter.data = &iter.hdr[n_obj];
  161. memset(iter.hdr, 0, iter.data - iter.start);
  162. etnaviv_core_dump_registers(&iter, gpu);
  163. etnaviv_core_dump_mmu(&iter, gpu, mmu_size);
  164. etnaviv_core_dump_mem(&iter, ETDUMP_BUF_RING, gpu->buffer.vaddr,
  165. gpu->buffer.size,
  166. etnaviv_cmdbuf_get_va(&gpu->buffer));
  167. spin_lock(&gpu->sched.job_list_lock);
  168. list_for_each_entry(s_job, &gpu->sched.ring_mirror_list, node) {
  169. submit = to_etnaviv_submit(s_job);
  170. etnaviv_core_dump_mem(&iter, ETDUMP_BUF_CMD,
  171. submit->cmdbuf.vaddr, submit->cmdbuf.size,
  172. etnaviv_cmdbuf_get_va(&submit->cmdbuf));
  173. }
  174. spin_unlock(&gpu->sched.job_list_lock);
  175. /* Reserve space for the bomap */
  176. if (n_bomap_pages) {
  177. bomap_start = bomap = iter.data;
  178. memset(bomap, 0, sizeof(*bomap) * n_bomap_pages);
  179. etnaviv_core_dump_header(&iter, ETDUMP_BUF_BOMAP,
  180. bomap + n_bomap_pages);
  181. } else {
  182. /* Silence warning */
  183. bomap_start = bomap = NULL;
  184. }
  185. list_for_each_entry(vram, &gpu->mmu->mappings, mmu_node) {
  186. struct page **pages;
  187. void *vaddr;
  188. if (vram->use == 0)
  189. continue;
  190. obj = vram->object;
  191. mutex_lock(&obj->lock);
  192. pages = etnaviv_gem_get_pages(obj);
  193. mutex_unlock(&obj->lock);
  194. if (pages) {
  195. int j;
  196. iter.hdr->data[0] = bomap - bomap_start;
  197. for (j = 0; j < obj->base.size >> PAGE_SHIFT; j++)
  198. *bomap++ = cpu_to_le64(page_to_phys(*pages++));
  199. }
  200. iter.hdr->iova = cpu_to_le64(vram->iova);
  201. vaddr = etnaviv_gem_vmap(&obj->base);
  202. if (vaddr)
  203. memcpy(iter.data, vaddr, obj->base.size);
  204. etnaviv_core_dump_header(&iter, ETDUMP_BUF_BO, iter.data +
  205. obj->base.size);
  206. }
  207. etnaviv_core_dump_header(&iter, ETDUMP_BUF_END, iter.data);
  208. dev_coredumpv(gpu->dev, iter.start, iter.data - iter.start, GFP_KERNEL);
  209. }