etnaviv_dump.c 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  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_dump.h"
  18. #include "etnaviv_gem.h"
  19. #include "etnaviv_gpu.h"
  20. #include "etnaviv_mmu.h"
  21. #include "state.xml.h"
  22. #include "state_hi.xml.h"
  23. struct core_dump_iterator {
  24. void *start;
  25. struct etnaviv_dump_object_header *hdr;
  26. void *data;
  27. };
  28. static const unsigned short etnaviv_dump_registers[] = {
  29. VIVS_HI_AXI_STATUS,
  30. VIVS_HI_CLOCK_CONTROL,
  31. VIVS_HI_IDLE_STATE,
  32. VIVS_HI_AXI_CONFIG,
  33. VIVS_HI_INTR_ENBL,
  34. VIVS_HI_CHIP_IDENTITY,
  35. VIVS_HI_CHIP_FEATURE,
  36. VIVS_HI_CHIP_MODEL,
  37. VIVS_HI_CHIP_REV,
  38. VIVS_HI_CHIP_DATE,
  39. VIVS_HI_CHIP_TIME,
  40. VIVS_HI_CHIP_MINOR_FEATURE_0,
  41. VIVS_HI_CACHE_CONTROL,
  42. VIVS_HI_AXI_CONTROL,
  43. VIVS_PM_POWER_CONTROLS,
  44. VIVS_PM_MODULE_CONTROLS,
  45. VIVS_PM_MODULE_STATUS,
  46. VIVS_PM_PULSE_EATER,
  47. VIVS_MC_MMU_FE_PAGE_TABLE,
  48. VIVS_MC_MMU_TX_PAGE_TABLE,
  49. VIVS_MC_MMU_PE_PAGE_TABLE,
  50. VIVS_MC_MMU_PEZ_PAGE_TABLE,
  51. VIVS_MC_MMU_RA_PAGE_TABLE,
  52. VIVS_MC_DEBUG_MEMORY,
  53. VIVS_MC_MEMORY_BASE_ADDR_RA,
  54. VIVS_MC_MEMORY_BASE_ADDR_FE,
  55. VIVS_MC_MEMORY_BASE_ADDR_TX,
  56. VIVS_MC_MEMORY_BASE_ADDR_PEZ,
  57. VIVS_MC_MEMORY_BASE_ADDR_PE,
  58. VIVS_MC_MEMORY_TIMING_CONTROL,
  59. VIVS_MC_BUS_CONFIG,
  60. VIVS_FE_DMA_STATUS,
  61. VIVS_FE_DMA_DEBUG_STATE,
  62. VIVS_FE_DMA_ADDRESS,
  63. VIVS_FE_DMA_LOW,
  64. VIVS_FE_DMA_HIGH,
  65. VIVS_FE_AUTO_FLUSH,
  66. };
  67. static void etnaviv_core_dump_header(struct core_dump_iterator *iter,
  68. u32 type, void *data_end)
  69. {
  70. struct etnaviv_dump_object_header *hdr = iter->hdr;
  71. hdr->magic = cpu_to_le32(ETDUMP_MAGIC);
  72. hdr->type = cpu_to_le32(type);
  73. hdr->file_offset = cpu_to_le32(iter->data - iter->start);
  74. hdr->file_size = cpu_to_le32(data_end - iter->data);
  75. iter->hdr++;
  76. iter->data += hdr->file_size;
  77. }
  78. static void etnaviv_core_dump_registers(struct core_dump_iterator *iter,
  79. struct etnaviv_gpu *gpu)
  80. {
  81. struct etnaviv_dump_registers *reg = iter->data;
  82. unsigned int i;
  83. for (i = 0; i < ARRAY_SIZE(etnaviv_dump_registers); i++, reg++) {
  84. reg->reg = etnaviv_dump_registers[i];
  85. reg->value = gpu_read(gpu, etnaviv_dump_registers[i]);
  86. }
  87. etnaviv_core_dump_header(iter, ETDUMP_BUF_REG, reg);
  88. }
  89. static void etnaviv_core_dump_mmu(struct core_dump_iterator *iter,
  90. struct etnaviv_gpu *gpu, size_t mmu_size)
  91. {
  92. etnaviv_iommu_dump(gpu->mmu, iter->data);
  93. etnaviv_core_dump_header(iter, ETDUMP_BUF_MMU, iter->data + mmu_size);
  94. }
  95. static void etnaviv_core_dump_mem(struct core_dump_iterator *iter, u32 type,
  96. void *ptr, size_t size, u64 iova)
  97. {
  98. memcpy(iter->data, ptr, size);
  99. iter->hdr->iova = cpu_to_le64(iova);
  100. etnaviv_core_dump_header(iter, type, iter->data + size);
  101. }
  102. void etnaviv_core_dump(struct etnaviv_gpu *gpu)
  103. {
  104. struct core_dump_iterator iter;
  105. struct etnaviv_vram_mapping *vram;
  106. struct etnaviv_gem_object *obj;
  107. struct etnaviv_cmdbuf *cmd;
  108. unsigned int n_obj, n_bomap_pages;
  109. size_t file_size, mmu_size;
  110. __le64 *bomap, *bomap_start;
  111. mmu_size = etnaviv_iommu_dump_size(gpu->mmu);
  112. /* We always dump registers, mmu, ring and end marker */
  113. n_obj = 4;
  114. n_bomap_pages = 0;
  115. file_size = ARRAY_SIZE(etnaviv_dump_registers) *
  116. sizeof(struct etnaviv_dump_registers) +
  117. mmu_size + gpu->buffer->size;
  118. /* Add in the active command buffers */
  119. list_for_each_entry(cmd, &gpu->active_cmd_list, node) {
  120. file_size += cmd->size;
  121. n_obj++;
  122. }
  123. /* Add in the active buffer objects */
  124. list_for_each_entry(vram, &gpu->mmu->mappings, mmu_node) {
  125. if (!vram->use)
  126. continue;
  127. obj = vram->object;
  128. file_size += obj->base.size;
  129. n_bomap_pages += obj->base.size >> PAGE_SHIFT;
  130. n_obj++;
  131. }
  132. /* If we have any buffer objects, add a bomap object */
  133. if (n_bomap_pages) {
  134. file_size += n_bomap_pages * sizeof(__le64);
  135. n_obj++;
  136. }
  137. /* Add the size of the headers */
  138. file_size += sizeof(*iter.hdr) * n_obj;
  139. /* Allocate the file in vmalloc memory, it's likely to be big */
  140. iter.start = vmalloc(file_size);
  141. if (!iter.start) {
  142. dev_warn(gpu->dev, "failed to allocate devcoredump file\n");
  143. return;
  144. }
  145. /* Point the data member after the headers */
  146. iter.hdr = iter.start;
  147. iter.data = &iter.hdr[n_obj];
  148. memset(iter.hdr, 0, iter.data - iter.start);
  149. etnaviv_core_dump_registers(&iter, gpu);
  150. etnaviv_core_dump_mmu(&iter, gpu, mmu_size);
  151. etnaviv_core_dump_mem(&iter, ETDUMP_BUF_RING, gpu->buffer->vaddr,
  152. gpu->buffer->size,
  153. etnaviv_iommu_get_cmdbuf_va(gpu, gpu->buffer));
  154. list_for_each_entry(cmd, &gpu->active_cmd_list, node)
  155. etnaviv_core_dump_mem(&iter, ETDUMP_BUF_CMD, cmd->vaddr,
  156. cmd->size,
  157. etnaviv_iommu_get_cmdbuf_va(gpu, cmd));
  158. /* Reserve space for the bomap */
  159. if (n_bomap_pages) {
  160. bomap_start = bomap = iter.data;
  161. memset(bomap, 0, sizeof(*bomap) * n_bomap_pages);
  162. etnaviv_core_dump_header(&iter, ETDUMP_BUF_BOMAP,
  163. bomap + n_bomap_pages);
  164. } else {
  165. /* Silence warning */
  166. bomap_start = bomap = NULL;
  167. }
  168. list_for_each_entry(vram, &gpu->mmu->mappings, mmu_node) {
  169. struct page **pages;
  170. void *vaddr;
  171. if (vram->use == 0)
  172. continue;
  173. obj = vram->object;
  174. mutex_lock(&obj->lock);
  175. pages = etnaviv_gem_get_pages(obj);
  176. mutex_unlock(&obj->lock);
  177. if (pages) {
  178. int j;
  179. iter.hdr->data[0] = bomap - bomap_start;
  180. for (j = 0; j < obj->base.size >> PAGE_SHIFT; j++)
  181. *bomap++ = cpu_to_le64(page_to_phys(*pages++));
  182. }
  183. iter.hdr->iova = cpu_to_le64(vram->iova);
  184. vaddr = etnaviv_gem_vmap(&obj->base);
  185. if (vaddr)
  186. memcpy(iter.data, vaddr, obj->base.size);
  187. etnaviv_core_dump_header(&iter, ETDUMP_BUF_BO, iter.data +
  188. obj->base.size);
  189. }
  190. etnaviv_core_dump_header(&iter, ETDUMP_BUF_END, iter.data);
  191. dev_coredumpv(gpu->dev, iter.start, iter.data - iter.start, GFP_KERNEL);
  192. }