blob: 3b72b0f6d65610f1da099034b77798852820011c [file] [log] [blame]
Jordan Crouse156cfbc2012-01-24 09:32:04 -07001/* Copyright (c) 2012, Code Aurora Forum. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
13
14#ifndef _KGSL_SNAPSHOT_H_
15#define _KGSL_SNAPSHOT_H_
16
17#include <linux/types.h>
18
19/* Snapshot header */
20
21#define SNAPSHOT_MAGIC 0x504D0001
22
23/* GPU ID scheme:
24 * [16:31] - core identifer (0x0002 for 2D or 0x0003 for 3D)
25 * [00:16] - GPU specific identifier
26 */
27
28struct kgsl_snapshot_header {
29 __u32 magic; /* Magic identifier */
30 __u32 gpuid; /* GPU ID - see above */
31} __packed;
32
33/* Section header */
34#define SNAPSHOT_SECTION_MAGIC 0xABCD
35
36struct kgsl_snapshot_section_header {
37 __u16 magic; /* Magic identifier */
38 __u16 id; /* Type of section */
39 __u32 size; /* Size of the section including this header */
40} __packed;
41
42/* Section identifiers */
43#define KGSL_SNAPSHOT_SECTION_OS 0x0101
44#define KGSL_SNAPSHOT_SECTION_REGS 0x0201
45#define KGSL_SNAPSHOT_SECTION_RB 0x0301
46#define KGSL_SNAPSHOT_SECTION_IB 0x0401
47#define KGSL_SNAPSHOT_SECTION_INDEXED_REGS 0x0501
48#define KGSL_SNAPSHOT_SECTION_ISTORE 0x0801
49#define KGSL_SNAPSHOT_SECTION_DEBUG 0x0901
Jordan Crouse0c2761a2012-02-01 22:11:12 -070050#define KGSL_SNAPSHOT_SECTION_DEBUGBUS 0x0A01
Jordan Crouse156cfbc2012-01-24 09:32:04 -070051#define KGSL_SNAPSHOT_SECTION_END 0xFFFF
52
53/* OS sub-section header */
54#define KGSL_SNAPSHOT_OS_LINUX 0x0001
55
56/* Linux OS specific information */
57
58#define SNAPSHOT_STATE_HUNG 0
59#define SNAPSHOT_STATE_RUNNING 1
60
61struct kgsl_snapshot_linux {
62 int osid; /* subsection OS identifier */
63 int state; /* 1 if the thread is running, 0 for hung */
64 __u32 seconds; /* Unix timestamp for the snapshot */
65 __u32 power_flags; /* Current power flags */
66 __u32 power_level; /* Current power level */
67 __u32 power_interval_timeout; /* Power interval timeout */
68 __u32 grpclk; /* Current GP clock value */
69 __u32 busclk; /* Current busclk value */
70 __u32 ptbase; /* Current ptbase */
71 __u32 pid; /* PID of the process that owns the PT */
72 __u32 current_context; /* ID of the current context */
73 __u32 ctxtcount; /* Number of contexts appended to section */
74 unsigned char release[32]; /* kernel release */
75 unsigned char version[32]; /* kernel version */
76 unsigned char comm[16]; /* Name of the process that owns the PT */
77} __packed;
78
79/*
80 * This structure contains a record of an active context.
81 * These are appended one after another in the OS section below
82 * the header above
83 */
84
85struct kgsl_snapshot_linux_context {
86 __u32 id; /* The context ID */
87 __u32 timestamp_queued; /* The last queued timestamp */
88 __u32 timestamp_retired; /* The last timestamp retired by HW */
89};
90
91/* Ringbuffer sub-section header */
92struct kgsl_snapshot_rb {
93 int start; /* dword at the start of the dump */
94 int end; /* dword at the end of the dump */
95 int rbsize; /* Size (in dwords) of the ringbuffer */
96 int wptr; /* Current index of the CPU write pointer */
97 int rptr; /* Current index of the GPU read pointer */
98 int count; /* Number of dwords in the dump */
99} __packed;
100
101/* Indirect buffer sub-section header */
102struct kgsl_snapshot_ib {
103 __u32 gpuaddr; /* GPU address of the the IB */
104 __u32 ptbase; /* Base for the pagetable the GPU address is valid in */
105 int size; /* Size of the IB */
106} __packed;
107
108/* Register sub-section header */
109struct kgsl_snapshot_regs {
110 __u32 count; /* Number of register pairs in the section */
111} __packed;
112
113/* Indexed register sub-section header */
114struct kgsl_snapshot_indexed_regs {
115 __u32 index_reg; /* Offset of the index register for this section */
116 __u32 data_reg; /* Offset of the data register for this section */
117 int start; /* Starting index */
118 int count; /* Number of dwords in the data */
119} __packed;
120
121/* Istore sub-section header */
122struct kgsl_snapshot_istore {
123 int count; /* Number of instructions in the istore */
124} __packed;
125
126/* Debug data sub-section header */
127
Jordan Crouse0c2761a2012-02-01 22:11:12 -0700128/* A2XX debug sections */
Jordan Crouse156cfbc2012-01-24 09:32:04 -0700129#define SNAPSHOT_DEBUG_SX 1
130#define SNAPSHOT_DEBUG_CP 2
131#define SNAPSHOT_DEBUG_SQ 3
132#define SNAPSHOT_DEBUG_SQTHREAD 4
133#define SNAPSHOT_DEBUG_MIU 5
134
Jordan Crouse0c2761a2012-02-01 22:11:12 -0700135/* A3XX debug sections */
136#define SNAPSHOT_DEBUG_VPC_MEMORY 6
137#define SNAPSHOT_DEBUG_CP_MEQ 7
138#define SNAPSHOT_DEBUG_CP_PM4_RAM 8
139#define SNAPSHOT_DEBUG_CP_PFP_RAM 9
140#define SNAPSHOT_DEBUG_CP_ROQ 10
141
Jordan Crouse156cfbc2012-01-24 09:32:04 -0700142struct kgsl_snapshot_debug {
143 int type; /* Type identifier for the attached tata */
Jordan Crouse0c2761a2012-02-01 22:11:12 -0700144 int size; /* Size of the section in dwords */
145} __packed;
146
147struct kgsl_snapshot_debugbus {
148 int id; /* Debug bus ID */
149 int count; /* Number of dwords in the dump */
Jordan Crouse156cfbc2012-01-24 09:32:04 -0700150} __packed;
151
152#ifdef __KERNEL__
153
154/* Allocate 512K for each device snapshot */
155#define KGSL_SNAPSHOT_MEMSIZE (512 * 1024)
156
157struct kgsl_device;
158/*
159 * A helper macro to print out "not enough memory functions" - this
160 * makes it easy to standardize the messages as well as cut down on
161 * the number of strings in the binary
162 */
163
164#define SNAPSHOT_ERR_NOMEM(_d, _s) \
165 KGSL_DRV_ERR((_d), \
166 "snapshot: not enough snapshot memory for section %s\n", (_s))
167
168/*
169 * kgsl_snapshot_add_section - Add a new section to the GPU snapshot
170 * @device - the KGSL device being snapshotted
171 * @id - the section id
172 * @snapshot - pointer to the memory for the snapshot
173 * @remain - pointer to the number of bytes left in the snapshot region
174 * @func - Function pointer to fill the section
175 * @priv - Priv pointer to pass to the function
176 *
177 * Set up a KGSL snapshot header by filling the memory with the callback
178 * function and adding the standard section header
179 */
180
181static inline void *kgsl_snapshot_add_section(struct kgsl_device *device,
182 u16 id, void *snapshot, int *remain,
183 int (*func)(struct kgsl_device *, void *, int, void *), void *priv)
184{
185 struct kgsl_snapshot_section_header *header = snapshot;
186 void *data = snapshot + sizeof(*header);
187 int ret = 0;
188
189 /*
190 * Sanity check to make sure there is enough for the header. The
191 * callback will check to make sure there is enough for the rest
192 * of the data. If there isn't enough room then don't advance the
193 * pointer.
194 */
195
196 if (*remain < sizeof(*header))
197 return snapshot;
198
199 /* It is legal to have no function (i.e. - make an empty section) */
200
201 if (func) {
202 ret = func(device, data, *remain, priv);
203
204 /*
205 * If there wasn't enough room for the data then don't bother
206 * setting up the header.
207 */
208
209 if (ret == 0)
210 return snapshot;
211 }
212
213 header->magic = SNAPSHOT_SECTION_MAGIC;
214 header->id = id;
215 header->size = ret + sizeof(*header);
216
217 /* Decrement the room left in the snapshot region */
218 *remain -= header->size;
219 /* Advance the pointer to the end of the next function */
220 return snapshot + header->size;
221}
222
223/* A common helper function to dump a range of registers. This will be used in
224 * the GPU specific devices like this:
225 *
226 * struct kgsl_snapshot_registers priv;
227 * priv.regs = registers_array;;
228 * priv.count = num_registers;
229 *
230 * kgsl_snapshot_add_section(device, KGSL_SNAPSHOT_SECTION_REGS, snapshot,
231 * remain, kgsl_snapshot_dump_regs, &priv).
232 *
233 * Pass in an array of register range pairs in the form of:
234 * start reg, stop reg
235 * All the registers between start and stop inclusive will be dumped
236 */
237
238struct kgsl_snapshot_registers {
239 unsigned int *regs; /* Pointer to the array of register ranges */
240 int count; /* Number of entries in the array */
241};
242
243int kgsl_snapshot_dump_regs(struct kgsl_device *device, void *snapshot,
244 int remain, void *priv);
245
246/*
247 * A common helper function to dump a set of indexed registers. Use it
248 * like this:
249 *
250 * struct kgsl_snapshot_indexed_registers priv;
251 * priv.index = REG_INDEX;
252 * priv.data = REG_DATA;
253 * priv.count = num_registers
254 *
255 * kgsl_snapshot_add_section(device, KGSL_SNAPSHOT_SECTION_INDEXED_REGS,
256 * snapshot, remain, kgsl_snapshot_dump_indexed_regs, &priv).
257 *
258 * The callback function will write an index from 0 to priv.count to
259 * the index register and read the data from the data register.
260 */
261
262struct kgsl_snapshot_indexed_registers {
263 unsigned int index; /* Offset of the index register */
264 unsigned int data; /* Offset of the data register */
265 unsigned int start; /* Index to start with */
266 unsigned int count; /* Number of values to read from the pair */
267};
268
Jordan Crouse0c2761a2012-02-01 22:11:12 -0700269/* Helper function to snapshot a section of indexed registers */
270
271void *kgsl_snapshot_indexed_registers(struct kgsl_device *device,
272 void *snapshot, int *remain, unsigned int index,
273 unsigned int data, unsigned int start, unsigned int count);
274
Jordan Crouse156cfbc2012-01-24 09:32:04 -0700275
276#endif
277#endif