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