175758255SAlex Deucher /*
275758255SAlex Deucher * Copyright 2008 Advanced Micro Devices, Inc.
375758255SAlex Deucher * Copyright 2008 Red Hat Inc.
475758255SAlex Deucher * Copyright 2009 Jerome Glisse.
575758255SAlex Deucher *
675758255SAlex Deucher * Permission is hereby granted, free of charge, to any person obtaining a
775758255SAlex Deucher * copy of this software and associated documentation files (the "Software"),
875758255SAlex Deucher * to deal in the Software without restriction, including without limitation
975758255SAlex Deucher * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1075758255SAlex Deucher * and/or sell copies of the Software, and to permit persons to whom the
1175758255SAlex Deucher * Software is furnished to do so, subject to the following conditions:
1275758255SAlex Deucher *
1375758255SAlex Deucher * The above copyright notice and this permission notice shall be included in
1475758255SAlex Deucher * all copies or substantial portions of the Software.
1575758255SAlex Deucher *
1675758255SAlex Deucher * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1775758255SAlex Deucher * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1875758255SAlex Deucher * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1975758255SAlex Deucher * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
2075758255SAlex Deucher * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
2175758255SAlex Deucher * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
2275758255SAlex Deucher * OTHER DEALINGS IN THE SOFTWARE.
2375758255SAlex Deucher *
2475758255SAlex Deucher */
2575758255SAlex Deucher
2675758255SAlex Deucher #include <linux/kthread.h>
27fdf2f6c5SSam Ravnborg #include <linux/pci.h>
28fdf2f6c5SSam Ravnborg #include <linux/uaccess.h>
29a9ffe2a9SAlex Deucher #include <linux/pm_runtime.h>
30fdf2f6c5SSam Ravnborg
3175758255SAlex Deucher #include "amdgpu.h"
32a4c5b1bbSAlex Deucher #include "amdgpu_pm.h"
33d090e7dbSAlex Deucher #include "amdgpu_dm_debugfs.h"
3417cb04f2SStanley.Yang #include "amdgpu_ras.h"
35a4322e18SWenhui Sheng #include "amdgpu_rap.h"
36ecaafb7bSJinzhou Su #include "amdgpu_securedisplay.h"
3719ae3330SJohn Clements #include "amdgpu_fw_attestation.h"
3837df9560STom St Denis #include "amdgpu_umr.h"
3975758255SAlex Deucher
40d0fb18b5SAndrey Grodzovsky #include "amdgpu_reset.h"
41e50d9ba0SCandice Li #include "amdgpu_psp_ta.h"
42d0fb18b5SAndrey Grodzovsky
4375758255SAlex Deucher #if defined(CONFIG_DEBUG_FS)
4475758255SAlex Deucher
457e4237dbSTom St Denis /**
467e4237dbSTom St Denis * amdgpu_debugfs_process_reg_op - Handle MMIO register reads/writes
477e4237dbSTom St Denis *
487e4237dbSTom St Denis * @read: True if reading
497e4237dbSTom St Denis * @f: open file handle
507e4237dbSTom St Denis * @buf: User buffer to write/read to
517e4237dbSTom St Denis * @size: Number of bytes to write/read
527e4237dbSTom St Denis * @pos: Offset to seek to
537e4237dbSTom St Denis *
547e4237dbSTom St Denis * This debugfs entry has special meaning on the offset being sought.
557e4237dbSTom St Denis * Various bits have different meanings:
567e4237dbSTom St Denis *
577e4237dbSTom St Denis * Bit 62: Indicates a GRBM bank switch is needed
587e4237dbSTom St Denis * Bit 61: Indicates a SRBM bank switch is needed (implies bit 62 is
597e4237dbSTom St Denis * zero)
607e4237dbSTom St Denis * Bits 24..33: The SE or ME selector if needed
617e4237dbSTom St Denis * Bits 34..43: The SH (or SA) or PIPE selector if needed
627e4237dbSTom St Denis * Bits 44..53: The INSTANCE (or CU/WGP) or QUEUE selector if needed
637e4237dbSTom St Denis *
647e4237dbSTom St Denis * Bit 23: Indicates that the PM power gating lock should be held
657e4237dbSTom St Denis * This is necessary to read registers that might be
667e4237dbSTom St Denis * unreliable during a power gating transistion.
677e4237dbSTom St Denis *
687e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to read. This
697e4237dbSTom St Denis * allows reading multiple registers in a single call and having
707e4237dbSTom St Denis * the returned size reflect that.
717e4237dbSTom St Denis */
amdgpu_debugfs_process_reg_op(bool read,struct file * f,char __user * buf,size_t size,loff_t * pos)72f7a9ee81SAndrey Grodzovsky static int amdgpu_debugfs_process_reg_op(bool read, struct file *f,
73f7a9ee81SAndrey Grodzovsky char __user *buf, size_t size, loff_t *pos)
7475758255SAlex Deucher {
7575758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
7675758255SAlex Deucher ssize_t result = 0;
7775758255SAlex Deucher int r;
78f7a9ee81SAndrey Grodzovsky bool pm_pg_lock, use_bank, use_ring;
798fa76350SSrinivasan Shanmugam unsigned int instance_bank, sh_bank, se_bank, me, pipe, queue, vmid;
8075758255SAlex Deucher
81f7a9ee81SAndrey Grodzovsky pm_pg_lock = use_bank = use_ring = false;
820fa4246eSTom St Denis instance_bank = sh_bank = se_bank = me = pipe = queue = vmid = 0;
83f7a9ee81SAndrey Grodzovsky
84f7a9ee81SAndrey Grodzovsky if (size & 0x3 || *pos & 0x3 ||
85f7a9ee81SAndrey Grodzovsky ((*pos & (1ULL << 62)) && (*pos & (1ULL << 61))))
8675758255SAlex Deucher return -EINVAL;
8775758255SAlex Deucher
8875758255SAlex Deucher /* are we reading registers for which a PG lock is necessary? */
8975758255SAlex Deucher pm_pg_lock = (*pos >> 23) & 1;
9075758255SAlex Deucher
9175758255SAlex Deucher if (*pos & (1ULL << 62)) {
9275758255SAlex Deucher se_bank = (*pos & GENMASK_ULL(33, 24)) >> 24;
9375758255SAlex Deucher sh_bank = (*pos & GENMASK_ULL(43, 34)) >> 34;
9475758255SAlex Deucher instance_bank = (*pos & GENMASK_ULL(53, 44)) >> 44;
9575758255SAlex Deucher
9675758255SAlex Deucher if (se_bank == 0x3FF)
9775758255SAlex Deucher se_bank = 0xFFFFFFFF;
9875758255SAlex Deucher if (sh_bank == 0x3FF)
9975758255SAlex Deucher sh_bank = 0xFFFFFFFF;
10075758255SAlex Deucher if (instance_bank == 0x3FF)
10175758255SAlex Deucher instance_bank = 0xFFFFFFFF;
102c5b2bd5dSzhengbin use_bank = true;
103f7a9ee81SAndrey Grodzovsky } else if (*pos & (1ULL << 61)) {
104f7a9ee81SAndrey Grodzovsky
105f7a9ee81SAndrey Grodzovsky me = (*pos & GENMASK_ULL(33, 24)) >> 24;
106f7a9ee81SAndrey Grodzovsky pipe = (*pos & GENMASK_ULL(43, 34)) >> 34;
107f7a9ee81SAndrey Grodzovsky queue = (*pos & GENMASK_ULL(53, 44)) >> 44;
10888891430STom St Denis vmid = (*pos & GENMASK_ULL(58, 54)) >> 54;
109f7a9ee81SAndrey Grodzovsky
110c5b2bd5dSzhengbin use_ring = true;
11175758255SAlex Deucher } else {
112c5b2bd5dSzhengbin use_bank = use_ring = false;
11375758255SAlex Deucher }
11475758255SAlex Deucher
11575758255SAlex Deucher *pos &= (1UL << 22) - 1;
11675758255SAlex Deucher
1174a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
1189eee152aSAlex Deucher if (r < 0) {
1194a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
120a9ffe2a9SAlex Deucher return r;
1219eee152aSAlex Deucher }
122a9ffe2a9SAlex Deucher
12395a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
1249eee152aSAlex Deucher if (r < 0) {
1254a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
12695a2f917SYintian Tao return r;
1279eee152aSAlex Deucher }
12895a2f917SYintian Tao
12975758255SAlex Deucher if (use_bank) {
13075758255SAlex Deucher if ((sh_bank != 0xFFFFFFFF && sh_bank >= adev->gfx.config.max_sh_per_se) ||
131a9ffe2a9SAlex Deucher (se_bank != 0xFFFFFFFF && se_bank >= adev->gfx.config.max_shader_engines)) {
1324a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
1334a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
13495a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
13575758255SAlex Deucher return -EINVAL;
136a9ffe2a9SAlex Deucher }
13775758255SAlex Deucher mutex_lock(&adev->grbm_idx_mutex);
13875758255SAlex Deucher amdgpu_gfx_select_se_sh(adev, se_bank,
139d51ac6d0SLe Ma sh_bank, instance_bank, 0);
140f7a9ee81SAndrey Grodzovsky } else if (use_ring) {
141f7a9ee81SAndrey Grodzovsky mutex_lock(&adev->srbm_mutex);
142553f973aSTom St Denis amdgpu_gfx_select_me_pipe_q(adev, me, pipe, queue, vmid, 0);
14375758255SAlex Deucher }
14475758255SAlex Deucher
14575758255SAlex Deucher if (pm_pg_lock)
14675758255SAlex Deucher mutex_lock(&adev->pm.mutex);
14775758255SAlex Deucher
14875758255SAlex Deucher while (size) {
14975758255SAlex Deucher uint32_t value;
15075758255SAlex Deucher
151f7a9ee81SAndrey Grodzovsky if (read) {
15275758255SAlex Deucher value = RREG32(*pos >> 2);
15375758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
154f7a9ee81SAndrey Grodzovsky } else {
155f7a9ee81SAndrey Grodzovsky r = get_user(value, (uint32_t *)buf);
156f7a9ee81SAndrey Grodzovsky if (!r)
1578ed49dd1SVictor Lu amdgpu_mm_wreg_mmio_rlc(adev, *pos >> 2, value, 0);
158f7a9ee81SAndrey Grodzovsky }
15975758255SAlex Deucher if (r) {
16075758255SAlex Deucher result = r;
16175758255SAlex Deucher goto end;
16275758255SAlex Deucher }
16375758255SAlex Deucher
16475758255SAlex Deucher result += 4;
16575758255SAlex Deucher buf += 4;
16675758255SAlex Deucher *pos += 4;
16775758255SAlex Deucher size -= 4;
16875758255SAlex Deucher }
16975758255SAlex Deucher
17075758255SAlex Deucher end:
17175758255SAlex Deucher if (use_bank) {
172d51ac6d0SLe Ma amdgpu_gfx_select_se_sh(adev, 0xffffffff, 0xffffffff, 0xffffffff, 0);
17375758255SAlex Deucher mutex_unlock(&adev->grbm_idx_mutex);
174f7a9ee81SAndrey Grodzovsky } else if (use_ring) {
175553f973aSTom St Denis amdgpu_gfx_select_me_pipe_q(adev, 0, 0, 0, 0, 0);
176f7a9ee81SAndrey Grodzovsky mutex_unlock(&adev->srbm_mutex);
17775758255SAlex Deucher }
17875758255SAlex Deucher
17975758255SAlex Deucher if (pm_pg_lock)
18075758255SAlex Deucher mutex_unlock(&adev->pm.mutex);
18175758255SAlex Deucher
1824a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
1834a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
184a9ffe2a9SAlex Deucher
18595a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
18675758255SAlex Deucher return result;
18775758255SAlex Deucher }
18875758255SAlex Deucher
18920ed491bSLee Jones /*
1907e4237dbSTom St Denis * amdgpu_debugfs_regs_read - Callback for reading MMIO registers
1917e4237dbSTom St Denis */
amdgpu_debugfs_regs_read(struct file * f,char __user * buf,size_t size,loff_t * pos)192f7a9ee81SAndrey Grodzovsky static ssize_t amdgpu_debugfs_regs_read(struct file *f, char __user *buf,
193f7a9ee81SAndrey Grodzovsky size_t size, loff_t *pos)
194f7a9ee81SAndrey Grodzovsky {
195f7a9ee81SAndrey Grodzovsky return amdgpu_debugfs_process_reg_op(true, f, buf, size, pos);
196f7a9ee81SAndrey Grodzovsky }
197f7a9ee81SAndrey Grodzovsky
19820ed491bSLee Jones /*
1997e4237dbSTom St Denis * amdgpu_debugfs_regs_write - Callback for writing MMIO registers
2007e4237dbSTom St Denis */
amdgpu_debugfs_regs_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)20175758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_write(struct file *f, const char __user *buf,
20275758255SAlex Deucher size_t size, loff_t *pos)
20375758255SAlex Deucher {
204f7a9ee81SAndrey Grodzovsky return amdgpu_debugfs_process_reg_op(false, f, (char __user *)buf, size, pos);
20575758255SAlex Deucher }
20675758255SAlex Deucher
amdgpu_debugfs_regs2_open(struct inode * inode,struct file * file)20737df9560STom St Denis static int amdgpu_debugfs_regs2_open(struct inode *inode, struct file *file)
20837df9560STom St Denis {
20937df9560STom St Denis struct amdgpu_debugfs_regs2_data *rd;
21037df9560STom St Denis
2118fa76350SSrinivasan Shanmugam rd = kzalloc(sizeof(*rd), GFP_KERNEL);
21237df9560STom St Denis if (!rd)
21337df9560STom St Denis return -ENOMEM;
21437df9560STom St Denis rd->adev = file_inode(file)->i_private;
21537df9560STom St Denis file->private_data = rd;
21637df9560STom St Denis mutex_init(&rd->lock);
21737df9560STom St Denis
21837df9560STom St Denis return 0;
21937df9560STom St Denis }
22037df9560STom St Denis
amdgpu_debugfs_regs2_release(struct inode * inode,struct file * file)22137df9560STom St Denis static int amdgpu_debugfs_regs2_release(struct inode *inode, struct file *file)
22237df9560STom St Denis {
22337df9560STom St Denis struct amdgpu_debugfs_regs2_data *rd = file->private_data;
2248fa76350SSrinivasan Shanmugam
22537df9560STom St Denis mutex_destroy(&rd->lock);
22637df9560STom St Denis kfree(file->private_data);
22737df9560STom St Denis return 0;
22837df9560STom St Denis }
22937df9560STom St Denis
amdgpu_debugfs_regs2_op(struct file * f,char __user * buf,u32 offset,size_t size,int write_en)23037df9560STom St Denis static ssize_t amdgpu_debugfs_regs2_op(struct file *f, char __user *buf, u32 offset, size_t size, int write_en)
23137df9560STom St Denis {
23237df9560STom St Denis struct amdgpu_debugfs_regs2_data *rd = f->private_data;
23337df9560STom St Denis struct amdgpu_device *adev = rd->adev;
23437df9560STom St Denis ssize_t result = 0;
23537df9560STom St Denis int r;
23637df9560STom St Denis uint32_t value;
23737df9560STom St Denis
23837df9560STom St Denis if (size & 0x3 || offset & 0x3)
23937df9560STom St Denis return -EINVAL;
24037df9560STom St Denis
24137df9560STom St Denis r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
24237df9560STom St Denis if (r < 0) {
24337df9560STom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
24437df9560STom St Denis return r;
24537df9560STom St Denis }
24637df9560STom St Denis
24737df9560STom St Denis r = amdgpu_virt_enable_access_debugfs(adev);
24837df9560STom St Denis if (r < 0) {
24937df9560STom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
25037df9560STom St Denis return r;
25137df9560STom St Denis }
25237df9560STom St Denis
25337df9560STom St Denis mutex_lock(&rd->lock);
25437df9560STom St Denis
25537df9560STom St Denis if (rd->id.use_grbm) {
25637df9560STom St Denis if ((rd->id.grbm.sh != 0xFFFFFFFF && rd->id.grbm.sh >= adev->gfx.config.max_sh_per_se) ||
25737df9560STom St Denis (rd->id.grbm.se != 0xFFFFFFFF && rd->id.grbm.se >= adev->gfx.config.max_shader_engines)) {
25837df9560STom St Denis pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
25937df9560STom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
26037df9560STom St Denis amdgpu_virt_disable_access_debugfs(adev);
26137df9560STom St Denis mutex_unlock(&rd->lock);
26237df9560STom St Denis return -EINVAL;
26337df9560STom St Denis }
26437df9560STom St Denis mutex_lock(&adev->grbm_idx_mutex);
26537df9560STom St Denis amdgpu_gfx_select_se_sh(adev, rd->id.grbm.se,
26637df9560STom St Denis rd->id.grbm.sh,
267553f973aSTom St Denis rd->id.grbm.instance, rd->id.xcc_id);
26837df9560STom St Denis }
26937df9560STom St Denis
27037df9560STom St Denis if (rd->id.use_srbm) {
27137df9560STom St Denis mutex_lock(&adev->srbm_mutex);
27237df9560STom St Denis amdgpu_gfx_select_me_pipe_q(adev, rd->id.srbm.me, rd->id.srbm.pipe,
273553f973aSTom St Denis rd->id.srbm.queue, rd->id.srbm.vmid, rd->id.xcc_id);
27437df9560STom St Denis }
27537df9560STom St Denis
27637df9560STom St Denis if (rd->id.pg_lock)
27737df9560STom St Denis mutex_lock(&adev->pm.mutex);
27837df9560STom St Denis
27937df9560STom St Denis while (size) {
28037df9560STom St Denis if (!write_en) {
28137df9560STom St Denis value = RREG32(offset >> 2);
28237df9560STom St Denis r = put_user(value, (uint32_t *)buf);
28337df9560STom St Denis } else {
28437df9560STom St Denis r = get_user(value, (uint32_t *)buf);
28537df9560STom St Denis if (!r)
2868ed49dd1SVictor Lu amdgpu_mm_wreg_mmio_rlc(adev, offset >> 2, value, rd->id.xcc_id);
28737df9560STom St Denis }
28837df9560STom St Denis if (r) {
28937df9560STom St Denis result = r;
29037df9560STom St Denis goto end;
29137df9560STom St Denis }
29237df9560STom St Denis offset += 4;
29337df9560STom St Denis size -= 4;
29437df9560STom St Denis result += 4;
29537df9560STom St Denis buf += 4;
29637df9560STom St Denis }
29737df9560STom St Denis end:
29837df9560STom St Denis if (rd->id.use_grbm) {
299553f973aSTom St Denis amdgpu_gfx_select_se_sh(adev, 0xffffffff, 0xffffffff, 0xffffffff, rd->id.xcc_id);
30037df9560STom St Denis mutex_unlock(&adev->grbm_idx_mutex);
30137df9560STom St Denis }
30237df9560STom St Denis
30337df9560STom St Denis if (rd->id.use_srbm) {
304553f973aSTom St Denis amdgpu_gfx_select_me_pipe_q(adev, 0, 0, 0, 0, rd->id.xcc_id);
30537df9560STom St Denis mutex_unlock(&adev->srbm_mutex);
30637df9560STom St Denis }
30737df9560STom St Denis
30837df9560STom St Denis if (rd->id.pg_lock)
30937df9560STom St Denis mutex_unlock(&adev->pm.mutex);
31037df9560STom St Denis
31137df9560STom St Denis mutex_unlock(&rd->lock);
31237df9560STom St Denis
31337df9560STom St Denis pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
31437df9560STom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
31537df9560STom St Denis
31637df9560STom St Denis amdgpu_virt_disable_access_debugfs(adev);
31737df9560STom St Denis return result;
31837df9560STom St Denis }
31937df9560STom St Denis
amdgpu_debugfs_regs2_ioctl(struct file * f,unsigned int cmd,unsigned long data)32037df9560STom St Denis static long amdgpu_debugfs_regs2_ioctl(struct file *f, unsigned int cmd, unsigned long data)
32137df9560STom St Denis {
32237df9560STom St Denis struct amdgpu_debugfs_regs2_data *rd = f->private_data;
323553f973aSTom St Denis struct amdgpu_debugfs_regs2_iocdata v1_data;
32437df9560STom St Denis int r;
32537df9560STom St Denis
32637df9560STom St Denis mutex_lock(&rd->lock);
327553f973aSTom St Denis
328553f973aSTom St Denis switch (cmd) {
329553f973aSTom St Denis case AMDGPU_DEBUGFS_REGS2_IOC_SET_STATE_V2:
330553f973aSTom St Denis r = copy_from_user(&rd->id, (struct amdgpu_debugfs_regs2_iocdata_v2 *)data,
3318fa76350SSrinivasan Shanmugam sizeof(rd->id));
332553f973aSTom St Denis if (r)
333553f973aSTom St Denis r = -EINVAL;
334553f973aSTom St Denis goto done;
335553f973aSTom St Denis case AMDGPU_DEBUGFS_REGS2_IOC_SET_STATE:
336553f973aSTom St Denis r = copy_from_user(&v1_data, (struct amdgpu_debugfs_regs2_iocdata *)data,
337553f973aSTom St Denis sizeof(v1_data));
338553f973aSTom St Denis if (r) {
339553f973aSTom St Denis r = -EINVAL;
340553f973aSTom St Denis goto done;
34137df9560STom St Denis }
342553f973aSTom St Denis goto v1_copy;
343553f973aSTom St Denis default:
344553f973aSTom St Denis r = -EINVAL;
345553f973aSTom St Denis goto done;
346553f973aSTom St Denis }
347553f973aSTom St Denis
348553f973aSTom St Denis v1_copy:
349553f973aSTom St Denis rd->id.use_srbm = v1_data.use_srbm;
350553f973aSTom St Denis rd->id.use_grbm = v1_data.use_grbm;
351553f973aSTom St Denis rd->id.pg_lock = v1_data.pg_lock;
352553f973aSTom St Denis rd->id.grbm.se = v1_data.grbm.se;
353553f973aSTom St Denis rd->id.grbm.sh = v1_data.grbm.sh;
354553f973aSTom St Denis rd->id.grbm.instance = v1_data.grbm.instance;
355553f973aSTom St Denis rd->id.srbm.me = v1_data.srbm.me;
356553f973aSTom St Denis rd->id.srbm.pipe = v1_data.srbm.pipe;
357553f973aSTom St Denis rd->id.srbm.queue = v1_data.srbm.queue;
358553f973aSTom St Denis rd->id.xcc_id = 0;
359553f973aSTom St Denis done:
360553f973aSTom St Denis mutex_unlock(&rd->lock);
361553f973aSTom St Denis return r;
36237df9560STom St Denis }
36337df9560STom St Denis
amdgpu_debugfs_regs2_read(struct file * f,char __user * buf,size_t size,loff_t * pos)36437df9560STom St Denis static ssize_t amdgpu_debugfs_regs2_read(struct file *f, char __user *buf, size_t size, loff_t *pos)
36537df9560STom St Denis {
36637df9560STom St Denis return amdgpu_debugfs_regs2_op(f, buf, *pos, size, 0);
36737df9560STom St Denis }
36837df9560STom St Denis
amdgpu_debugfs_regs2_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)36937df9560STom St Denis static ssize_t amdgpu_debugfs_regs2_write(struct file *f, const char __user *buf, size_t size, loff_t *pos)
37037df9560STom St Denis {
37137df9560STom St Denis return amdgpu_debugfs_regs2_op(f, (char __user *)buf, *pos, size, 1);
37237df9560STom St Denis }
37337df9560STom St Denis
amdgpu_debugfs_gprwave_open(struct inode * inode,struct file * file)374553f973aSTom St Denis static int amdgpu_debugfs_gprwave_open(struct inode *inode, struct file *file)
375553f973aSTom St Denis {
376553f973aSTom St Denis struct amdgpu_debugfs_gprwave_data *rd;
377553f973aSTom St Denis
378ad19c200SPraful Swarnakar rd = kzalloc(sizeof(*rd), GFP_KERNEL);
379553f973aSTom St Denis if (!rd)
380553f973aSTom St Denis return -ENOMEM;
381553f973aSTom St Denis rd->adev = file_inode(file)->i_private;
382553f973aSTom St Denis file->private_data = rd;
383553f973aSTom St Denis mutex_init(&rd->lock);
384553f973aSTom St Denis
385553f973aSTom St Denis return 0;
386553f973aSTom St Denis }
387553f973aSTom St Denis
amdgpu_debugfs_gprwave_release(struct inode * inode,struct file * file)388553f973aSTom St Denis static int amdgpu_debugfs_gprwave_release(struct inode *inode, struct file *file)
389553f973aSTom St Denis {
390553f973aSTom St Denis struct amdgpu_debugfs_gprwave_data *rd = file->private_data;
391ad19c200SPraful Swarnakar
392553f973aSTom St Denis mutex_destroy(&rd->lock);
393553f973aSTom St Denis kfree(file->private_data);
394553f973aSTom St Denis return 0;
395553f973aSTom St Denis }
396553f973aSTom St Denis
amdgpu_debugfs_gprwave_read(struct file * f,char __user * buf,size_t size,loff_t * pos)397553f973aSTom St Denis static ssize_t amdgpu_debugfs_gprwave_read(struct file *f, char __user *buf, size_t size, loff_t *pos)
398553f973aSTom St Denis {
399553f973aSTom St Denis struct amdgpu_debugfs_gprwave_data *rd = f->private_data;
400553f973aSTom St Denis struct amdgpu_device *adev = rd->adev;
401553f973aSTom St Denis ssize_t result = 0;
402553f973aSTom St Denis int r;
403553f973aSTom St Denis uint32_t *data, x;
404553f973aSTom St Denis
405f5d873f5SAlex Deucher if (size > 4096 || size & 0x3 || *pos & 0x3)
406553f973aSTom St Denis return -EINVAL;
407553f973aSTom St Denis
408553f973aSTom St Denis r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
409553f973aSTom St Denis if (r < 0) {
410553f973aSTom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
411553f973aSTom St Denis return r;
412553f973aSTom St Denis }
413553f973aSTom St Denis
414553f973aSTom St Denis r = amdgpu_virt_enable_access_debugfs(adev);
415553f973aSTom St Denis if (r < 0) {
416553f973aSTom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
417553f973aSTom St Denis return r;
418553f973aSTom St Denis }
419553f973aSTom St Denis
420553f973aSTom St Denis data = kcalloc(1024, sizeof(*data), GFP_KERNEL);
421553f973aSTom St Denis if (!data) {
422553f973aSTom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
423553f973aSTom St Denis amdgpu_virt_disable_access_debugfs(adev);
424553f973aSTom St Denis return -ENOMEM;
425553f973aSTom St Denis }
426553f973aSTom St Denis
427553f973aSTom St Denis /* switch to the specific se/sh/cu */
428553f973aSTom St Denis mutex_lock(&adev->grbm_idx_mutex);
429553f973aSTom St Denis amdgpu_gfx_select_se_sh(adev, rd->id.se, rd->id.sh, rd->id.cu, rd->id.xcc_id);
430553f973aSTom St Denis
431553f973aSTom St Denis if (!rd->id.gpr_or_wave) {
432553f973aSTom St Denis x = 0;
433553f973aSTom St Denis if (adev->gfx.funcs->read_wave_data)
434553f973aSTom St Denis adev->gfx.funcs->read_wave_data(adev, rd->id.xcc_id, rd->id.simd, rd->id.wave, data, &x);
435553f973aSTom St Denis } else {
436553f973aSTom St Denis x = size >> 2;
437553f973aSTom St Denis if (rd->id.gpr.vpgr_or_sgpr) {
438553f973aSTom St Denis if (adev->gfx.funcs->read_wave_vgprs)
439553f973aSTom St Denis adev->gfx.funcs->read_wave_vgprs(adev, rd->id.xcc_id, rd->id.simd, rd->id.wave, rd->id.gpr.thread, *pos, size>>2, data);
440553f973aSTom St Denis } else {
441553f973aSTom St Denis if (adev->gfx.funcs->read_wave_sgprs)
442553f973aSTom St Denis adev->gfx.funcs->read_wave_sgprs(adev, rd->id.xcc_id, rd->id.simd, rd->id.wave, *pos, size>>2, data);
443553f973aSTom St Denis }
444553f973aSTom St Denis }
445553f973aSTom St Denis
446553f973aSTom St Denis amdgpu_gfx_select_se_sh(adev, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, rd->id.xcc_id);
447553f973aSTom St Denis mutex_unlock(&adev->grbm_idx_mutex);
448553f973aSTom St Denis
449553f973aSTom St Denis pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
450553f973aSTom St Denis pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
451553f973aSTom St Denis
452553f973aSTom St Denis if (!x) {
453553f973aSTom St Denis result = -EINVAL;
454553f973aSTom St Denis goto done;
455553f973aSTom St Denis }
456553f973aSTom St Denis
457553f973aSTom St Denis while (size && (*pos < x * 4)) {
458553f973aSTom St Denis uint32_t value;
459553f973aSTom St Denis
460553f973aSTom St Denis value = data[*pos >> 2];
461553f973aSTom St Denis r = put_user(value, (uint32_t *)buf);
462553f973aSTom St Denis if (r) {
463553f973aSTom St Denis result = r;
464553f973aSTom St Denis goto done;
465553f973aSTom St Denis }
466553f973aSTom St Denis
467553f973aSTom St Denis result += 4;
468553f973aSTom St Denis buf += 4;
469553f973aSTom St Denis *pos += 4;
470553f973aSTom St Denis size -= 4;
471553f973aSTom St Denis }
472553f973aSTom St Denis
473553f973aSTom St Denis done:
474553f973aSTom St Denis amdgpu_virt_disable_access_debugfs(adev);
475553f973aSTom St Denis kfree(data);
476553f973aSTom St Denis return result;
477553f973aSTom St Denis }
478553f973aSTom St Denis
amdgpu_debugfs_gprwave_ioctl(struct file * f,unsigned int cmd,unsigned long data)479553f973aSTom St Denis static long amdgpu_debugfs_gprwave_ioctl(struct file *f, unsigned int cmd, unsigned long data)
480553f973aSTom St Denis {
481553f973aSTom St Denis struct amdgpu_debugfs_gprwave_data *rd = f->private_data;
4829c9d501bSDan Carpenter int r = 0;
483553f973aSTom St Denis
484553f973aSTom St Denis mutex_lock(&rd->lock);
485553f973aSTom St Denis
486553f973aSTom St Denis switch (cmd) {
487553f973aSTom St Denis case AMDGPU_DEBUGFS_GPRWAVE_IOC_SET_STATE:
4889c9d501bSDan Carpenter if (copy_from_user(&rd->id,
4899c9d501bSDan Carpenter (struct amdgpu_debugfs_gprwave_iocdata *)data,
4909c9d501bSDan Carpenter sizeof(rd->id)))
4919c9d501bSDan Carpenter r = -EFAULT;
492553f973aSTom St Denis goto done;
493553f973aSTom St Denis default:
494553f973aSTom St Denis r = -EINVAL;
495553f973aSTom St Denis goto done;
496553f973aSTom St Denis }
497553f973aSTom St Denis
498553f973aSTom St Denis done:
499553f973aSTom St Denis mutex_unlock(&rd->lock);
500553f973aSTom St Denis return r;
501553f973aSTom St Denis }
502553f973aSTom St Denis
503553f973aSTom St Denis
504553f973aSTom St Denis
5057e4237dbSTom St Denis
5067e4237dbSTom St Denis /**
5077e4237dbSTom St Denis * amdgpu_debugfs_regs_pcie_read - Read from a PCIE register
5087e4237dbSTom St Denis *
5097e4237dbSTom St Denis * @f: open file handle
5107e4237dbSTom St Denis * @buf: User buffer to store read data in
5117e4237dbSTom St Denis * @size: Number of bytes to read
5127e4237dbSTom St Denis * @pos: Offset to seek to
5137e4237dbSTom St Denis *
5147e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to read. This
5157e4237dbSTom St Denis * allows reading multiple registers in a single call and having
5167e4237dbSTom St Denis * the returned size reflect that.
5177e4237dbSTom St Denis */
amdgpu_debugfs_regs_pcie_read(struct file * f,char __user * buf,size_t size,loff_t * pos)51875758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_pcie_read(struct file *f, char __user *buf,
51975758255SAlex Deucher size_t size, loff_t *pos)
52075758255SAlex Deucher {
52175758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
52275758255SAlex Deucher ssize_t result = 0;
52375758255SAlex Deucher int r;
52475758255SAlex Deucher
52575758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
52675758255SAlex Deucher return -EINVAL;
52775758255SAlex Deucher
5284a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
5299eee152aSAlex Deucher if (r < 0) {
5304a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
531a9ffe2a9SAlex Deucher return r;
5329eee152aSAlex Deucher }
533a9ffe2a9SAlex Deucher
53495a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
5359eee152aSAlex Deucher if (r < 0) {
5364a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
53795a2f917SYintian Tao return r;
5389eee152aSAlex Deucher }
53995a2f917SYintian Tao
54075758255SAlex Deucher while (size) {
54175758255SAlex Deucher uint32_t value;
54275758255SAlex Deucher
5435eb8094aSMangesh Gadre if (upper_32_bits(*pos))
5445eb8094aSMangesh Gadre value = RREG32_PCIE_EXT(*pos);
5455eb8094aSMangesh Gadre else
54643fb6c19SKevin Wang value = RREG32_PCIE(*pos);
5475eb8094aSMangesh Gadre
54875758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
549edadd6fcSAndré Almeida if (r)
550edadd6fcSAndré Almeida goto out;
55175758255SAlex Deucher
55275758255SAlex Deucher result += 4;
55375758255SAlex Deucher buf += 4;
55475758255SAlex Deucher *pos += 4;
55575758255SAlex Deucher size -= 4;
55675758255SAlex Deucher }
55775758255SAlex Deucher
558edadd6fcSAndré Almeida r = result;
559edadd6fcSAndré Almeida out:
5604a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
5614a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
56295a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
563edadd6fcSAndré Almeida return r;
56475758255SAlex Deucher }
56575758255SAlex Deucher
5667e4237dbSTom St Denis /**
5677e4237dbSTom St Denis * amdgpu_debugfs_regs_pcie_write - Write to a PCIE register
5687e4237dbSTom St Denis *
5697e4237dbSTom St Denis * @f: open file handle
5707e4237dbSTom St Denis * @buf: User buffer to write data from
5717e4237dbSTom St Denis * @size: Number of bytes to write
5727e4237dbSTom St Denis * @pos: Offset to seek to
5737e4237dbSTom St Denis *
5747e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to write. This
5757e4237dbSTom St Denis * allows writing multiple registers in a single call and having
5767e4237dbSTom St Denis * the returned size reflect that.
5777e4237dbSTom St Denis */
amdgpu_debugfs_regs_pcie_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)57875758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_pcie_write(struct file *f, const char __user *buf,
57975758255SAlex Deucher size_t size, loff_t *pos)
58075758255SAlex Deucher {
58175758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
58275758255SAlex Deucher ssize_t result = 0;
58375758255SAlex Deucher int r;
58475758255SAlex Deucher
58575758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
58675758255SAlex Deucher return -EINVAL;
58775758255SAlex Deucher
5884a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
5899eee152aSAlex Deucher if (r < 0) {
5904a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
591a9ffe2a9SAlex Deucher return r;
5929eee152aSAlex Deucher }
593a9ffe2a9SAlex Deucher
59495a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
5959eee152aSAlex Deucher if (r < 0) {
5964a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
59795a2f917SYintian Tao return r;
5989eee152aSAlex Deucher }
59995a2f917SYintian Tao
60075758255SAlex Deucher while (size) {
60175758255SAlex Deucher uint32_t value;
60275758255SAlex Deucher
60375758255SAlex Deucher r = get_user(value, (uint32_t *)buf);
604edadd6fcSAndré Almeida if (r)
605edadd6fcSAndré Almeida goto out;
60675758255SAlex Deucher
6075eb8094aSMangesh Gadre if (upper_32_bits(*pos))
6085eb8094aSMangesh Gadre WREG32_PCIE_EXT(*pos, value);
6095eb8094aSMangesh Gadre else
61043fb6c19SKevin Wang WREG32_PCIE(*pos, value);
61175758255SAlex Deucher
61275758255SAlex Deucher result += 4;
61375758255SAlex Deucher buf += 4;
61475758255SAlex Deucher *pos += 4;
61575758255SAlex Deucher size -= 4;
61675758255SAlex Deucher }
61775758255SAlex Deucher
618edadd6fcSAndré Almeida r = result;
619edadd6fcSAndré Almeida out:
6204a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
6214a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
62295a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
623edadd6fcSAndré Almeida return r;
62475758255SAlex Deucher }
62575758255SAlex Deucher
6267e4237dbSTom St Denis /**
6277e4237dbSTom St Denis * amdgpu_debugfs_regs_didt_read - Read from a DIDT register
6287e4237dbSTom St Denis *
6297e4237dbSTom St Denis * @f: open file handle
6307e4237dbSTom St Denis * @buf: User buffer to store read data in
6317e4237dbSTom St Denis * @size: Number of bytes to read
6327e4237dbSTom St Denis * @pos: Offset to seek to
6337e4237dbSTom St Denis *
6347e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to read. This
6357e4237dbSTom St Denis * allows reading multiple registers in a single call and having
6367e4237dbSTom St Denis * the returned size reflect that.
6377e4237dbSTom St Denis */
amdgpu_debugfs_regs_didt_read(struct file * f,char __user * buf,size_t size,loff_t * pos)63875758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_didt_read(struct file *f, char __user *buf,
63975758255SAlex Deucher size_t size, loff_t *pos)
64075758255SAlex Deucher {
64175758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
64275758255SAlex Deucher ssize_t result = 0;
64375758255SAlex Deucher int r;
64475758255SAlex Deucher
64575758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
64675758255SAlex Deucher return -EINVAL;
64775758255SAlex Deucher
6487b194fdcSLu Yao if (!adev->didt_rreg)
6497b194fdcSLu Yao return -EOPNOTSUPP;
6507b194fdcSLu Yao
6514a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
6529eee152aSAlex Deucher if (r < 0) {
6534a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
654a9ffe2a9SAlex Deucher return r;
6559eee152aSAlex Deucher }
656a9ffe2a9SAlex Deucher
65795a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
6589eee152aSAlex Deucher if (r < 0) {
6594a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
66095a2f917SYintian Tao return r;
6619eee152aSAlex Deucher }
66295a2f917SYintian Tao
66375758255SAlex Deucher while (size) {
66475758255SAlex Deucher uint32_t value;
66575758255SAlex Deucher
66675758255SAlex Deucher value = RREG32_DIDT(*pos >> 2);
66775758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
668edadd6fcSAndré Almeida if (r)
669edadd6fcSAndré Almeida goto out;
67075758255SAlex Deucher
67175758255SAlex Deucher result += 4;
67275758255SAlex Deucher buf += 4;
67375758255SAlex Deucher *pos += 4;
67475758255SAlex Deucher size -= 4;
67575758255SAlex Deucher }
67675758255SAlex Deucher
677edadd6fcSAndré Almeida r = result;
678edadd6fcSAndré Almeida out:
6794a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
6804a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
68195a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
682edadd6fcSAndré Almeida return r;
68375758255SAlex Deucher }
68475758255SAlex Deucher
6857e4237dbSTom St Denis /**
6867e4237dbSTom St Denis * amdgpu_debugfs_regs_didt_write - Write to a DIDT register
6877e4237dbSTom St Denis *
6887e4237dbSTom St Denis * @f: open file handle
6897e4237dbSTom St Denis * @buf: User buffer to write data from
6907e4237dbSTom St Denis * @size: Number of bytes to write
6917e4237dbSTom St Denis * @pos: Offset to seek to
6927e4237dbSTom St Denis *
6937e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to write. This
6947e4237dbSTom St Denis * allows writing multiple registers in a single call and having
6957e4237dbSTom St Denis * the returned size reflect that.
6967e4237dbSTom St Denis */
amdgpu_debugfs_regs_didt_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)69775758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_didt_write(struct file *f, const char __user *buf,
69875758255SAlex Deucher size_t size, loff_t *pos)
69975758255SAlex Deucher {
70075758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
70175758255SAlex Deucher ssize_t result = 0;
70275758255SAlex Deucher int r;
70375758255SAlex Deucher
70475758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
70575758255SAlex Deucher return -EINVAL;
70675758255SAlex Deucher
7077b194fdcSLu Yao if (!adev->didt_wreg)
7087b194fdcSLu Yao return -EOPNOTSUPP;
7097b194fdcSLu Yao
7104a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
7119eee152aSAlex Deucher if (r < 0) {
7124a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
713a9ffe2a9SAlex Deucher return r;
7149eee152aSAlex Deucher }
715a9ffe2a9SAlex Deucher
71695a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
7179eee152aSAlex Deucher if (r < 0) {
7184a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
71995a2f917SYintian Tao return r;
7209eee152aSAlex Deucher }
72195a2f917SYintian Tao
72275758255SAlex Deucher while (size) {
72375758255SAlex Deucher uint32_t value;
72475758255SAlex Deucher
72575758255SAlex Deucher r = get_user(value, (uint32_t *)buf);
726edadd6fcSAndré Almeida if (r)
727edadd6fcSAndré Almeida goto out;
72875758255SAlex Deucher
72975758255SAlex Deucher WREG32_DIDT(*pos >> 2, value);
73075758255SAlex Deucher
73175758255SAlex Deucher result += 4;
73275758255SAlex Deucher buf += 4;
73375758255SAlex Deucher *pos += 4;
73475758255SAlex Deucher size -= 4;
73575758255SAlex Deucher }
73675758255SAlex Deucher
737edadd6fcSAndré Almeida r = result;
738edadd6fcSAndré Almeida out:
7394a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
7404a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
74195a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
742edadd6fcSAndré Almeida return r;
74375758255SAlex Deucher }
74475758255SAlex Deucher
7457e4237dbSTom St Denis /**
7467e4237dbSTom St Denis * amdgpu_debugfs_regs_smc_read - Read from a SMC register
7477e4237dbSTom St Denis *
7487e4237dbSTom St Denis * @f: open file handle
7497e4237dbSTom St Denis * @buf: User buffer to store read data in
7507e4237dbSTom St Denis * @size: Number of bytes to read
7517e4237dbSTom St Denis * @pos: Offset to seek to
7527e4237dbSTom St Denis *
7537e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to read. This
7547e4237dbSTom St Denis * allows reading multiple registers in a single call and having
7557e4237dbSTom St Denis * the returned size reflect that.
7567e4237dbSTom St Denis */
amdgpu_debugfs_regs_smc_read(struct file * f,char __user * buf,size_t size,loff_t * pos)75775758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_smc_read(struct file *f, char __user *buf,
75875758255SAlex Deucher size_t size, loff_t *pos)
75975758255SAlex Deucher {
76075758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
76175758255SAlex Deucher ssize_t result = 0;
76275758255SAlex Deucher int r;
76375758255SAlex Deucher
7645104fdf5SQu Huang if (!adev->smc_rreg)
765afe58346SAlex Deucher return -EOPNOTSUPP;
7665104fdf5SQu Huang
76775758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
76875758255SAlex Deucher return -EINVAL;
76975758255SAlex Deucher
7704a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
7719eee152aSAlex Deucher if (r < 0) {
7724a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
773a9ffe2a9SAlex Deucher return r;
7749eee152aSAlex Deucher }
775a9ffe2a9SAlex Deucher
77695a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
7779eee152aSAlex Deucher if (r < 0) {
7784a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
77995a2f917SYintian Tao return r;
7809eee152aSAlex Deucher }
78195a2f917SYintian Tao
78275758255SAlex Deucher while (size) {
78375758255SAlex Deucher uint32_t value;
78475758255SAlex Deucher
78575758255SAlex Deucher value = RREG32_SMC(*pos);
78675758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
787edadd6fcSAndré Almeida if (r)
788edadd6fcSAndré Almeida goto out;
78975758255SAlex Deucher
79075758255SAlex Deucher result += 4;
79175758255SAlex Deucher buf += 4;
79275758255SAlex Deucher *pos += 4;
79375758255SAlex Deucher size -= 4;
79475758255SAlex Deucher }
79575758255SAlex Deucher
796edadd6fcSAndré Almeida r = result;
797edadd6fcSAndré Almeida out:
7984a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
7994a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
80095a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
801edadd6fcSAndré Almeida return r;
80275758255SAlex Deucher }
80375758255SAlex Deucher
8047e4237dbSTom St Denis /**
8057e4237dbSTom St Denis * amdgpu_debugfs_regs_smc_write - Write to a SMC register
8067e4237dbSTom St Denis *
8077e4237dbSTom St Denis * @f: open file handle
8087e4237dbSTom St Denis * @buf: User buffer to write data from
8097e4237dbSTom St Denis * @size: Number of bytes to write
8107e4237dbSTom St Denis * @pos: Offset to seek to
8117e4237dbSTom St Denis *
8127e4237dbSTom St Denis * The lower bits are the BYTE offset of the register to write. This
8137e4237dbSTom St Denis * allows writing multiple registers in a single call and having
8147e4237dbSTom St Denis * the returned size reflect that.
8157e4237dbSTom St Denis */
amdgpu_debugfs_regs_smc_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)81675758255SAlex Deucher static ssize_t amdgpu_debugfs_regs_smc_write(struct file *f, const char __user *buf,
81775758255SAlex Deucher size_t size, loff_t *pos)
81875758255SAlex Deucher {
81975758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
82075758255SAlex Deucher ssize_t result = 0;
82175758255SAlex Deucher int r;
82275758255SAlex Deucher
8235104fdf5SQu Huang if (!adev->smc_wreg)
824afe58346SAlex Deucher return -EOPNOTSUPP;
8255104fdf5SQu Huang
82675758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
82775758255SAlex Deucher return -EINVAL;
82875758255SAlex Deucher
8294a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
8309eee152aSAlex Deucher if (r < 0) {
8314a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
832a9ffe2a9SAlex Deucher return r;
8339eee152aSAlex Deucher }
834a9ffe2a9SAlex Deucher
83595a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
8369eee152aSAlex Deucher if (r < 0) {
8374a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
83895a2f917SYintian Tao return r;
8399eee152aSAlex Deucher }
84095a2f917SYintian Tao
84175758255SAlex Deucher while (size) {
84275758255SAlex Deucher uint32_t value;
84375758255SAlex Deucher
84475758255SAlex Deucher r = get_user(value, (uint32_t *)buf);
845edadd6fcSAndré Almeida if (r)
846edadd6fcSAndré Almeida goto out;
84775758255SAlex Deucher
84875758255SAlex Deucher WREG32_SMC(*pos, value);
84975758255SAlex Deucher
85075758255SAlex Deucher result += 4;
85175758255SAlex Deucher buf += 4;
85275758255SAlex Deucher *pos += 4;
85375758255SAlex Deucher size -= 4;
85475758255SAlex Deucher }
85575758255SAlex Deucher
856edadd6fcSAndré Almeida r = result;
857edadd6fcSAndré Almeida out:
8584a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
8594a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
86095a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
861edadd6fcSAndré Almeida return r;
86275758255SAlex Deucher }
86375758255SAlex Deucher
8647e4237dbSTom St Denis /**
8657e4237dbSTom St Denis * amdgpu_debugfs_gca_config_read - Read from gfx config data
8667e4237dbSTom St Denis *
8677e4237dbSTom St Denis * @f: open file handle
8687e4237dbSTom St Denis * @buf: User buffer to store read data in
8697e4237dbSTom St Denis * @size: Number of bytes to read
8707e4237dbSTom St Denis * @pos: Offset to seek to
8717e4237dbSTom St Denis *
8727e4237dbSTom St Denis * This file is used to access configuration data in a somewhat
8737e4237dbSTom St Denis * stable fashion. The format is a series of DWORDs with the first
8747e4237dbSTom St Denis * indicating which revision it is. New content is appended to the
8757e4237dbSTom St Denis * end so that older software can still read the data.
8767e4237dbSTom St Denis */
8777e4237dbSTom St Denis
amdgpu_debugfs_gca_config_read(struct file * f,char __user * buf,size_t size,loff_t * pos)87875758255SAlex Deucher static ssize_t amdgpu_debugfs_gca_config_read(struct file *f, char __user *buf,
87975758255SAlex Deucher size_t size, loff_t *pos)
88075758255SAlex Deucher {
88175758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
88275758255SAlex Deucher ssize_t result = 0;
88375758255SAlex Deucher int r;
88475758255SAlex Deucher uint32_t *config, no_regs = 0;
88575758255SAlex Deucher
88675758255SAlex Deucher if (size & 0x3 || *pos & 0x3)
88775758255SAlex Deucher return -EINVAL;
88875758255SAlex Deucher
88975758255SAlex Deucher config = kmalloc_array(256, sizeof(*config), GFP_KERNEL);
89075758255SAlex Deucher if (!config)
89175758255SAlex Deucher return -ENOMEM;
89275758255SAlex Deucher
89375758255SAlex Deucher /* version, increment each time something is added */
894dc2947b3STom St Denis config[no_regs++] = 5;
89575758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_shader_engines;
89675758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_tile_pipes;
89775758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_cu_per_sh;
89875758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_sh_per_se;
89975758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_backends_per_se;
90075758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_texture_channel_caches;
90175758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_gprs;
90275758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_gs_threads;
90375758255SAlex Deucher config[no_regs++] = adev->gfx.config.max_hw_contexts;
90475758255SAlex Deucher config[no_regs++] = adev->gfx.config.sc_prim_fifo_size_frontend;
90575758255SAlex Deucher config[no_regs++] = adev->gfx.config.sc_prim_fifo_size_backend;
90675758255SAlex Deucher config[no_regs++] = adev->gfx.config.sc_hiz_tile_fifo_size;
90775758255SAlex Deucher config[no_regs++] = adev->gfx.config.sc_earlyz_tile_fifo_size;
90875758255SAlex Deucher config[no_regs++] = adev->gfx.config.num_tile_pipes;
90975758255SAlex Deucher config[no_regs++] = adev->gfx.config.backend_enable_mask;
91075758255SAlex Deucher config[no_regs++] = adev->gfx.config.mem_max_burst_length_bytes;
91175758255SAlex Deucher config[no_regs++] = adev->gfx.config.mem_row_size_in_kb;
91275758255SAlex Deucher config[no_regs++] = adev->gfx.config.shader_engine_tile_size;
91375758255SAlex Deucher config[no_regs++] = adev->gfx.config.num_gpus;
91475758255SAlex Deucher config[no_regs++] = adev->gfx.config.multi_gpu_tile_size;
91575758255SAlex Deucher config[no_regs++] = adev->gfx.config.mc_arb_ramcfg;
91675758255SAlex Deucher config[no_regs++] = adev->gfx.config.gb_addr_config;
91775758255SAlex Deucher config[no_regs++] = adev->gfx.config.num_rbs;
91875758255SAlex Deucher
91975758255SAlex Deucher /* rev==1 */
92075758255SAlex Deucher config[no_regs++] = adev->rev_id;
921806e8c55SJesse Zhang config[no_regs++] = adev->pg_flags;
92225faeddcSEvan Quan config[no_regs++] = lower_32_bits(adev->cg_flags);
92375758255SAlex Deucher
92475758255SAlex Deucher /* rev==2 */
92575758255SAlex Deucher config[no_regs++] = adev->family;
92675758255SAlex Deucher config[no_regs++] = adev->external_rev_id;
92775758255SAlex Deucher
92875758255SAlex Deucher /* rev==3 */
92975758255SAlex Deucher config[no_regs++] = adev->pdev->device;
93075758255SAlex Deucher config[no_regs++] = adev->pdev->revision;
93175758255SAlex Deucher config[no_regs++] = adev->pdev->subsystem_device;
93275758255SAlex Deucher config[no_regs++] = adev->pdev->subsystem_vendor;
93375758255SAlex Deucher
9348f74f68dSTom St Denis /* rev==4 APU flag */
9358f74f68dSTom St Denis config[no_regs++] = adev->flags & AMD_IS_APU ? 1 : 0;
9368f74f68dSTom St Denis
937dc2947b3STom St Denis /* rev==5 PG/CG flag upper 32bit */
938806e8c55SJesse Zhang config[no_regs++] = 0;
93925faeddcSEvan Quan config[no_regs++] = upper_32_bits(adev->cg_flags);
94025faeddcSEvan Quan
94175758255SAlex Deucher while (size && (*pos < no_regs * 4)) {
94275758255SAlex Deucher uint32_t value;
94375758255SAlex Deucher
94475758255SAlex Deucher value = config[*pos >> 2];
94575758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
94675758255SAlex Deucher if (r) {
94775758255SAlex Deucher kfree(config);
94875758255SAlex Deucher return r;
94975758255SAlex Deucher }
95075758255SAlex Deucher
95175758255SAlex Deucher result += 4;
95275758255SAlex Deucher buf += 4;
95375758255SAlex Deucher *pos += 4;
95475758255SAlex Deucher size -= 4;
95575758255SAlex Deucher }
95675758255SAlex Deucher
95775758255SAlex Deucher kfree(config);
95875758255SAlex Deucher return result;
95975758255SAlex Deucher }
96075758255SAlex Deucher
9617e4237dbSTom St Denis /**
9627e4237dbSTom St Denis * amdgpu_debugfs_sensor_read - Read from the powerplay sensors
9637e4237dbSTom St Denis *
9647e4237dbSTom St Denis * @f: open file handle
9657e4237dbSTom St Denis * @buf: User buffer to store read data in
9667e4237dbSTom St Denis * @size: Number of bytes to read
9677e4237dbSTom St Denis * @pos: Offset to seek to
9687e4237dbSTom St Denis *
9697e4237dbSTom St Denis * The offset is treated as the BYTE address of one of the sensors
9707e4237dbSTom St Denis * enumerated in amd/include/kgd_pp_interface.h under the
9717e4237dbSTom St Denis * 'amd_pp_sensors' enumeration. For instance to read the UVD VCLK
9727e4237dbSTom St Denis * you would use the offset 3 * 4 = 12.
9737e4237dbSTom St Denis */
amdgpu_debugfs_sensor_read(struct file * f,char __user * buf,size_t size,loff_t * pos)97475758255SAlex Deucher static ssize_t amdgpu_debugfs_sensor_read(struct file *f, char __user *buf,
97575758255SAlex Deucher size_t size, loff_t *pos)
97675758255SAlex Deucher {
97775758255SAlex Deucher struct amdgpu_device *adev = file_inode(f)->i_private;
97875758255SAlex Deucher int idx, x, outsize, r, valuesize;
97975758255SAlex Deucher uint32_t values[16];
98075758255SAlex Deucher
98175758255SAlex Deucher if (size & 3 || *pos & 0x3)
98275758255SAlex Deucher return -EINVAL;
98375758255SAlex Deucher
984b13aa109SRex Zhu if (!adev->pm.dpm_enabled)
98575758255SAlex Deucher return -EINVAL;
98675758255SAlex Deucher
98775758255SAlex Deucher /* convert offset to sensor number */
98875758255SAlex Deucher idx = *pos >> 2;
98975758255SAlex Deucher
99075758255SAlex Deucher valuesize = sizeof(values);
991a9ffe2a9SAlex Deucher
9924a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
9939eee152aSAlex Deucher if (r < 0) {
9944a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
995a9ffe2a9SAlex Deucher return r;
9969eee152aSAlex Deucher }
997a9ffe2a9SAlex Deucher
99895a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
9999eee152aSAlex Deucher if (r < 0) {
10004a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
100195a2f917SYintian Tao return r;
10029eee152aSAlex Deucher }
100395a2f917SYintian Tao
100475758255SAlex Deucher r = amdgpu_dpm_read_sensor(adev, idx, &values[0], &valuesize);
1005a9ffe2a9SAlex Deucher
10064a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
10074a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1008a9ffe2a9SAlex Deucher
100995a2f917SYintian Tao if (r) {
101095a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
10114a5a2de6SKevin Wang return r;
101295a2f917SYintian Tao }
101375758255SAlex Deucher
101495a2f917SYintian Tao if (size > valuesize) {
101595a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
101675758255SAlex Deucher return -EINVAL;
101795a2f917SYintian Tao }
101875758255SAlex Deucher
101975758255SAlex Deucher outsize = 0;
102075758255SAlex Deucher x = 0;
102175758255SAlex Deucher if (!r) {
102275758255SAlex Deucher while (size) {
102375758255SAlex Deucher r = put_user(values[x++], (int32_t *)buf);
102475758255SAlex Deucher buf += 4;
102575758255SAlex Deucher size -= 4;
102675758255SAlex Deucher outsize += 4;
102775758255SAlex Deucher }
102875758255SAlex Deucher }
102975758255SAlex Deucher
103095a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
103175758255SAlex Deucher return !r ? outsize : r;
103275758255SAlex Deucher }
103375758255SAlex Deucher
10347e4237dbSTom St Denis /** amdgpu_debugfs_wave_read - Read WAVE STATUS data
10357e4237dbSTom St Denis *
10367e4237dbSTom St Denis * @f: open file handle
10377e4237dbSTom St Denis * @buf: User buffer to store read data in
10387e4237dbSTom St Denis * @size: Number of bytes to read
10397e4237dbSTom St Denis * @pos: Offset to seek to
10407e4237dbSTom St Denis *
10417e4237dbSTom St Denis * The offset being sought changes which wave that the status data
10427e4237dbSTom St Denis * will be returned for. The bits are used as follows:
10437e4237dbSTom St Denis *
10447e4237dbSTom St Denis * Bits 0..6: Byte offset into data
10457e4237dbSTom St Denis * Bits 7..14: SE selector
10467e4237dbSTom St Denis * Bits 15..22: SH/SA selector
10477e4237dbSTom St Denis * Bits 23..30: CU/{WGP+SIMD} selector
10487e4237dbSTom St Denis * Bits 31..36: WAVE ID selector
10497e4237dbSTom St Denis * Bits 37..44: SIMD ID selector
10507e4237dbSTom St Denis *
10517e4237dbSTom St Denis * The returned data begins with one DWORD of version information
10527e4237dbSTom St Denis * Followed by WAVE STATUS registers relevant to the GFX IP version
10537e4237dbSTom St Denis * being used. See gfx_v8_0_read_wave_data() for an example output.
10547e4237dbSTom St Denis */
amdgpu_debugfs_wave_read(struct file * f,char __user * buf,size_t size,loff_t * pos)105575758255SAlex Deucher static ssize_t amdgpu_debugfs_wave_read(struct file *f, char __user *buf,
105675758255SAlex Deucher size_t size, loff_t *pos)
105775758255SAlex Deucher {
105875758255SAlex Deucher struct amdgpu_device *adev = f->f_inode->i_private;
105975758255SAlex Deucher int r, x;
106075758255SAlex Deucher ssize_t result = 0;
106175758255SAlex Deucher uint32_t offset, se, sh, cu, wave, simd, data[32];
106275758255SAlex Deucher
106375758255SAlex Deucher if (size & 3 || *pos & 3)
106475758255SAlex Deucher return -EINVAL;
106575758255SAlex Deucher
106675758255SAlex Deucher /* decode offset */
106775758255SAlex Deucher offset = (*pos & GENMASK_ULL(6, 0));
106875758255SAlex Deucher se = (*pos & GENMASK_ULL(14, 7)) >> 7;
106975758255SAlex Deucher sh = (*pos & GENMASK_ULL(22, 15)) >> 15;
107075758255SAlex Deucher cu = (*pos & GENMASK_ULL(30, 23)) >> 23;
107175758255SAlex Deucher wave = (*pos & GENMASK_ULL(36, 31)) >> 31;
107275758255SAlex Deucher simd = (*pos & GENMASK_ULL(44, 37)) >> 37;
107375758255SAlex Deucher
10744a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
10759eee152aSAlex Deucher if (r < 0) {
10764a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1077a9ffe2a9SAlex Deucher return r;
10789eee152aSAlex Deucher }
1079a9ffe2a9SAlex Deucher
108095a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
10819eee152aSAlex Deucher if (r < 0) {
10824a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
108395a2f917SYintian Tao return r;
10849eee152aSAlex Deucher }
108595a2f917SYintian Tao
108675758255SAlex Deucher /* switch to the specific se/sh/cu */
108775758255SAlex Deucher mutex_lock(&adev->grbm_idx_mutex);
1088d51ac6d0SLe Ma amdgpu_gfx_select_se_sh(adev, se, sh, cu, 0);
108975758255SAlex Deucher
109075758255SAlex Deucher x = 0;
109175758255SAlex Deucher if (adev->gfx.funcs->read_wave_data)
1092553f973aSTom St Denis adev->gfx.funcs->read_wave_data(adev, 0, simd, wave, data, &x);
109375758255SAlex Deucher
1094d51ac6d0SLe Ma amdgpu_gfx_select_se_sh(adev, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0);
109575758255SAlex Deucher mutex_unlock(&adev->grbm_idx_mutex);
109675758255SAlex Deucher
10974a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
10984a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1099a9ffe2a9SAlex Deucher
110095a2f917SYintian Tao if (!x) {
110195a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
110275758255SAlex Deucher return -EINVAL;
110395a2f917SYintian Tao }
110475758255SAlex Deucher
110575758255SAlex Deucher while (size && (offset < x * 4)) {
110675758255SAlex Deucher uint32_t value;
110775758255SAlex Deucher
110875758255SAlex Deucher value = data[offset >> 2];
110975758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
111095a2f917SYintian Tao if (r) {
111195a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
111275758255SAlex Deucher return r;
111395a2f917SYintian Tao }
111475758255SAlex Deucher
111575758255SAlex Deucher result += 4;
111675758255SAlex Deucher buf += 4;
111775758255SAlex Deucher offset += 4;
111875758255SAlex Deucher size -= 4;
111975758255SAlex Deucher }
112075758255SAlex Deucher
112195a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
112275758255SAlex Deucher return result;
112375758255SAlex Deucher }
112475758255SAlex Deucher
11257e4237dbSTom St Denis /** amdgpu_debugfs_gpr_read - Read wave gprs
11267e4237dbSTom St Denis *
11277e4237dbSTom St Denis * @f: open file handle
11287e4237dbSTom St Denis * @buf: User buffer to store read data in
11297e4237dbSTom St Denis * @size: Number of bytes to read
11307e4237dbSTom St Denis * @pos: Offset to seek to
11317e4237dbSTom St Denis *
11327e4237dbSTom St Denis * The offset being sought changes which wave that the status data
11337e4237dbSTom St Denis * will be returned for. The bits are used as follows:
11347e4237dbSTom St Denis *
11357e4237dbSTom St Denis * Bits 0..11: Byte offset into data
11367e4237dbSTom St Denis * Bits 12..19: SE selector
11377e4237dbSTom St Denis * Bits 20..27: SH/SA selector
11387e4237dbSTom St Denis * Bits 28..35: CU/{WGP+SIMD} selector
11397e4237dbSTom St Denis * Bits 36..43: WAVE ID selector
11407e4237dbSTom St Denis * Bits 37..44: SIMD ID selector
11417e4237dbSTom St Denis * Bits 52..59: Thread selector
11427e4237dbSTom St Denis * Bits 60..61: Bank selector (VGPR=0,SGPR=1)
11437e4237dbSTom St Denis *
11447e4237dbSTom St Denis * The return data comes from the SGPR or VGPR register bank for
11457e4237dbSTom St Denis * the selected operational unit.
11467e4237dbSTom St Denis */
amdgpu_debugfs_gpr_read(struct file * f,char __user * buf,size_t size,loff_t * pos)114775758255SAlex Deucher static ssize_t amdgpu_debugfs_gpr_read(struct file *f, char __user *buf,
114875758255SAlex Deucher size_t size, loff_t *pos)
114975758255SAlex Deucher {
115075758255SAlex Deucher struct amdgpu_device *adev = f->f_inode->i_private;
115175758255SAlex Deucher int r;
115275758255SAlex Deucher ssize_t result = 0;
115375758255SAlex Deucher uint32_t offset, se, sh, cu, wave, simd, thread, bank, *data;
115475758255SAlex Deucher
11556397ec58STom St Denis if (size > 4096 || size & 3 || *pos & 3)
115675758255SAlex Deucher return -EINVAL;
115775758255SAlex Deucher
115875758255SAlex Deucher /* decode offset */
11596397ec58STom St Denis offset = (*pos & GENMASK_ULL(11, 0)) >> 2;
116075758255SAlex Deucher se = (*pos & GENMASK_ULL(19, 12)) >> 12;
116175758255SAlex Deucher sh = (*pos & GENMASK_ULL(27, 20)) >> 20;
116275758255SAlex Deucher cu = (*pos & GENMASK_ULL(35, 28)) >> 28;
116375758255SAlex Deucher wave = (*pos & GENMASK_ULL(43, 36)) >> 36;
116475758255SAlex Deucher simd = (*pos & GENMASK_ULL(51, 44)) >> 44;
116575758255SAlex Deucher thread = (*pos & GENMASK_ULL(59, 52)) >> 52;
116675758255SAlex Deucher bank = (*pos & GENMASK_ULL(61, 60)) >> 60;
116775758255SAlex Deucher
1168929e571cSWang Xiayang data = kcalloc(1024, sizeof(*data), GFP_KERNEL);
116975758255SAlex Deucher if (!data)
117075758255SAlex Deucher return -ENOMEM;
117175758255SAlex Deucher
11724a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
1173a9ffe2a9SAlex Deucher if (r < 0)
11743e4aeff3SChen Tao goto err;
1175a9ffe2a9SAlex Deucher
117695a2f917SYintian Tao r = amdgpu_virt_enable_access_debugfs(adev);
117795a2f917SYintian Tao if (r < 0)
1178888e32d7SChen Tao goto err;
117995a2f917SYintian Tao
118075758255SAlex Deucher /* switch to the specific se/sh/cu */
118175758255SAlex Deucher mutex_lock(&adev->grbm_idx_mutex);
1182d51ac6d0SLe Ma amdgpu_gfx_select_se_sh(adev, se, sh, cu, 0);
118375758255SAlex Deucher
118475758255SAlex Deucher if (bank == 0) {
118575758255SAlex Deucher if (adev->gfx.funcs->read_wave_vgprs)
1186553f973aSTom St Denis adev->gfx.funcs->read_wave_vgprs(adev, 0, simd, wave, thread, offset, size>>2, data);
118775758255SAlex Deucher } else {
118875758255SAlex Deucher if (adev->gfx.funcs->read_wave_sgprs)
1189553f973aSTom St Denis adev->gfx.funcs->read_wave_sgprs(adev, 0, simd, wave, offset, size>>2, data);
119075758255SAlex Deucher }
119175758255SAlex Deucher
1192d51ac6d0SLe Ma amdgpu_gfx_select_se_sh(adev, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0);
119375758255SAlex Deucher mutex_unlock(&adev->grbm_idx_mutex);
119475758255SAlex Deucher
11954a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
11964a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1197a9ffe2a9SAlex Deucher
119875758255SAlex Deucher while (size) {
119975758255SAlex Deucher uint32_t value;
120075758255SAlex Deucher
12016397ec58STom St Denis value = data[result >> 2];
120275758255SAlex Deucher r = put_user(value, (uint32_t *)buf);
120375758255SAlex Deucher if (r) {
12043e4aeff3SChen Tao amdgpu_virt_disable_access_debugfs(adev);
120575758255SAlex Deucher goto err;
120675758255SAlex Deucher }
120775758255SAlex Deucher
120875758255SAlex Deucher result += 4;
120975758255SAlex Deucher buf += 4;
121075758255SAlex Deucher size -= 4;
121175758255SAlex Deucher }
121275758255SAlex Deucher
121375758255SAlex Deucher kfree(data);
121495a2f917SYintian Tao amdgpu_virt_disable_access_debugfs(adev);
121575758255SAlex Deucher return result;
12163e4aeff3SChen Tao
12173e4aeff3SChen Tao err:
12184a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
12193e4aeff3SChen Tao kfree(data);
12203e4aeff3SChen Tao return r;
122175758255SAlex Deucher }
122275758255SAlex Deucher
1223669e2f91STom St Denis /**
12240ad7347aSAndré Almeida * amdgpu_debugfs_gfxoff_residency_read - Read GFXOFF residency
12250ad7347aSAndré Almeida *
12260ad7347aSAndré Almeida * @f: open file handle
12270ad7347aSAndré Almeida * @buf: User buffer to store read data in
12280ad7347aSAndré Almeida * @size: Number of bytes to read
12290ad7347aSAndré Almeida * @pos: Offset to seek to
12300ad7347aSAndré Almeida *
12310ad7347aSAndré Almeida * Read the last residency value logged. It doesn't auto update, one needs to
12320ad7347aSAndré Almeida * stop logging before getting the current value.
12330ad7347aSAndré Almeida */
amdgpu_debugfs_gfxoff_residency_read(struct file * f,char __user * buf,size_t size,loff_t * pos)12340ad7347aSAndré Almeida static ssize_t amdgpu_debugfs_gfxoff_residency_read(struct file *f, char __user *buf,
12350ad7347aSAndré Almeida size_t size, loff_t *pos)
12360ad7347aSAndré Almeida {
12370ad7347aSAndré Almeida struct amdgpu_device *adev = file_inode(f)->i_private;
12380ad7347aSAndré Almeida ssize_t result = 0;
12390ad7347aSAndré Almeida int r;
12400ad7347aSAndré Almeida
12410ad7347aSAndré Almeida if (size & 0x3 || *pos & 0x3)
12420ad7347aSAndré Almeida return -EINVAL;
12430ad7347aSAndré Almeida
12440ad7347aSAndré Almeida r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
12450ad7347aSAndré Almeida if (r < 0) {
12460ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
12470ad7347aSAndré Almeida return r;
12480ad7347aSAndré Almeida }
12490ad7347aSAndré Almeida
12500ad7347aSAndré Almeida while (size) {
12510ad7347aSAndré Almeida uint32_t value;
12520ad7347aSAndré Almeida
12530ad7347aSAndré Almeida r = amdgpu_get_gfx_off_residency(adev, &value);
12540ad7347aSAndré Almeida if (r)
12550ad7347aSAndré Almeida goto out;
12560ad7347aSAndré Almeida
12570ad7347aSAndré Almeida r = put_user(value, (uint32_t *)buf);
12580ad7347aSAndré Almeida if (r)
12590ad7347aSAndré Almeida goto out;
12600ad7347aSAndré Almeida
12610ad7347aSAndré Almeida result += 4;
12620ad7347aSAndré Almeida buf += 4;
12630ad7347aSAndré Almeida *pos += 4;
12640ad7347aSAndré Almeida size -= 4;
12650ad7347aSAndré Almeida }
12660ad7347aSAndré Almeida
12670ad7347aSAndré Almeida r = result;
12680ad7347aSAndré Almeida out:
12690ad7347aSAndré Almeida pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
12700ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
12710ad7347aSAndré Almeida
12720ad7347aSAndré Almeida return r;
12730ad7347aSAndré Almeida }
12740ad7347aSAndré Almeida
12750ad7347aSAndré Almeida /**
12760ad7347aSAndré Almeida * amdgpu_debugfs_gfxoff_residency_write - Log GFXOFF Residency
12770ad7347aSAndré Almeida *
12780ad7347aSAndré Almeida * @f: open file handle
12790ad7347aSAndré Almeida * @buf: User buffer to write data from
12800ad7347aSAndré Almeida * @size: Number of bytes to write
12810ad7347aSAndré Almeida * @pos: Offset to seek to
12820ad7347aSAndré Almeida *
12830ad7347aSAndré Almeida * Write a 32-bit non-zero to start logging; write a 32-bit zero to stop
12840ad7347aSAndré Almeida */
amdgpu_debugfs_gfxoff_residency_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)12850ad7347aSAndré Almeida static ssize_t amdgpu_debugfs_gfxoff_residency_write(struct file *f, const char __user *buf,
12860ad7347aSAndré Almeida size_t size, loff_t *pos)
12870ad7347aSAndré Almeida {
12880ad7347aSAndré Almeida struct amdgpu_device *adev = file_inode(f)->i_private;
12890ad7347aSAndré Almeida ssize_t result = 0;
12900ad7347aSAndré Almeida int r;
12910ad7347aSAndré Almeida
12920ad7347aSAndré Almeida if (size & 0x3 || *pos & 0x3)
12930ad7347aSAndré Almeida return -EINVAL;
12940ad7347aSAndré Almeida
12950ad7347aSAndré Almeida r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
12960ad7347aSAndré Almeida if (r < 0) {
12970ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
12980ad7347aSAndré Almeida return r;
12990ad7347aSAndré Almeida }
13000ad7347aSAndré Almeida
13010ad7347aSAndré Almeida while (size) {
13020ad7347aSAndré Almeida u32 value;
13030ad7347aSAndré Almeida
13040ad7347aSAndré Almeida r = get_user(value, (uint32_t *)buf);
13050ad7347aSAndré Almeida if (r)
13060ad7347aSAndré Almeida goto out;
13070ad7347aSAndré Almeida
13080ad7347aSAndré Almeida amdgpu_set_gfx_off_residency(adev, value ? true : false);
13090ad7347aSAndré Almeida
13100ad7347aSAndré Almeida result += 4;
13110ad7347aSAndré Almeida buf += 4;
13120ad7347aSAndré Almeida *pos += 4;
13130ad7347aSAndré Almeida size -= 4;
13140ad7347aSAndré Almeida }
13150ad7347aSAndré Almeida
13160ad7347aSAndré Almeida r = result;
13170ad7347aSAndré Almeida out:
13180ad7347aSAndré Almeida pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
13190ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
13200ad7347aSAndré Almeida
13210ad7347aSAndré Almeida return r;
13220ad7347aSAndré Almeida }
13230ad7347aSAndré Almeida
13240ad7347aSAndré Almeida
13250ad7347aSAndré Almeida /**
13260ad7347aSAndré Almeida * amdgpu_debugfs_gfxoff_count_read - Read GFXOFF entry count
13270ad7347aSAndré Almeida *
13280ad7347aSAndré Almeida * @f: open file handle
13290ad7347aSAndré Almeida * @buf: User buffer to store read data in
13300ad7347aSAndré Almeida * @size: Number of bytes to read
13310ad7347aSAndré Almeida * @pos: Offset to seek to
13320ad7347aSAndré Almeida */
amdgpu_debugfs_gfxoff_count_read(struct file * f,char __user * buf,size_t size,loff_t * pos)13330ad7347aSAndré Almeida static ssize_t amdgpu_debugfs_gfxoff_count_read(struct file *f, char __user *buf,
13340ad7347aSAndré Almeida size_t size, loff_t *pos)
13350ad7347aSAndré Almeida {
13360ad7347aSAndré Almeida struct amdgpu_device *adev = file_inode(f)->i_private;
13370ad7347aSAndré Almeida ssize_t result = 0;
13380ad7347aSAndré Almeida int r;
13390ad7347aSAndré Almeida
13400ad7347aSAndré Almeida if (size & 0x3 || *pos & 0x3)
13410ad7347aSAndré Almeida return -EINVAL;
13420ad7347aSAndré Almeida
13430ad7347aSAndré Almeida r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
13440ad7347aSAndré Almeida if (r < 0) {
13450ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
13460ad7347aSAndré Almeida return r;
13470ad7347aSAndré Almeida }
13480ad7347aSAndré Almeida
13490ad7347aSAndré Almeida while (size) {
13500ad7347aSAndré Almeida u64 value = 0;
13510ad7347aSAndré Almeida
13520ad7347aSAndré Almeida r = amdgpu_get_gfx_off_entrycount(adev, &value);
13530ad7347aSAndré Almeida if (r)
13540ad7347aSAndré Almeida goto out;
13550ad7347aSAndré Almeida
13560ad7347aSAndré Almeida r = put_user(value, (u64 *)buf);
13570ad7347aSAndré Almeida if (r)
13580ad7347aSAndré Almeida goto out;
13590ad7347aSAndré Almeida
13600ad7347aSAndré Almeida result += 4;
13610ad7347aSAndré Almeida buf += 4;
13620ad7347aSAndré Almeida *pos += 4;
13630ad7347aSAndré Almeida size -= 4;
13640ad7347aSAndré Almeida }
13650ad7347aSAndré Almeida
13660ad7347aSAndré Almeida r = result;
13670ad7347aSAndré Almeida out:
13680ad7347aSAndré Almeida pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
13690ad7347aSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
13700ad7347aSAndré Almeida
13710ad7347aSAndré Almeida return r;
13720ad7347aSAndré Almeida }
13730ad7347aSAndré Almeida
13740ad7347aSAndré Almeida /**
1375e72d4a8bSLee Jones * amdgpu_debugfs_gfxoff_write - Enable/disable GFXOFF
1376669e2f91STom St Denis *
1377669e2f91STom St Denis * @f: open file handle
1378669e2f91STom St Denis * @buf: User buffer to write data from
1379669e2f91STom St Denis * @size: Number of bytes to write
1380669e2f91STom St Denis * @pos: Offset to seek to
1381669e2f91STom St Denis *
1382669e2f91STom St Denis * Write a 32-bit zero to disable or a 32-bit non-zero to enable
1383669e2f91STom St Denis */
amdgpu_debugfs_gfxoff_write(struct file * f,const char __user * buf,size_t size,loff_t * pos)1384669e2f91STom St Denis static ssize_t amdgpu_debugfs_gfxoff_write(struct file *f, const char __user *buf,
1385669e2f91STom St Denis size_t size, loff_t *pos)
1386669e2f91STom St Denis {
1387669e2f91STom St Denis struct amdgpu_device *adev = file_inode(f)->i_private;
1388669e2f91STom St Denis ssize_t result = 0;
1389669e2f91STom St Denis int r;
1390669e2f91STom St Denis
1391669e2f91STom St Denis if (size & 0x3 || *pos & 0x3)
1392669e2f91STom St Denis return -EINVAL;
1393669e2f91STom St Denis
13944a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
13959eee152aSAlex Deucher if (r < 0) {
13964a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1397669e2f91STom St Denis return r;
13989eee152aSAlex Deucher }
1399669e2f91STom St Denis
1400669e2f91STom St Denis while (size) {
1401669e2f91STom St Denis uint32_t value;
1402669e2f91STom St Denis
1403669e2f91STom St Denis r = get_user(value, (uint32_t *)buf);
1404edadd6fcSAndré Almeida if (r)
1405edadd6fcSAndré Almeida goto out;
1406669e2f91STom St Denis
1407669e2f91STom St Denis amdgpu_gfx_off_ctrl(adev, value ? true : false);
1408669e2f91STom St Denis
1409669e2f91STom St Denis result += 4;
1410669e2f91STom St Denis buf += 4;
1411669e2f91STom St Denis *pos += 4;
1412669e2f91STom St Denis size -= 4;
1413669e2f91STom St Denis }
1414669e2f91STom St Denis
1415edadd6fcSAndré Almeida r = result;
1416edadd6fcSAndré Almeida out:
14174a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
14184a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1419669e2f91STom St Denis
1420edadd6fcSAndré Almeida return r;
1421669e2f91STom St Denis }
1422669e2f91STom St Denis
1423669e2f91STom St Denis
1424443c7f3cSJinzhou.Su /**
1425e72d4a8bSLee Jones * amdgpu_debugfs_gfxoff_read - read gfxoff status
1426443c7f3cSJinzhou.Su *
1427443c7f3cSJinzhou.Su * @f: open file handle
1428443c7f3cSJinzhou.Su * @buf: User buffer to store read data in
1429443c7f3cSJinzhou.Su * @size: Number of bytes to read
1430443c7f3cSJinzhou.Su * @pos: Offset to seek to
1431443c7f3cSJinzhou.Su */
amdgpu_debugfs_gfxoff_read(struct file * f,char __user * buf,size_t size,loff_t * pos)1432443c7f3cSJinzhou.Su static ssize_t amdgpu_debugfs_gfxoff_read(struct file *f, char __user *buf,
1433443c7f3cSJinzhou.Su size_t size, loff_t *pos)
1434443c7f3cSJinzhou.Su {
1435443c7f3cSJinzhou.Su struct amdgpu_device *adev = file_inode(f)->i_private;
1436443c7f3cSJinzhou.Su ssize_t result = 0;
1437443c7f3cSJinzhou.Su int r;
1438443c7f3cSJinzhou.Su
1439443c7f3cSJinzhou.Su if (size & 0x3 || *pos & 0x3)
1440443c7f3cSJinzhou.Su return -EINVAL;
1441443c7f3cSJinzhou.Su
14424a580877SLuben Tuikov r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
14434bd8dd0dSYongzhi Liu if (r < 0) {
14444bd8dd0dSYongzhi Liu pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1445443c7f3cSJinzhou.Su return r;
14464bd8dd0dSYongzhi Liu }
1447443c7f3cSJinzhou.Su
1448443c7f3cSJinzhou.Su while (size) {
14494686177fSAndré Almeida u32 value = adev->gfx.gfx_off_state;
14504686177fSAndré Almeida
14514686177fSAndré Almeida r = put_user(value, (u32 *)buf);
14524686177fSAndré Almeida if (r)
14534686177fSAndré Almeida goto out;
14544686177fSAndré Almeida
14554686177fSAndré Almeida result += 4;
14564686177fSAndré Almeida buf += 4;
14574686177fSAndré Almeida *pos += 4;
14584686177fSAndré Almeida size -= 4;
14594686177fSAndré Almeida }
14604686177fSAndré Almeida
14614686177fSAndré Almeida r = result;
14624686177fSAndré Almeida out:
14634686177fSAndré Almeida pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
14644686177fSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
14654686177fSAndré Almeida
14664686177fSAndré Almeida return r;
14674686177fSAndré Almeida }
14684686177fSAndré Almeida
amdgpu_debugfs_gfxoff_status_read(struct file * f,char __user * buf,size_t size,loff_t * pos)14694686177fSAndré Almeida static ssize_t amdgpu_debugfs_gfxoff_status_read(struct file *f, char __user *buf,
14704686177fSAndré Almeida size_t size, loff_t *pos)
14714686177fSAndré Almeida {
14724686177fSAndré Almeida struct amdgpu_device *adev = file_inode(f)->i_private;
14734686177fSAndré Almeida ssize_t result = 0;
14744686177fSAndré Almeida int r;
14754686177fSAndré Almeida
14764686177fSAndré Almeida if (size & 0x3 || *pos & 0x3)
14774686177fSAndré Almeida return -EINVAL;
14784686177fSAndré Almeida
14794686177fSAndré Almeida r = pm_runtime_get_sync(adev_to_drm(adev)->dev);
14804686177fSAndré Almeida if (r < 0) {
14814686177fSAndré Almeida pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
14824686177fSAndré Almeida return r;
14834686177fSAndré Almeida }
14844686177fSAndré Almeida
14854686177fSAndré Almeida while (size) {
14864686177fSAndré Almeida u32 value;
1487443c7f3cSJinzhou.Su
1488443c7f3cSJinzhou.Su r = amdgpu_get_gfx_off_status(adev, &value);
1489edadd6fcSAndré Almeida if (r)
1490edadd6fcSAndré Almeida goto out;
1491443c7f3cSJinzhou.Su
14924686177fSAndré Almeida r = put_user(value, (u32 *)buf);
1493edadd6fcSAndré Almeida if (r)
1494edadd6fcSAndré Almeida goto out;
1495443c7f3cSJinzhou.Su
1496443c7f3cSJinzhou.Su result += 4;
1497443c7f3cSJinzhou.Su buf += 4;
1498443c7f3cSJinzhou.Su *pos += 4;
1499443c7f3cSJinzhou.Su size -= 4;
1500443c7f3cSJinzhou.Su }
1501443c7f3cSJinzhou.Su
1502edadd6fcSAndré Almeida r = result;
1503edadd6fcSAndré Almeida out:
15044a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
15054a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
1506443c7f3cSJinzhou.Su
1507edadd6fcSAndré Almeida return r;
1508443c7f3cSJinzhou.Su }
1509443c7f3cSJinzhou.Su
151037df9560STom St Denis static const struct file_operations amdgpu_debugfs_regs2_fops = {
151137df9560STom St Denis .owner = THIS_MODULE,
151237df9560STom St Denis .unlocked_ioctl = amdgpu_debugfs_regs2_ioctl,
151337df9560STom St Denis .read = amdgpu_debugfs_regs2_read,
151437df9560STom St Denis .write = amdgpu_debugfs_regs2_write,
151537df9560STom St Denis .open = amdgpu_debugfs_regs2_open,
151637df9560STom St Denis .release = amdgpu_debugfs_regs2_release,
151737df9560STom St Denis .llseek = default_llseek
151837df9560STom St Denis };
151937df9560STom St Denis
1520553f973aSTom St Denis static const struct file_operations amdgpu_debugfs_gprwave_fops = {
1521553f973aSTom St Denis .owner = THIS_MODULE,
1522553f973aSTom St Denis .unlocked_ioctl = amdgpu_debugfs_gprwave_ioctl,
1523553f973aSTom St Denis .read = amdgpu_debugfs_gprwave_read,
1524553f973aSTom St Denis .open = amdgpu_debugfs_gprwave_open,
1525553f973aSTom St Denis .release = amdgpu_debugfs_gprwave_release,
1526553f973aSTom St Denis .llseek = default_llseek
1527553f973aSTom St Denis };
1528553f973aSTom St Denis
152975758255SAlex Deucher static const struct file_operations amdgpu_debugfs_regs_fops = {
153075758255SAlex Deucher .owner = THIS_MODULE,
153175758255SAlex Deucher .read = amdgpu_debugfs_regs_read,
153275758255SAlex Deucher .write = amdgpu_debugfs_regs_write,
153375758255SAlex Deucher .llseek = default_llseek
153475758255SAlex Deucher };
153575758255SAlex Deucher static const struct file_operations amdgpu_debugfs_regs_didt_fops = {
153675758255SAlex Deucher .owner = THIS_MODULE,
153775758255SAlex Deucher .read = amdgpu_debugfs_regs_didt_read,
153875758255SAlex Deucher .write = amdgpu_debugfs_regs_didt_write,
153975758255SAlex Deucher .llseek = default_llseek
154075758255SAlex Deucher };
154175758255SAlex Deucher static const struct file_operations amdgpu_debugfs_regs_pcie_fops = {
154275758255SAlex Deucher .owner = THIS_MODULE,
154375758255SAlex Deucher .read = amdgpu_debugfs_regs_pcie_read,
154475758255SAlex Deucher .write = amdgpu_debugfs_regs_pcie_write,
154575758255SAlex Deucher .llseek = default_llseek
154675758255SAlex Deucher };
154775758255SAlex Deucher static const struct file_operations amdgpu_debugfs_regs_smc_fops = {
154875758255SAlex Deucher .owner = THIS_MODULE,
154975758255SAlex Deucher .read = amdgpu_debugfs_regs_smc_read,
155075758255SAlex Deucher .write = amdgpu_debugfs_regs_smc_write,
155175758255SAlex Deucher .llseek = default_llseek
155275758255SAlex Deucher };
155375758255SAlex Deucher
155475758255SAlex Deucher static const struct file_operations amdgpu_debugfs_gca_config_fops = {
155575758255SAlex Deucher .owner = THIS_MODULE,
155675758255SAlex Deucher .read = amdgpu_debugfs_gca_config_read,
155775758255SAlex Deucher .llseek = default_llseek
155875758255SAlex Deucher };
155975758255SAlex Deucher
156075758255SAlex Deucher static const struct file_operations amdgpu_debugfs_sensors_fops = {
156175758255SAlex Deucher .owner = THIS_MODULE,
156275758255SAlex Deucher .read = amdgpu_debugfs_sensor_read,
156375758255SAlex Deucher .llseek = default_llseek
156475758255SAlex Deucher };
156575758255SAlex Deucher
156675758255SAlex Deucher static const struct file_operations amdgpu_debugfs_wave_fops = {
156775758255SAlex Deucher .owner = THIS_MODULE,
156875758255SAlex Deucher .read = amdgpu_debugfs_wave_read,
156975758255SAlex Deucher .llseek = default_llseek
157075758255SAlex Deucher };
157175758255SAlex Deucher static const struct file_operations amdgpu_debugfs_gpr_fops = {
157275758255SAlex Deucher .owner = THIS_MODULE,
157375758255SAlex Deucher .read = amdgpu_debugfs_gpr_read,
157475758255SAlex Deucher .llseek = default_llseek
157575758255SAlex Deucher };
157675758255SAlex Deucher
1577669e2f91STom St Denis static const struct file_operations amdgpu_debugfs_gfxoff_fops = {
1578669e2f91STom St Denis .owner = THIS_MODULE,
1579443c7f3cSJinzhou.Su .read = amdgpu_debugfs_gfxoff_read,
1580669e2f91STom St Denis .write = amdgpu_debugfs_gfxoff_write,
1581443c7f3cSJinzhou.Su .llseek = default_llseek
1582669e2f91STom St Denis };
1583669e2f91STom St Denis
15844686177fSAndré Almeida static const struct file_operations amdgpu_debugfs_gfxoff_status_fops = {
15854686177fSAndré Almeida .owner = THIS_MODULE,
15864686177fSAndré Almeida .read = amdgpu_debugfs_gfxoff_status_read,
15874686177fSAndré Almeida .llseek = default_llseek
15884686177fSAndré Almeida };
15894686177fSAndré Almeida
15900ad7347aSAndré Almeida static const struct file_operations amdgpu_debugfs_gfxoff_count_fops = {
15910ad7347aSAndré Almeida .owner = THIS_MODULE,
15920ad7347aSAndré Almeida .read = amdgpu_debugfs_gfxoff_count_read,
15930ad7347aSAndré Almeida .llseek = default_llseek
15940ad7347aSAndré Almeida };
15950ad7347aSAndré Almeida
15960ad7347aSAndré Almeida static const struct file_operations amdgpu_debugfs_gfxoff_residency_fops = {
15970ad7347aSAndré Almeida .owner = THIS_MODULE,
15980ad7347aSAndré Almeida .read = amdgpu_debugfs_gfxoff_residency_read,
15990ad7347aSAndré Almeida .write = amdgpu_debugfs_gfxoff_residency_write,
16000ad7347aSAndré Almeida .llseek = default_llseek
16010ad7347aSAndré Almeida };
16020ad7347aSAndré Almeida
160375758255SAlex Deucher static const struct file_operations *debugfs_regs[] = {
160475758255SAlex Deucher &amdgpu_debugfs_regs_fops,
160537df9560STom St Denis &amdgpu_debugfs_regs2_fops,
1606553f973aSTom St Denis &amdgpu_debugfs_gprwave_fops,
160775758255SAlex Deucher &amdgpu_debugfs_regs_didt_fops,
160875758255SAlex Deucher &amdgpu_debugfs_regs_pcie_fops,
160975758255SAlex Deucher &amdgpu_debugfs_regs_smc_fops,
161075758255SAlex Deucher &amdgpu_debugfs_gca_config_fops,
161175758255SAlex Deucher &amdgpu_debugfs_sensors_fops,
161275758255SAlex Deucher &amdgpu_debugfs_wave_fops,
161375758255SAlex Deucher &amdgpu_debugfs_gpr_fops,
1614669e2f91STom St Denis &amdgpu_debugfs_gfxoff_fops,
16154686177fSAndré Almeida &amdgpu_debugfs_gfxoff_status_fops,
16160ad7347aSAndré Almeida &amdgpu_debugfs_gfxoff_count_fops,
16170ad7347aSAndré Almeida &amdgpu_debugfs_gfxoff_residency_fops,
161875758255SAlex Deucher };
161975758255SAlex Deucher
16208fa76350SSrinivasan Shanmugam static const char * const debugfs_regs_names[] = {
162175758255SAlex Deucher "amdgpu_regs",
162237df9560STom St Denis "amdgpu_regs2",
1623553f973aSTom St Denis "amdgpu_gprwave",
162475758255SAlex Deucher "amdgpu_regs_didt",
162575758255SAlex Deucher "amdgpu_regs_pcie",
162675758255SAlex Deucher "amdgpu_regs_smc",
162775758255SAlex Deucher "amdgpu_gca_config",
162875758255SAlex Deucher "amdgpu_sensors",
162975758255SAlex Deucher "amdgpu_wave",
163075758255SAlex Deucher "amdgpu_gpr",
1631669e2f91STom St Denis "amdgpu_gfxoff",
16324686177fSAndré Almeida "amdgpu_gfxoff_status",
16330ad7347aSAndré Almeida "amdgpu_gfxoff_count",
16340ad7347aSAndré Almeida "amdgpu_gfxoff_residency",
163575758255SAlex Deucher };
163675758255SAlex Deucher
16377e4237dbSTom St Denis /**
16387e4237dbSTom St Denis * amdgpu_debugfs_regs_init - Initialize debugfs entries that provide
16397e4237dbSTom St Denis * register access.
16407e4237dbSTom St Denis *
16417e4237dbSTom St Denis * @adev: The device to attach the debugfs entries to
16427e4237dbSTom St Denis */
amdgpu_debugfs_regs_init(struct amdgpu_device * adev)164375758255SAlex Deucher int amdgpu_debugfs_regs_init(struct amdgpu_device *adev)
164475758255SAlex Deucher {
16454a580877SLuben Tuikov struct drm_minor *minor = adev_to_drm(adev)->primary;
164675758255SAlex Deucher struct dentry *ent, *root = minor->debugfs_root;
1647d344b21bSDan Carpenter unsigned int i;
164875758255SAlex Deucher
164975758255SAlex Deucher for (i = 0; i < ARRAY_SIZE(debugfs_regs); i++) {
165075758255SAlex Deucher ent = debugfs_create_file(debugfs_regs_names[i],
1651c0cfd2e6SAlex Deucher S_IFREG | 0400, root,
165275758255SAlex Deucher adev, debugfs_regs[i]);
1653d344b21bSDan Carpenter if (!i && !IS_ERR_OR_NULL(ent))
165475758255SAlex Deucher i_size_write(ent->d_inode, adev->rmmio_size);
165575758255SAlex Deucher }
165675758255SAlex Deucher
165775758255SAlex Deucher return 0;
165875758255SAlex Deucher }
165975758255SAlex Deucher
amdgpu_debugfs_test_ib_show(struct seq_file * m,void * unused)166098d28ac2SNirmoy Das static int amdgpu_debugfs_test_ib_show(struct seq_file *m, void *unused)
166175758255SAlex Deucher {
1662109b4d8cSSu Hui struct amdgpu_device *adev = m->private;
166398d28ac2SNirmoy Das struct drm_device *dev = adev_to_drm(adev);
166475758255SAlex Deucher int r = 0, i;
166575758255SAlex Deucher
1666a9ffe2a9SAlex Deucher r = pm_runtime_get_sync(dev->dev);
16679eee152aSAlex Deucher if (r < 0) {
166898d28ac2SNirmoy Das pm_runtime_put_autosuspend(dev->dev);
1669a9ffe2a9SAlex Deucher return r;
16709eee152aSAlex Deucher }
1671a9ffe2a9SAlex Deucher
1672a28fda31SAndrey Grodzovsky /* Avoid accidently unparking the sched thread during GPU reset */
1673d0fb18b5SAndrey Grodzovsky r = down_write_killable(&adev->reset_domain->sem);
16746049db43SDennis Li if (r)
16756049db43SDennis Li return r;
1676a28fda31SAndrey Grodzovsky
167775758255SAlex Deucher /* hold on the scheduler */
167875758255SAlex Deucher for (i = 0; i < AMDGPU_MAX_RINGS; i++) {
167975758255SAlex Deucher struct amdgpu_ring *ring = adev->rings[i];
168075758255SAlex Deucher
16819749c868SMa Jun if (!amdgpu_ring_sched_ready(ring))
168275758255SAlex Deucher continue;
168335963cf2SMatthew Brost drm_sched_wqueue_stop(&ring->sched);
168475758255SAlex Deucher }
168575758255SAlex Deucher
16868fa76350SSrinivasan Shanmugam seq_puts(m, "run ib test:\n");
168775758255SAlex Deucher r = amdgpu_ib_ring_tests(adev);
168875758255SAlex Deucher if (r)
168975758255SAlex Deucher seq_printf(m, "ib ring tests failed (%d).\n", r);
169075758255SAlex Deucher else
16918fa76350SSrinivasan Shanmugam seq_puts(m, "ib ring tests passed.\n");
169275758255SAlex Deucher
169375758255SAlex Deucher /* go on the scheduler */
169475758255SAlex Deucher for (i = 0; i < AMDGPU_MAX_RINGS; i++) {
169575758255SAlex Deucher struct amdgpu_ring *ring = adev->rings[i];
169675758255SAlex Deucher
16979749c868SMa Jun if (!amdgpu_ring_sched_ready(ring))
169875758255SAlex Deucher continue;
169935963cf2SMatthew Brost drm_sched_wqueue_start(&ring->sched);
170075758255SAlex Deucher }
170175758255SAlex Deucher
1702d0fb18b5SAndrey Grodzovsky up_write(&adev->reset_domain->sem);
1703a28fda31SAndrey Grodzovsky
1704a9ffe2a9SAlex Deucher pm_runtime_mark_last_busy(dev->dev);
1705a9ffe2a9SAlex Deucher pm_runtime_put_autosuspend(dev->dev);
1706a9ffe2a9SAlex Deucher
170775758255SAlex Deucher return 0;
170875758255SAlex Deucher }
170975758255SAlex Deucher
amdgpu_debugfs_evict_vram(void * data,u64 * val)171098d28ac2SNirmoy Das static int amdgpu_debugfs_evict_vram(void *data, u64 *val)
171175758255SAlex Deucher {
171298d28ac2SNirmoy Das struct amdgpu_device *adev = (struct amdgpu_device *)data;
171398d28ac2SNirmoy Das struct drm_device *dev = adev_to_drm(adev);
1714a9ffe2a9SAlex Deucher int r;
1715a9ffe2a9SAlex Deucher
1716a9ffe2a9SAlex Deucher r = pm_runtime_get_sync(dev->dev);
17179eee152aSAlex Deucher if (r < 0) {
171898d28ac2SNirmoy Das pm_runtime_put_autosuspend(dev->dev);
1719a9ffe2a9SAlex Deucher return r;
17209eee152aSAlex Deucher }
172175758255SAlex Deucher
172258144d28SNirmoy Das *val = amdgpu_ttm_evict_resources(adev, TTM_PL_VRAM);
1723a9ffe2a9SAlex Deucher
1724a9ffe2a9SAlex Deucher pm_runtime_mark_last_busy(dev->dev);
1725a9ffe2a9SAlex Deucher pm_runtime_put_autosuspend(dev->dev);
1726a9ffe2a9SAlex Deucher
172775758255SAlex Deucher return 0;
172875758255SAlex Deucher }
172975758255SAlex Deucher
173098d28ac2SNirmoy Das
amdgpu_debugfs_evict_gtt(void * data,u64 * val)173198d28ac2SNirmoy Das static int amdgpu_debugfs_evict_gtt(void *data, u64 *val)
173287e90c76SChristian König {
173398d28ac2SNirmoy Das struct amdgpu_device *adev = (struct amdgpu_device *)data;
173498d28ac2SNirmoy Das struct drm_device *dev = adev_to_drm(adev);
1735a9ffe2a9SAlex Deucher int r;
1736a9ffe2a9SAlex Deucher
1737a9ffe2a9SAlex Deucher r = pm_runtime_get_sync(dev->dev);
17389eee152aSAlex Deucher if (r < 0) {
173958144d28SNirmoy Das pm_runtime_put_autosuspend(dev->dev);
1740a9ffe2a9SAlex Deucher return r;
17419eee152aSAlex Deucher }
174287e90c76SChristian König
174358144d28SNirmoy Das *val = amdgpu_ttm_evict_resources(adev, TTM_PL_TT);
1744a9ffe2a9SAlex Deucher
1745a9ffe2a9SAlex Deucher pm_runtime_mark_last_busy(dev->dev);
1746a9ffe2a9SAlex Deucher pm_runtime_put_autosuspend(dev->dev);
1747a9ffe2a9SAlex Deucher
174887e90c76SChristian König return 0;
174987e90c76SChristian König }
175087e90c76SChristian König
amdgpu_debugfs_benchmark(void * data,u64 val)1751e7c47231SAlex Deucher static int amdgpu_debugfs_benchmark(void *data, u64 val)
1752e7c47231SAlex Deucher {
1753e7c47231SAlex Deucher struct amdgpu_device *adev = (struct amdgpu_device *)data;
1754e7c47231SAlex Deucher struct drm_device *dev = adev_to_drm(adev);
1755e7c47231SAlex Deucher int r;
1756e7c47231SAlex Deucher
1757e7c47231SAlex Deucher r = pm_runtime_get_sync(dev->dev);
1758e7c47231SAlex Deucher if (r < 0) {
1759e7c47231SAlex Deucher pm_runtime_put_autosuspend(dev->dev);
1760e7c47231SAlex Deucher return r;
1761e7c47231SAlex Deucher }
1762e7c47231SAlex Deucher
1763e7c47231SAlex Deucher r = amdgpu_benchmark(adev, val);
1764e7c47231SAlex Deucher
1765e7c47231SAlex Deucher pm_runtime_mark_last_busy(dev->dev);
1766e7c47231SAlex Deucher pm_runtime_put_autosuspend(dev->dev);
1767e7c47231SAlex Deucher
1768e7c47231SAlex Deucher return r;
1769e7c47231SAlex Deucher }
177098d28ac2SNirmoy Das
amdgpu_debugfs_vm_info_show(struct seq_file * m,void * unused)177198d28ac2SNirmoy Das static int amdgpu_debugfs_vm_info_show(struct seq_file *m, void *unused)
1772ff72bc40SMihir Bhogilal Patel {
1773109b4d8cSSu Hui struct amdgpu_device *adev = m->private;
177498d28ac2SNirmoy Das struct drm_device *dev = adev_to_drm(adev);
1775ff72bc40SMihir Bhogilal Patel struct drm_file *file;
1776ff72bc40SMihir Bhogilal Patel int r;
1777ff72bc40SMihir Bhogilal Patel
1778ff72bc40SMihir Bhogilal Patel r = mutex_lock_interruptible(&dev->filelist_mutex);
1779ff72bc40SMihir Bhogilal Patel if (r)
1780ff72bc40SMihir Bhogilal Patel return r;
1781ff72bc40SMihir Bhogilal Patel
1782ff72bc40SMihir Bhogilal Patel list_for_each_entry(file, &dev->filelist, lhead) {
1783ff72bc40SMihir Bhogilal Patel struct amdgpu_fpriv *fpriv = file->driver_priv;
1784ff72bc40SMihir Bhogilal Patel struct amdgpu_vm *vm = &fpriv->vm;
1785b8f67b9dSShashank Sharma struct amdgpu_task_info *ti;
1786ff72bc40SMihir Bhogilal Patel
1787b8f67b9dSShashank Sharma ti = amdgpu_vm_get_task_info_vm(vm);
1788b8f67b9dSShashank Sharma if (ti) {
1789b8f67b9dSShashank Sharma seq_printf(m, "pid:%d\tProcess:%s ----------\n", ti->pid, ti->process_name);
1790b8f67b9dSShashank Sharma amdgpu_vm_put_task_info(ti);
1791b8f67b9dSShashank Sharma }
1792b8f67b9dSShashank Sharma
1793391629bdSNirmoy Das r = amdgpu_bo_reserve(vm->root.bo, true);
1794ff72bc40SMihir Bhogilal Patel if (r)
1795ff72bc40SMihir Bhogilal Patel break;
1796ff72bc40SMihir Bhogilal Patel amdgpu_debugfs_vm_bo_info(vm, m);
1797391629bdSNirmoy Das amdgpu_bo_unreserve(vm->root.bo);
1798ff72bc40SMihir Bhogilal Patel }
1799ff72bc40SMihir Bhogilal Patel
1800ff72bc40SMihir Bhogilal Patel mutex_unlock(&dev->filelist_mutex);
1801ff72bc40SMihir Bhogilal Patel
1802ff72bc40SMihir Bhogilal Patel return r;
1803ff72bc40SMihir Bhogilal Patel }
1804ff72bc40SMihir Bhogilal Patel
180598d28ac2SNirmoy Das DEFINE_SHOW_ATTRIBUTE(amdgpu_debugfs_test_ib);
180698d28ac2SNirmoy Das DEFINE_SHOW_ATTRIBUTE(amdgpu_debugfs_vm_info);
180798d28ac2SNirmoy Das DEFINE_DEBUGFS_ATTRIBUTE(amdgpu_evict_vram_fops, amdgpu_debugfs_evict_vram,
180898d28ac2SNirmoy Das NULL, "%lld\n");
180998d28ac2SNirmoy Das DEFINE_DEBUGFS_ATTRIBUTE(amdgpu_evict_gtt_fops, amdgpu_debugfs_evict_gtt,
181098d28ac2SNirmoy Das NULL, "%lld\n");
1811e7c47231SAlex Deucher DEFINE_DEBUGFS_ATTRIBUTE(amdgpu_benchmark_fops, NULL, amdgpu_debugfs_benchmark,
1812e7c47231SAlex Deucher "%lld\n");
181375758255SAlex Deucher
amdgpu_ib_preempt_fences_swap(struct amdgpu_ring * ring,struct dma_fence ** fences)18146698a3d0SJack Xiao static void amdgpu_ib_preempt_fences_swap(struct amdgpu_ring *ring,
18156698a3d0SJack Xiao struct dma_fence **fences)
18166698a3d0SJack Xiao {
18176698a3d0SJack Xiao struct amdgpu_fence_driver *drv = &ring->fence_drv;
18186698a3d0SJack Xiao uint32_t sync_seq, last_seq;
18196698a3d0SJack Xiao
18206698a3d0SJack Xiao last_seq = atomic_read(&ring->fence_drv.last_seq);
18216698a3d0SJack Xiao sync_seq = ring->fence_drv.sync_seq;
18226698a3d0SJack Xiao
18236698a3d0SJack Xiao last_seq &= drv->num_fences_mask;
18246698a3d0SJack Xiao sync_seq &= drv->num_fences_mask;
18256698a3d0SJack Xiao
18266698a3d0SJack Xiao do {
18276698a3d0SJack Xiao struct dma_fence *fence, **ptr;
18286698a3d0SJack Xiao
18296698a3d0SJack Xiao ++last_seq;
18306698a3d0SJack Xiao last_seq &= drv->num_fences_mask;
18316698a3d0SJack Xiao ptr = &drv->fences[last_seq];
18326698a3d0SJack Xiao
18336698a3d0SJack Xiao fence = rcu_dereference_protected(*ptr, 1);
18346698a3d0SJack Xiao RCU_INIT_POINTER(*ptr, NULL);
18356698a3d0SJack Xiao
18366698a3d0SJack Xiao if (!fence)
18376698a3d0SJack Xiao continue;
18386698a3d0SJack Xiao
18396698a3d0SJack Xiao fences[last_seq] = fence;
18406698a3d0SJack Xiao
18416698a3d0SJack Xiao } while (last_seq != sync_seq);
18426698a3d0SJack Xiao }
18436698a3d0SJack Xiao
amdgpu_ib_preempt_signal_fences(struct dma_fence ** fences,int length)18446698a3d0SJack Xiao static void amdgpu_ib_preempt_signal_fences(struct dma_fence **fences,
18456698a3d0SJack Xiao int length)
18466698a3d0SJack Xiao {
18476698a3d0SJack Xiao int i;
18486698a3d0SJack Xiao struct dma_fence *fence;
18496698a3d0SJack Xiao
18506698a3d0SJack Xiao for (i = 0; i < length; i++) {
18516698a3d0SJack Xiao fence = fences[i];
18526698a3d0SJack Xiao if (!fence)
18536698a3d0SJack Xiao continue;
18546698a3d0SJack Xiao dma_fence_signal(fence);
18556698a3d0SJack Xiao dma_fence_put(fence);
18566698a3d0SJack Xiao }
18576698a3d0SJack Xiao }
18586698a3d0SJack Xiao
amdgpu_ib_preempt_job_recovery(struct drm_gpu_scheduler * sched)18596698a3d0SJack Xiao static void amdgpu_ib_preempt_job_recovery(struct drm_gpu_scheduler *sched)
18606698a3d0SJack Xiao {
18616698a3d0SJack Xiao struct drm_sched_job *s_job;
18626698a3d0SJack Xiao struct dma_fence *fence;
18636698a3d0SJack Xiao
18646698a3d0SJack Xiao spin_lock(&sched->job_list_lock);
18656efa4b46SLuben Tuikov list_for_each_entry(s_job, &sched->pending_list, list) {
18666698a3d0SJack Xiao fence = sched->ops->run_job(s_job);
18676698a3d0SJack Xiao dma_fence_put(fence);
18686698a3d0SJack Xiao }
18696698a3d0SJack Xiao spin_unlock(&sched->job_list_lock);
18706698a3d0SJack Xiao }
18716698a3d0SJack Xiao
amdgpu_ib_preempt_mark_partial_job(struct amdgpu_ring * ring)187280f8fb91SJack Xiao static void amdgpu_ib_preempt_mark_partial_job(struct amdgpu_ring *ring)
187380f8fb91SJack Xiao {
187480f8fb91SJack Xiao struct amdgpu_job *job;
18757bdb0899SJack Xiao struct drm_sched_job *s_job, *tmp;
187680f8fb91SJack Xiao uint32_t preempt_seq;
187780f8fb91SJack Xiao struct dma_fence *fence, **ptr;
187880f8fb91SJack Xiao struct amdgpu_fence_driver *drv = &ring->fence_drv;
187980f8fb91SJack Xiao struct drm_gpu_scheduler *sched = &ring->sched;
18807bdb0899SJack Xiao bool preempted = true;
188180f8fb91SJack Xiao
188280f8fb91SJack Xiao if (ring->funcs->type != AMDGPU_RING_TYPE_GFX)
188380f8fb91SJack Xiao return;
188480f8fb91SJack Xiao
188580f8fb91SJack Xiao preempt_seq = le32_to_cpu(*(drv->cpu_addr + 2));
18867bdb0899SJack Xiao if (preempt_seq <= atomic_read(&drv->last_seq)) {
18877bdb0899SJack Xiao preempted = false;
18887bdb0899SJack Xiao goto no_preempt;
18897bdb0899SJack Xiao }
189080f8fb91SJack Xiao
189180f8fb91SJack Xiao preempt_seq &= drv->num_fences_mask;
189280f8fb91SJack Xiao ptr = &drv->fences[preempt_seq];
189380f8fb91SJack Xiao fence = rcu_dereference_protected(*ptr, 1);
189480f8fb91SJack Xiao
18957bdb0899SJack Xiao no_preempt:
189680f8fb91SJack Xiao spin_lock(&sched->job_list_lock);
18976efa4b46SLuben Tuikov list_for_each_entry_safe(s_job, tmp, &sched->pending_list, list) {
18987bdb0899SJack Xiao if (dma_fence_is_signaled(&s_job->s_fence->finished)) {
18997bdb0899SJack Xiao /* remove job from ring_mirror_list */
19008935ff00SLuben Tuikov list_del_init(&s_job->list);
19017bdb0899SJack Xiao sched->ops->free_job(s_job);
19027bdb0899SJack Xiao continue;
19037bdb0899SJack Xiao }
190480f8fb91SJack Xiao job = to_amdgpu_job(s_job);
1905c530b02fSJack Zhang if (preempted && (&job->hw_fence) == fence)
190680f8fb91SJack Xiao /* mark the job as preempted */
190780f8fb91SJack Xiao job->preemption_status |= AMDGPU_IB_PREEMPTED;
190880f8fb91SJack Xiao }
190980f8fb91SJack Xiao spin_unlock(&sched->job_list_lock);
191080f8fb91SJack Xiao }
191180f8fb91SJack Xiao
amdgpu_debugfs_ib_preempt(void * data,u64 val)19126698a3d0SJack Xiao static int amdgpu_debugfs_ib_preempt(void *data, u64 val)
19136698a3d0SJack Xiao {
1914cd3a8a59SChristian König int r, length;
19156698a3d0SJack Xiao struct amdgpu_ring *ring;
19166698a3d0SJack Xiao struct dma_fence **fences = NULL;
19176698a3d0SJack Xiao struct amdgpu_device *adev = (struct amdgpu_device *)data;
19186698a3d0SJack Xiao
19196698a3d0SJack Xiao if (val >= AMDGPU_MAX_RINGS)
19206698a3d0SJack Xiao return -EINVAL;
19216698a3d0SJack Xiao
19226698a3d0SJack Xiao ring = adev->rings[val];
19236698a3d0SJack Xiao
19249749c868SMa Jun if (!amdgpu_ring_sched_ready(ring) ||
19259749c868SMa Jun !ring->funcs->preempt_ib)
19266698a3d0SJack Xiao return -EINVAL;
19276698a3d0SJack Xiao
19286698a3d0SJack Xiao /* the last preemption failed */
19296698a3d0SJack Xiao if (ring->trail_seq != le32_to_cpu(*ring->trail_fence_cpu_addr))
19306698a3d0SJack Xiao return -EBUSY;
19316698a3d0SJack Xiao
19326698a3d0SJack Xiao length = ring->fence_drv.num_fences_mask + 1;
19336698a3d0SJack Xiao fences = kcalloc(length, sizeof(void *), GFP_KERNEL);
19346698a3d0SJack Xiao if (!fences)
19356698a3d0SJack Xiao return -ENOMEM;
19366698a3d0SJack Xiao
1937a28fda31SAndrey Grodzovsky /* Avoid accidently unparking the sched thread during GPU reset */
1938d0fb18b5SAndrey Grodzovsky r = down_read_killable(&adev->reset_domain->sem);
19396049db43SDennis Li if (r)
19406049db43SDennis Li goto pro_end;
1941a28fda31SAndrey Grodzovsky
19426698a3d0SJack Xiao /* stop the scheduler */
194335963cf2SMatthew Brost drm_sched_wqueue_stop(&ring->sched);
19446698a3d0SJack Xiao
19456698a3d0SJack Xiao /* preempt the IB */
19466698a3d0SJack Xiao r = amdgpu_ring_preempt_ib(ring);
19476698a3d0SJack Xiao if (r) {
19486698a3d0SJack Xiao DRM_WARN("failed to preempt ring %d\n", ring->idx);
19496698a3d0SJack Xiao goto failure;
19506698a3d0SJack Xiao }
19516698a3d0SJack Xiao
19526698a3d0SJack Xiao amdgpu_fence_process(ring);
19536698a3d0SJack Xiao
19546698a3d0SJack Xiao if (atomic_read(&ring->fence_drv.last_seq) !=
19556698a3d0SJack Xiao ring->fence_drv.sync_seq) {
19566698a3d0SJack Xiao DRM_INFO("ring %d was preempted\n", ring->idx);
19576698a3d0SJack Xiao
195880f8fb91SJack Xiao amdgpu_ib_preempt_mark_partial_job(ring);
195980f8fb91SJack Xiao
19606698a3d0SJack Xiao /* swap out the old fences */
19616698a3d0SJack Xiao amdgpu_ib_preempt_fences_swap(ring, fences);
19626698a3d0SJack Xiao
19636698a3d0SJack Xiao amdgpu_fence_driver_force_completion(ring);
19646698a3d0SJack Xiao
19656698a3d0SJack Xiao /* resubmit unfinished jobs */
19666698a3d0SJack Xiao amdgpu_ib_preempt_job_recovery(&ring->sched);
19676698a3d0SJack Xiao
19686698a3d0SJack Xiao /* wait for jobs finished */
19696698a3d0SJack Xiao amdgpu_fence_wait_empty(ring);
19706698a3d0SJack Xiao
19716698a3d0SJack Xiao /* signal the old fences */
19726698a3d0SJack Xiao amdgpu_ib_preempt_signal_fences(fences, length);
19736698a3d0SJack Xiao }
19746698a3d0SJack Xiao
19756698a3d0SJack Xiao failure:
19766698a3d0SJack Xiao /* restart the scheduler */
197735963cf2SMatthew Brost drm_sched_wqueue_start(&ring->sched);
19786698a3d0SJack Xiao
1979d0fb18b5SAndrey Grodzovsky up_read(&adev->reset_domain->sem);
1980a28fda31SAndrey Grodzovsky
19816049db43SDennis Li pro_end:
19826698a3d0SJack Xiao kfree(fences);
19836698a3d0SJack Xiao
19846049db43SDennis Li return r;
19856698a3d0SJack Xiao }
19866698a3d0SJack Xiao
amdgpu_debugfs_sclk_set(void * data,u64 val)19870cf64555SChengming Gui static int amdgpu_debugfs_sclk_set(void *data, u64 val)
19880cf64555SChengming Gui {
19890cf64555SChengming Gui int ret = 0;
19900cf64555SChengming Gui uint32_t max_freq, min_freq;
19910cf64555SChengming Gui struct amdgpu_device *adev = (struct amdgpu_device *)data;
19920cf64555SChengming Gui
1993*8d5e70baSEmily Deng if (amdgpu_sriov_multi_vf_mode(adev))
19940cf64555SChengming Gui return -EINVAL;
19950cf64555SChengming Gui
19964a580877SLuben Tuikov ret = pm_runtime_get_sync(adev_to_drm(adev)->dev);
19979eee152aSAlex Deucher if (ret < 0) {
19984a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
19990cf64555SChengming Gui return ret;
20009eee152aSAlex Deucher }
20010cf64555SChengming Gui
2002bc143d8bSEvan Quan ret = amdgpu_dpm_get_dpm_freq_range(adev, PP_SCLK, &min_freq, &max_freq);
2003bc143d8bSEvan Quan if (ret == -EOPNOTSUPP) {
2004bc143d8bSEvan Quan ret = 0;
2005bc143d8bSEvan Quan goto out;
2006bc143d8bSEvan Quan }
2007bc143d8bSEvan Quan if (ret || val > max_freq || val < min_freq) {
2008bc143d8bSEvan Quan ret = -EINVAL;
2009bc143d8bSEvan Quan goto out;
20100cf64555SChengming Gui }
20110cf64555SChengming Gui
2012bc143d8bSEvan Quan ret = amdgpu_dpm_set_soft_freq_range(adev, PP_SCLK, (uint32_t)val, (uint32_t)val);
2013bc143d8bSEvan Quan if (ret)
2014bc143d8bSEvan Quan ret = -EINVAL;
2015bc143d8bSEvan Quan
2016bc143d8bSEvan Quan out:
20174a580877SLuben Tuikov pm_runtime_mark_last_busy(adev_to_drm(adev)->dev);
20184a580877SLuben Tuikov pm_runtime_put_autosuspend(adev_to_drm(adev)->dev);
20190cf64555SChengming Gui
2020bc143d8bSEvan Quan return ret;
20210cf64555SChengming Gui }
20220cf64555SChengming Gui
20237271a5c2SYang Li DEFINE_DEBUGFS_ATTRIBUTE(fops_ib_preempt, NULL,
20246698a3d0SJack Xiao amdgpu_debugfs_ib_preempt, "%llu\n");
20256698a3d0SJack Xiao
20267271a5c2SYang Li DEFINE_DEBUGFS_ATTRIBUTE(fops_sclk_set, NULL,
20270cf64555SChengming Gui amdgpu_debugfs_sclk_set, "%llu\n");
20280cf64555SChengming Gui
amdgpu_debugfs_init(struct amdgpu_device * adev)202975758255SAlex Deucher int amdgpu_debugfs_init(struct amdgpu_device *adev)
203075758255SAlex Deucher {
203198d28ac2SNirmoy Das struct dentry *root = adev_to_drm(adev)->primary->debugfs_root;
203288293c03SNirmoy Das struct dentry *ent;
2033fd23cfccSAlex Deucher int r, i;
2034c5820361SAlex Deucher
20355b9581dfSNirmoy Das if (!debugfs_initialized())
20365b9581dfSNirmoy Das return 0;
20375b9581dfSNirmoy Das
20386ff7fddbSLang Yu debugfs_create_x32("amdgpu_smu_debug", 0600, root,
20397e31a858SEvan Quan &adev->pm.smu_debug_mask);
20406ff7fddbSLang Yu
204198d28ac2SNirmoy Das ent = debugfs_create_file("amdgpu_preempt_ib", 0600, root, adev,
204244b582b3SGeert Uytterhoeven &fops_ib_preempt);
204359715cffSNirmoy Das if (IS_ERR(ent)) {
20446698a3d0SJack Xiao DRM_ERROR("unable to create amdgpu_preempt_ib debugsfs file\n");
204559715cffSNirmoy Das return PTR_ERR(ent);
20466698a3d0SJack Xiao }
20476698a3d0SJack Xiao
204898d28ac2SNirmoy Das ent = debugfs_create_file("amdgpu_force_sclk", 0200, root, adev,
20490cf64555SChengming Gui &fops_sclk_set);
205059715cffSNirmoy Das if (IS_ERR(ent)) {
20510cf64555SChengming Gui DRM_ERROR("unable to create amdgpu_set_sclk debugsfs file\n");
205259715cffSNirmoy Das return PTR_ERR(ent);
20530cf64555SChengming Gui }
20540cf64555SChengming Gui
2055c5820361SAlex Deucher /* Register debugfs entries for amdgpu_ttm */
205698d28ac2SNirmoy Das amdgpu_ttm_debugfs_init(adev);
2057373720f7SNirmoy Das amdgpu_debugfs_pm_init(adev);
205898d28ac2SNirmoy Das amdgpu_debugfs_sa_init(adev);
205998d28ac2SNirmoy Das amdgpu_debugfs_fence_init(adev);
206098d28ac2SNirmoy Das amdgpu_debugfs_gem_init(adev);
20613f5cea67SAlex Deucher
2062f9d64e6cSAlex Deucher r = amdgpu_debugfs_regs_init(adev);
2063f9d64e6cSAlex Deucher if (r)
2064f9d64e6cSAlex Deucher DRM_ERROR("registering register debugfs failed (%d).\n", r);
2065f9d64e6cSAlex Deucher
206698d28ac2SNirmoy Das amdgpu_debugfs_firmware_init(adev);
2067e50d9ba0SCandice Li amdgpu_ta_if_debugfs_init(adev);
2068cd9e29e7SAlex Deucher
2069b2662d4cSshaoyunl amdgpu_debugfs_mes_event_log_init(adev);
2070b2662d4cSshaoyunl
2071d090e7dbSAlex Deucher #if defined(CONFIG_DRM_AMD_DC)
2072d09ef243SAlex Deucher if (adev->dc_enabled)
2073afd3a359SNirmoy Das dtn_debugfs_init(adev);
2074d090e7dbSAlex Deucher #endif
2075d090e7dbSAlex Deucher
2076fd23cfccSAlex Deucher for (i = 0; i < AMDGPU_MAX_RINGS; ++i) {
2077fd23cfccSAlex Deucher struct amdgpu_ring *ring = adev->rings[i];
2078fd23cfccSAlex Deucher
2079fd23cfccSAlex Deucher if (!ring)
2080fd23cfccSAlex Deucher continue;
2081fd23cfccSAlex Deucher
208262d266b2SNirmoy Das amdgpu_debugfs_ring_init(adev, ring);
2083fd23cfccSAlex Deucher }
2084fd23cfccSAlex Deucher
208511eb648dSRuijing Dong for (i = 0; i < adev->vcn.num_vcn_inst; i++) {
208611eb648dSRuijing Dong if (!amdgpu_vcnfw_log)
208711eb648dSRuijing Dong break;
208811eb648dSRuijing Dong
208911eb648dSRuijing Dong if (adev->vcn.harvest_config & (1 << i))
209011eb648dSRuijing Dong continue;
209111eb648dSRuijing Dong
209211eb648dSRuijing Dong amdgpu_debugfs_vcn_fwlog_init(adev, i, &adev->vcn.inst[i]);
209311eb648dSRuijing Dong }
209411eb648dSRuijing Dong
209598a2e3a0SSaleemkhan Jamadar if (amdgpu_umsch_mm & amdgpu_umsch_mm_fwlog)
209698a2e3a0SSaleemkhan Jamadar amdgpu_debugfs_umsch_fwlog_init(adev, &adev->umsch_mm);
209798a2e3a0SSaleemkhan Jamadar
2098de258d06SSathishkumar S amdgpu_debugfs_vcn_sched_mask_init(adev);
2099f0b19b84SSathishkumar S amdgpu_debugfs_jpeg_sched_mask_init(adev);
2100c5c63d9cSJesse Zhang amdgpu_debugfs_gfx_sched_mask_init(adev);
2101c5c63d9cSJesse Zhang amdgpu_debugfs_compute_sched_mask_init(adev);
2102d2e3961aSJesse Zhang amdgpu_debugfs_sdma_sched_mask_init(adev);
2103f0b19b84SSathishkumar S
2104204eaac6STao Zhou amdgpu_ras_debugfs_create_all(adev);
2105a4322e18SWenhui Sheng amdgpu_rap_debugfs_init(adev);
2106ecaafb7bSJinzhou Su amdgpu_securedisplay_debugfs_init(adev);
210719ae3330SJohn Clements amdgpu_fw_attestation_debugfs_init(adev);
210819ae3330SJohn Clements
21097ba93954SAlex Deucher debugfs_create_file("amdgpu_evict_vram", 0400, root, adev,
211098d28ac2SNirmoy Das &amdgpu_evict_vram_fops);
21117ba93954SAlex Deucher debugfs_create_file("amdgpu_evict_gtt", 0400, root, adev,
211298d28ac2SNirmoy Das &amdgpu_evict_gtt_fops);
21137ba93954SAlex Deucher debugfs_create_file("amdgpu_test_ib", 0400, root, adev,
211498d28ac2SNirmoy Das &amdgpu_debugfs_test_ib_fops);
211598d28ac2SNirmoy Das debugfs_create_file("amdgpu_vm_info", 0444, root, adev,
211698d28ac2SNirmoy Das &amdgpu_debugfs_vm_info_fops);
2117e7c47231SAlex Deucher debugfs_create_file("amdgpu_benchmark", 0200, root, adev,
2118e7c47231SAlex Deucher &amdgpu_benchmark_fops);
211998d28ac2SNirmoy Das
212098d28ac2SNirmoy Das adev->debugfs_vbios_blob.data = adev->bios;
212198d28ac2SNirmoy Das adev->debugfs_vbios_blob.size = adev->bios_size;
212298d28ac2SNirmoy Das debugfs_create_blob("amdgpu_vbios", 0444, root,
212398d28ac2SNirmoy Das &adev->debugfs_vbios_blob);
212498d28ac2SNirmoy Das
212581d1bf01SAlex Deucher adev->debugfs_discovery_blob.data = adev->mman.discovery_bin;
212681d1bf01SAlex Deucher adev->debugfs_discovery_blob.size = adev->mman.discovery_tmr_size;
212781d1bf01SAlex Deucher debugfs_create_blob("amdgpu_discovery", 0444, root,
212881d1bf01SAlex Deucher &adev->debugfs_discovery_blob);
212981d1bf01SAlex Deucher
213098d28ac2SNirmoy Das return 0;
213175758255SAlex Deucher }
213275758255SAlex Deucher
213375758255SAlex Deucher #else
amdgpu_debugfs_init(struct amdgpu_device * adev)213475758255SAlex Deucher int amdgpu_debugfs_init(struct amdgpu_device *adev)
213575758255SAlex Deucher {
213675758255SAlex Deucher return 0;
213775758255SAlex Deucher }
amdgpu_debugfs_regs_init(struct amdgpu_device * adev)213875758255SAlex Deucher int amdgpu_debugfs_regs_init(struct amdgpu_device *adev)
213975758255SAlex Deucher {
214075758255SAlex Deucher return 0;
214175758255SAlex Deucher }
214275758255SAlex Deucher #endif
2143