xref: /linux/drivers/gpu/drm/amd/amdkfd/kfd_mqd_manager_v12.c (revision 20371ba120635d9ab7fc7670497105af8f33eb08)
148f0bdf4SDavid Belanger // SPDX-License-Identifier: GPL-2.0 OR MIT
248f0bdf4SDavid Belanger /*
348f0bdf4SDavid Belanger  * Copyright 2023 Advanced Micro Devices, Inc.
448f0bdf4SDavid Belanger  *
548f0bdf4SDavid Belanger  * Permission is hereby granted, free of charge, to any person obtaining a
648f0bdf4SDavid Belanger  * copy of this software and associated documentation files (the "Software"),
748f0bdf4SDavid Belanger  * to deal in the Software without restriction, including without limitation
848f0bdf4SDavid Belanger  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
948f0bdf4SDavid Belanger  * and/or sell copies of the Software, and to permit persons to whom the
1048f0bdf4SDavid Belanger  * Software is furnished to do so, subject to the following conditions:
1148f0bdf4SDavid Belanger  *
1248f0bdf4SDavid Belanger  * The above copyright notice and this permission notice shall be included in
1348f0bdf4SDavid Belanger  * all copies or substantial portions of the Software.
1448f0bdf4SDavid Belanger  *
1548f0bdf4SDavid Belanger  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1648f0bdf4SDavid Belanger  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1748f0bdf4SDavid Belanger  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
1848f0bdf4SDavid Belanger  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
1948f0bdf4SDavid Belanger  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
2048f0bdf4SDavid Belanger  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
2148f0bdf4SDavid Belanger  * OTHER DEALINGS IN THE SOFTWARE.
2248f0bdf4SDavid Belanger  *
2348f0bdf4SDavid Belanger  */
2448f0bdf4SDavid Belanger 
2548f0bdf4SDavid Belanger #include <linux/printk.h>
2648f0bdf4SDavid Belanger #include <linux/slab.h>
2748f0bdf4SDavid Belanger #include <linux/uaccess.h>
2848f0bdf4SDavid Belanger #include "kfd_priv.h"
2948f0bdf4SDavid Belanger #include "kfd_mqd_manager.h"
3048f0bdf4SDavid Belanger #include "v12_structs.h"
3148f0bdf4SDavid Belanger #include "gc/gc_12_0_0_sh_mask.h"
3248f0bdf4SDavid Belanger #include "amdgpu_amdkfd.h"
3348f0bdf4SDavid Belanger 
get_mqd(void * mqd)3448f0bdf4SDavid Belanger static inline struct v12_compute_mqd *get_mqd(void *mqd)
3548f0bdf4SDavid Belanger {
3648f0bdf4SDavid Belanger 	return (struct v12_compute_mqd *)mqd;
3748f0bdf4SDavid Belanger }
3848f0bdf4SDavid Belanger 
get_sdma_mqd(void * mqd)3948f0bdf4SDavid Belanger static inline struct v12_sdma_mqd *get_sdma_mqd(void *mqd)
4048f0bdf4SDavid Belanger {
4148f0bdf4SDavid Belanger 	return (struct v12_sdma_mqd *)mqd;
4248f0bdf4SDavid Belanger }
4348f0bdf4SDavid Belanger 
update_cu_mask(struct mqd_manager * mm,void * mqd,struct mqd_update_info * minfo)4448f0bdf4SDavid Belanger static void update_cu_mask(struct mqd_manager *mm, void *mqd,
4548f0bdf4SDavid Belanger 			   struct mqd_update_info *minfo)
4648f0bdf4SDavid Belanger {
4748f0bdf4SDavid Belanger 	struct v12_compute_mqd *m;
4848f0bdf4SDavid Belanger 	uint32_t se_mask[KFD_MAX_NUM_SE] = {0};
4948f0bdf4SDavid Belanger 
5048f0bdf4SDavid Belanger 	if (!minfo || !minfo->cu_mask.ptr)
5148f0bdf4SDavid Belanger 		return;
5248f0bdf4SDavid Belanger 
5348f0bdf4SDavid Belanger 	mqd_symmetrically_map_cu_mask(mm,
5448f0bdf4SDavid Belanger 		minfo->cu_mask.ptr, minfo->cu_mask.count, se_mask, 0);
5548f0bdf4SDavid Belanger 
5648f0bdf4SDavid Belanger 	m = get_mqd(mqd);
5748f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se0 = se_mask[0];
5848f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se1 = se_mask[1];
5948f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se2 = se_mask[2];
6048f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se3 = se_mask[3];
6148f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se4 = se_mask[4];
6248f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se5 = se_mask[5];
6348f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se6 = se_mask[6];
6448f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se7 = se_mask[7];
6548f0bdf4SDavid Belanger 
6648f0bdf4SDavid Belanger 	pr_debug("update cu mask to %#x %#x %#x %#x %#x %#x %#x %#x\n",
6748f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se0,
6848f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se1,
6948f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se2,
7048f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se3,
7148f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se4,
7248f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se5,
7348f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se6,
7448f0bdf4SDavid Belanger 		m->compute_static_thread_mgmt_se7);
7548f0bdf4SDavid Belanger }
7648f0bdf4SDavid Belanger 
set_priority(struct v12_compute_mqd * m,struct queue_properties * q)7748f0bdf4SDavid Belanger static void set_priority(struct v12_compute_mqd *m, struct queue_properties *q)
7848f0bdf4SDavid Belanger {
7948f0bdf4SDavid Belanger 	m->cp_hqd_pipe_priority = pipe_priority_map[q->priority];
8048f0bdf4SDavid Belanger 	m->cp_hqd_queue_priority = q->priority;
8148f0bdf4SDavid Belanger }
8248f0bdf4SDavid Belanger 
allocate_mqd(struct kfd_node * node,struct queue_properties * q)8348f0bdf4SDavid Belanger static struct kfd_mem_obj *allocate_mqd(struct kfd_node *node,
8448f0bdf4SDavid Belanger 		struct queue_properties *q)
8548f0bdf4SDavid Belanger {
8648f0bdf4SDavid Belanger 	struct kfd_mem_obj *mqd_mem_obj;
8748f0bdf4SDavid Belanger 
8848f0bdf4SDavid Belanger 	/*
8948f0bdf4SDavid Belanger 	 * Allocate one PAGE_SIZE memory for MQD as MES writes to areas beyond
9048f0bdf4SDavid Belanger 	 * struct MQD size.
9148f0bdf4SDavid Belanger 	 */
9248f0bdf4SDavid Belanger 	if (kfd_gtt_sa_allocate(node, PAGE_SIZE, &mqd_mem_obj))
9348f0bdf4SDavid Belanger 		return NULL;
9448f0bdf4SDavid Belanger 
9548f0bdf4SDavid Belanger 	return mqd_mem_obj;
9648f0bdf4SDavid Belanger }
9748f0bdf4SDavid Belanger 
init_mqd(struct mqd_manager * mm,void ** mqd,struct kfd_mem_obj * mqd_mem_obj,uint64_t * gart_addr,struct queue_properties * q)9848f0bdf4SDavid Belanger static void init_mqd(struct mqd_manager *mm, void **mqd,
9948f0bdf4SDavid Belanger 			struct kfd_mem_obj *mqd_mem_obj, uint64_t *gart_addr,
10048f0bdf4SDavid Belanger 			struct queue_properties *q)
10148f0bdf4SDavid Belanger {
10248f0bdf4SDavid Belanger 	uint64_t addr;
10348f0bdf4SDavid Belanger 	struct v12_compute_mqd *m;
10448f0bdf4SDavid Belanger 
10548f0bdf4SDavid Belanger 	m = (struct v12_compute_mqd *) mqd_mem_obj->cpu_ptr;
10648f0bdf4SDavid Belanger 	addr = mqd_mem_obj->gpu_addr;
10748f0bdf4SDavid Belanger 
10848f0bdf4SDavid Belanger 	memset(m, 0, PAGE_SIZE);
10948f0bdf4SDavid Belanger 
11048f0bdf4SDavid Belanger 	m->header = 0xC0310800;
11148f0bdf4SDavid Belanger 	m->compute_pipelinestat_enable = 1;
11248f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se0 = 0xFFFFFFFF;
11348f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se1 = 0xFFFFFFFF;
11448f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se2 = 0xFFFFFFFF;
11548f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se3 = 0xFFFFFFFF;
11648f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se4 = 0xFFFFFFFF;
11748f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se5 = 0xFFFFFFFF;
11848f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se6 = 0xFFFFFFFF;
11948f0bdf4SDavid Belanger 	m->compute_static_thread_mgmt_se7 = 0xFFFFFFFF;
12048f0bdf4SDavid Belanger 
12148f0bdf4SDavid Belanger 	m->cp_hqd_persistent_state = CP_HQD_PERSISTENT_STATE__PRELOAD_REQ_MASK |
12248f0bdf4SDavid Belanger 			0x55 << CP_HQD_PERSISTENT_STATE__PRELOAD_SIZE__SHIFT;
12348f0bdf4SDavid Belanger 
12448f0bdf4SDavid Belanger 	m->cp_mqd_control = 1 << CP_MQD_CONTROL__PRIV_STATE__SHIFT;
12548f0bdf4SDavid Belanger 
12648f0bdf4SDavid Belanger 	m->cp_mqd_base_addr_lo        = lower_32_bits(addr);
12748f0bdf4SDavid Belanger 	m->cp_mqd_base_addr_hi        = upper_32_bits(addr);
12848f0bdf4SDavid Belanger 
12948f0bdf4SDavid Belanger 	m->cp_hqd_quantum = 1 << CP_HQD_QUANTUM__QUANTUM_EN__SHIFT |
13048f0bdf4SDavid Belanger 			1 << CP_HQD_QUANTUM__QUANTUM_SCALE__SHIFT |
13148f0bdf4SDavid Belanger 			1 << CP_HQD_QUANTUM__QUANTUM_DURATION__SHIFT;
13248f0bdf4SDavid Belanger 
13348f0bdf4SDavid Belanger 	/* Set cp_hqd_hq_status0.c_queue_debug_en to 1 to have the CP set up the
13448f0bdf4SDavid Belanger 	 * DISPATCH_PTR.  This is required for the kfd debugger
13548f0bdf4SDavid Belanger 	 */
13648f0bdf4SDavid Belanger 	m->cp_hqd_hq_status0 = 1 << 14;
13748f0bdf4SDavid Belanger 
138c5faf18bSDavid Belanger 	if (amdgpu_amdkfd_have_atomics_support(mm->dev->adev))
139c5faf18bSDavid Belanger 		m->cp_hqd_hq_status0 |= 1 << 29;
140c5faf18bSDavid Belanger 
14148f0bdf4SDavid Belanger 	if (q->format == KFD_QUEUE_FORMAT_AQL) {
14248f0bdf4SDavid Belanger 		m->cp_hqd_aql_control =
14348f0bdf4SDavid Belanger 			1 << CP_HQD_AQL_CONTROL__CONTROL0__SHIFT;
14448f0bdf4SDavid Belanger 	}
14548f0bdf4SDavid Belanger 
14648f0bdf4SDavid Belanger 	if (mm->dev->kfd->cwsr_enabled) {
14748f0bdf4SDavid Belanger 		m->cp_hqd_persistent_state |=
14848f0bdf4SDavid Belanger 			(1 << CP_HQD_PERSISTENT_STATE__QSWITCH_MODE__SHIFT);
14948f0bdf4SDavid Belanger 		m->cp_hqd_ctx_save_base_addr_lo =
15048f0bdf4SDavid Belanger 			lower_32_bits(q->ctx_save_restore_area_address);
15148f0bdf4SDavid Belanger 		m->cp_hqd_ctx_save_base_addr_hi =
15248f0bdf4SDavid Belanger 			upper_32_bits(q->ctx_save_restore_area_address);
15348f0bdf4SDavid Belanger 		m->cp_hqd_ctx_save_size = q->ctx_save_restore_area_size;
15448f0bdf4SDavid Belanger 		m->cp_hqd_cntl_stack_size = q->ctl_stack_size;
15548f0bdf4SDavid Belanger 		m->cp_hqd_cntl_stack_offset = q->ctl_stack_size;
15648f0bdf4SDavid Belanger 		m->cp_hqd_wg_state_offset = q->ctl_stack_size;
15748f0bdf4SDavid Belanger 	}
15848f0bdf4SDavid Belanger 
15948f0bdf4SDavid Belanger 	*mqd = m;
16048f0bdf4SDavid Belanger 	if (gart_addr)
16148f0bdf4SDavid Belanger 		*gart_addr = addr;
16248f0bdf4SDavid Belanger 	mm->update_mqd(mm, m, q, NULL);
16348f0bdf4SDavid Belanger }
16448f0bdf4SDavid Belanger 
load_mqd(struct mqd_manager * mm,void * mqd,uint32_t pipe_id,uint32_t queue_id,struct queue_properties * p,struct mm_struct * mms)16548f0bdf4SDavid Belanger static int load_mqd(struct mqd_manager *mm, void *mqd,
16648f0bdf4SDavid Belanger 			uint32_t pipe_id, uint32_t queue_id,
16748f0bdf4SDavid Belanger 			struct queue_properties *p, struct mm_struct *mms)
16848f0bdf4SDavid Belanger {
16948f0bdf4SDavid Belanger 	int r = 0;
17048f0bdf4SDavid Belanger 	/* AQL write pointer counts in 64B packets, PM4/CP counts in dwords. */
17148f0bdf4SDavid Belanger 	uint32_t wptr_shift = (p->format == KFD_QUEUE_FORMAT_AQL ? 4 : 0);
17248f0bdf4SDavid Belanger 
17348f0bdf4SDavid Belanger 	r = mm->dev->kfd2kgd->hqd_load(mm->dev->adev, mqd, pipe_id, queue_id,
17448f0bdf4SDavid Belanger 					  (uint32_t __user *)p->write_ptr,
17548f0bdf4SDavid Belanger 					  wptr_shift, 0, mms, 0);
17648f0bdf4SDavid Belanger 	return r;
17748f0bdf4SDavid Belanger }
17848f0bdf4SDavid Belanger 
update_mqd(struct mqd_manager * mm,void * mqd,struct queue_properties * q,struct mqd_update_info * minfo)17948f0bdf4SDavid Belanger static void update_mqd(struct mqd_manager *mm, void *mqd,
18048f0bdf4SDavid Belanger 		       struct queue_properties *q,
18148f0bdf4SDavid Belanger 		       struct mqd_update_info *minfo)
18248f0bdf4SDavid Belanger {
18348f0bdf4SDavid Belanger 	struct v12_compute_mqd *m;
18448f0bdf4SDavid Belanger 
18548f0bdf4SDavid Belanger 	m = get_mqd(mqd);
18648f0bdf4SDavid Belanger 
18748f0bdf4SDavid Belanger 	m->cp_hqd_pq_control = 5 << CP_HQD_PQ_CONTROL__RPTR_BLOCK_SIZE__SHIFT;
18848f0bdf4SDavid Belanger 	m->cp_hqd_pq_control |=
18948f0bdf4SDavid Belanger 			ffs(q->queue_size / sizeof(unsigned int)) - 1 - 1;
190*959fc102SAlex Deucher 	m->cp_hqd_pq_control |= CP_HQD_PQ_CONTROL__UNORD_DISPATCH_MASK;
19148f0bdf4SDavid Belanger 	pr_debug("cp_hqd_pq_control 0x%x\n", m->cp_hqd_pq_control);
19248f0bdf4SDavid Belanger 
19348f0bdf4SDavid Belanger 	m->cp_hqd_pq_base_lo = lower_32_bits((uint64_t)q->queue_address >> 8);
19448f0bdf4SDavid Belanger 	m->cp_hqd_pq_base_hi = upper_32_bits((uint64_t)q->queue_address >> 8);
19548f0bdf4SDavid Belanger 
19648f0bdf4SDavid Belanger 	m->cp_hqd_pq_rptr_report_addr_lo = lower_32_bits((uint64_t)q->read_ptr);
19748f0bdf4SDavid Belanger 	m->cp_hqd_pq_rptr_report_addr_hi = upper_32_bits((uint64_t)q->read_ptr);
19848f0bdf4SDavid Belanger 	m->cp_hqd_pq_wptr_poll_addr_lo = lower_32_bits((uint64_t)q->write_ptr);
19948f0bdf4SDavid Belanger 	m->cp_hqd_pq_wptr_poll_addr_hi = upper_32_bits((uint64_t)q->write_ptr);
20048f0bdf4SDavid Belanger 
20148f0bdf4SDavid Belanger 	m->cp_hqd_pq_doorbell_control =
20248f0bdf4SDavid Belanger 		q->doorbell_off <<
20348f0bdf4SDavid Belanger 			CP_HQD_PQ_DOORBELL_CONTROL__DOORBELL_OFFSET__SHIFT;
20448f0bdf4SDavid Belanger 	pr_debug("cp_hqd_pq_doorbell_control 0x%x\n",
20548f0bdf4SDavid Belanger 			m->cp_hqd_pq_doorbell_control);
20648f0bdf4SDavid Belanger 
20748f0bdf4SDavid Belanger 	m->cp_hqd_ib_control = 3 << CP_HQD_IB_CONTROL__MIN_IB_AVAIL_SIZE__SHIFT;
20848f0bdf4SDavid Belanger 
20948f0bdf4SDavid Belanger 	/*
21048f0bdf4SDavid Belanger 	 * HW does not clamp this field correctly. Maximum EOP queue size
21148f0bdf4SDavid Belanger 	 * is constrained by per-SE EOP done signal count, which is 8-bit.
21248f0bdf4SDavid Belanger 	 * Limit is 0xFF EOP entries (= 0x7F8 dwords). CP will not submit
21348f0bdf4SDavid Belanger 	 * more than (EOP entry count - 1) so a queue size of 0x800 dwords
21448f0bdf4SDavid Belanger 	 * is safe, giving a maximum field value of 0xA.
21548f0bdf4SDavid Belanger 	 */
21648f0bdf4SDavid Belanger 	m->cp_hqd_eop_control = min(0xA,
21748f0bdf4SDavid Belanger 		ffs(q->eop_ring_buffer_size / sizeof(unsigned int)) - 1 - 1);
21848f0bdf4SDavid Belanger 	m->cp_hqd_eop_base_addr_lo =
21948f0bdf4SDavid Belanger 			lower_32_bits(q->eop_ring_buffer_address >> 8);
22048f0bdf4SDavid Belanger 	m->cp_hqd_eop_base_addr_hi =
22148f0bdf4SDavid Belanger 			upper_32_bits(q->eop_ring_buffer_address >> 8);
22248f0bdf4SDavid Belanger 
22348f0bdf4SDavid Belanger 	m->cp_hqd_iq_timer = 0;
22448f0bdf4SDavid Belanger 
22548f0bdf4SDavid Belanger 	m->cp_hqd_vmid = q->vmid;
22648f0bdf4SDavid Belanger 
22748f0bdf4SDavid Belanger 	if (q->format == KFD_QUEUE_FORMAT_AQL) {
22848f0bdf4SDavid Belanger 		/* GC 10 removed WPP_CLAMP from PQ Control */
22948f0bdf4SDavid Belanger 		m->cp_hqd_pq_control |= CP_HQD_PQ_CONTROL__NO_UPDATE_RPTR_MASK |
23048f0bdf4SDavid Belanger 				2 << CP_HQD_PQ_CONTROL__SLOT_BASED_WPTR__SHIFT |
23148f0bdf4SDavid Belanger 				1 << CP_HQD_PQ_CONTROL__QUEUE_FULL_EN__SHIFT;
23248f0bdf4SDavid Belanger 		m->cp_hqd_pq_doorbell_control |=
23348f0bdf4SDavid Belanger 			1 << CP_HQD_PQ_DOORBELL_CONTROL__DOORBELL_BIF_DROP__SHIFT;
23448f0bdf4SDavid Belanger 	}
23548f0bdf4SDavid Belanger 	if (mm->dev->kfd->cwsr_enabled)
23648f0bdf4SDavid Belanger 		m->cp_hqd_ctx_save_control = 0;
23748f0bdf4SDavid Belanger 
23848f0bdf4SDavid Belanger 	update_cu_mask(mm, mqd, minfo);
23948f0bdf4SDavid Belanger 	set_priority(m, q);
24048f0bdf4SDavid Belanger 
24148f0bdf4SDavid Belanger 	q->is_active = QUEUE_IS_ACTIVE(*q);
24248f0bdf4SDavid Belanger }
24348f0bdf4SDavid Belanger 
check_preemption_failed(struct mqd_manager * mm,void * mqd)24448f0bdf4SDavid Belanger static bool check_preemption_failed(struct mqd_manager *mm, void *mqd)
24548f0bdf4SDavid Belanger {
24648f0bdf4SDavid Belanger 	struct v12_compute_mqd *m = (struct v12_compute_mqd *)mqd;
24748f0bdf4SDavid Belanger 
24848f0bdf4SDavid Belanger 	return kfd_check_hiq_mqd_doorbell_id(mm->dev, m->queue_doorbell_id0, 0);
24948f0bdf4SDavid Belanger }
25048f0bdf4SDavid Belanger 
get_wave_state(struct mqd_manager * mm,void * mqd,struct queue_properties * q,void __user * ctl_stack,u32 * ctl_stack_used_size,u32 * save_area_used_size)25148f0bdf4SDavid Belanger static int get_wave_state(struct mqd_manager *mm, void *mqd,
25248f0bdf4SDavid Belanger 			  struct queue_properties *q,
25348f0bdf4SDavid Belanger 			  void __user *ctl_stack,
25448f0bdf4SDavid Belanger 			  u32 *ctl_stack_used_size,
25548f0bdf4SDavid Belanger 			  u32 *save_area_used_size)
25648f0bdf4SDavid Belanger {
25748f0bdf4SDavid Belanger 	struct v12_compute_mqd *m;
25848f0bdf4SDavid Belanger 	struct mqd_user_context_save_area_header header;
25948f0bdf4SDavid Belanger 
26048f0bdf4SDavid Belanger 	m = get_mqd(mqd);
26148f0bdf4SDavid Belanger 
26248f0bdf4SDavid Belanger 	/* Control stack is written backwards, while workgroup context data
26348f0bdf4SDavid Belanger 	 * is written forwards. Both starts from m->cp_hqd_cntl_stack_size.
26448f0bdf4SDavid Belanger 	 * Current position is at m->cp_hqd_cntl_stack_offset and
26548f0bdf4SDavid Belanger 	 * m->cp_hqd_wg_state_offset, respectively.
26648f0bdf4SDavid Belanger 	 */
26748f0bdf4SDavid Belanger 	*ctl_stack_used_size = m->cp_hqd_cntl_stack_size -
26848f0bdf4SDavid Belanger 		m->cp_hqd_cntl_stack_offset;
26948f0bdf4SDavid Belanger 	*save_area_used_size = m->cp_hqd_wg_state_offset -
27048f0bdf4SDavid Belanger 		m->cp_hqd_cntl_stack_size;
27148f0bdf4SDavid Belanger 
27248f0bdf4SDavid Belanger 	/* Control stack is not copied to user mode for GFXv12 because
27348f0bdf4SDavid Belanger 	 * it's part of the context save area that is already
27448f0bdf4SDavid Belanger 	 * accessible to user mode
27548f0bdf4SDavid Belanger 	 */
27648f0bdf4SDavid Belanger 	header.control_stack_size = *ctl_stack_used_size;
27748f0bdf4SDavid Belanger 	header.wave_state_size = *save_area_used_size;
27848f0bdf4SDavid Belanger 
27948f0bdf4SDavid Belanger 	header.wave_state_offset = m->cp_hqd_wg_state_offset;
28048f0bdf4SDavid Belanger 	header.control_stack_offset = m->cp_hqd_cntl_stack_offset;
28148f0bdf4SDavid Belanger 
28248f0bdf4SDavid Belanger 	if (copy_to_user(ctl_stack, &header, sizeof(header)))
28348f0bdf4SDavid Belanger 		return -EFAULT;
28448f0bdf4SDavid Belanger 
28548f0bdf4SDavid Belanger 	return 0;
28648f0bdf4SDavid Belanger }
28748f0bdf4SDavid Belanger 
init_mqd_hiq(struct mqd_manager * mm,void ** mqd,struct kfd_mem_obj * mqd_mem_obj,uint64_t * gart_addr,struct queue_properties * q)28848f0bdf4SDavid Belanger static void init_mqd_hiq(struct mqd_manager *mm, void **mqd,
28948f0bdf4SDavid Belanger 			struct kfd_mem_obj *mqd_mem_obj, uint64_t *gart_addr,
29048f0bdf4SDavid Belanger 			struct queue_properties *q)
29148f0bdf4SDavid Belanger {
29248f0bdf4SDavid Belanger 	struct v12_compute_mqd *m;
29348f0bdf4SDavid Belanger 
29448f0bdf4SDavid Belanger 	init_mqd(mm, mqd, mqd_mem_obj, gart_addr, q);
29548f0bdf4SDavid Belanger 
29648f0bdf4SDavid Belanger 	m = get_mqd(*mqd);
29748f0bdf4SDavid Belanger 
29848f0bdf4SDavid Belanger 	m->cp_hqd_pq_control |= 1 << CP_HQD_PQ_CONTROL__PRIV_STATE__SHIFT |
29948f0bdf4SDavid Belanger 			1 << CP_HQD_PQ_CONTROL__KMD_QUEUE__SHIFT;
30048f0bdf4SDavid Belanger }
30148f0bdf4SDavid Belanger 
init_mqd_sdma(struct mqd_manager * mm,void ** mqd,struct kfd_mem_obj * mqd_mem_obj,uint64_t * gart_addr,struct queue_properties * q)30248f0bdf4SDavid Belanger static void init_mqd_sdma(struct mqd_manager *mm, void **mqd,
30348f0bdf4SDavid Belanger 		struct kfd_mem_obj *mqd_mem_obj, uint64_t *gart_addr,
30448f0bdf4SDavid Belanger 		struct queue_properties *q)
30548f0bdf4SDavid Belanger {
30648f0bdf4SDavid Belanger 	struct v12_sdma_mqd *m;
30748f0bdf4SDavid Belanger 
30848f0bdf4SDavid Belanger 	m = (struct v12_sdma_mqd *) mqd_mem_obj->cpu_ptr;
30948f0bdf4SDavid Belanger 
31048f0bdf4SDavid Belanger 	memset(m, 0, sizeof(struct v12_sdma_mqd));
31148f0bdf4SDavid Belanger 
31248f0bdf4SDavid Belanger 	*mqd = m;
31348f0bdf4SDavid Belanger 	if (gart_addr)
31448f0bdf4SDavid Belanger 		*gart_addr = mqd_mem_obj->gpu_addr;
31548f0bdf4SDavid Belanger 
31648f0bdf4SDavid Belanger 	mm->update_mqd(mm, m, q, NULL);
31748f0bdf4SDavid Belanger }
31848f0bdf4SDavid Belanger 
31948f0bdf4SDavid Belanger #define SDMA_RLC_DUMMY_DEFAULT 0xf
32048f0bdf4SDavid Belanger 
update_mqd_sdma(struct mqd_manager * mm,void * mqd,struct queue_properties * q,struct mqd_update_info * minfo)32148f0bdf4SDavid Belanger static void update_mqd_sdma(struct mqd_manager *mm, void *mqd,
32248f0bdf4SDavid Belanger 		struct queue_properties *q,
32348f0bdf4SDavid Belanger 		struct mqd_update_info *minfo)
32448f0bdf4SDavid Belanger {
32548f0bdf4SDavid Belanger 	struct v12_sdma_mqd *m;
32648f0bdf4SDavid Belanger 
32748f0bdf4SDavid Belanger 	m = get_sdma_mqd(mqd);
32848f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_cntl = (ffs(q->queue_size / sizeof(unsigned int)) - 1)
32948f0bdf4SDavid Belanger 		<< SDMA0_QUEUE0_RB_CNTL__RB_SIZE__SHIFT |
33048f0bdf4SDavid Belanger 		q->vmid << SDMA0_QUEUE0_RB_CNTL__RB_VMID__SHIFT |
33148f0bdf4SDavid Belanger 		1 << SDMA0_QUEUE0_RB_CNTL__RPTR_WRITEBACK_ENABLE__SHIFT |
33248f0bdf4SDavid Belanger 		6 << SDMA0_QUEUE0_RB_CNTL__RPTR_WRITEBACK_TIMER__SHIFT |
33348f0bdf4SDavid Belanger 		1 << SDMA0_QUEUE0_RB_CNTL__MCU_WPTR_POLL_ENABLE__SHIFT;
33448f0bdf4SDavid Belanger 
33548f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_base = lower_32_bits(q->queue_address >> 8);
33648f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_base_hi = upper_32_bits(q->queue_address >> 8);
33748f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_rptr_addr_lo = lower_32_bits((uint64_t)q->read_ptr);
33848f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_rptr_addr_hi = upper_32_bits((uint64_t)q->read_ptr);
33948f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_wptr_poll_addr_lo = lower_32_bits((uint64_t)q->write_ptr);
34048f0bdf4SDavid Belanger 	m->sdmax_rlcx_rb_wptr_poll_addr_hi = upper_32_bits((uint64_t)q->write_ptr);
34148f0bdf4SDavid Belanger 	m->sdmax_rlcx_doorbell_offset =
34248f0bdf4SDavid Belanger 		q->doorbell_off << SDMA0_QUEUE0_DOORBELL_OFFSET__OFFSET__SHIFT;
34348f0bdf4SDavid Belanger 
34448f0bdf4SDavid Belanger 	m->sdma_engine_id = q->sdma_engine_id;
34548f0bdf4SDavid Belanger 	m->sdma_queue_id = q->sdma_queue_id;
34648f0bdf4SDavid Belanger 
34748f0bdf4SDavid Belanger 	m->sdmax_rlcx_dummy_reg = SDMA_RLC_DUMMY_DEFAULT;
34848f0bdf4SDavid Belanger 
34948f0bdf4SDavid Belanger 	q->is_active = QUEUE_IS_ACTIVE(*q);
35048f0bdf4SDavid Belanger }
35148f0bdf4SDavid Belanger 
35248f0bdf4SDavid Belanger #if defined(CONFIG_DEBUG_FS)
35348f0bdf4SDavid Belanger 
debugfs_show_mqd(struct seq_file * m,void * data)35448f0bdf4SDavid Belanger static int debugfs_show_mqd(struct seq_file *m, void *data)
35548f0bdf4SDavid Belanger {
35648f0bdf4SDavid Belanger 	seq_hex_dump(m, "    ", DUMP_PREFIX_OFFSET, 32, 4,
35748f0bdf4SDavid Belanger 		     data, sizeof(struct v12_compute_mqd), false);
35848f0bdf4SDavid Belanger 	return 0;
35948f0bdf4SDavid Belanger }
36048f0bdf4SDavid Belanger 
debugfs_show_mqd_sdma(struct seq_file * m,void * data)36148f0bdf4SDavid Belanger static int debugfs_show_mqd_sdma(struct seq_file *m, void *data)
36248f0bdf4SDavid Belanger {
36348f0bdf4SDavid Belanger 	seq_hex_dump(m, "    ", DUMP_PREFIX_OFFSET, 32, 4,
36448f0bdf4SDavid Belanger 		     data, sizeof(struct v12_sdma_mqd), false);
36548f0bdf4SDavid Belanger 	return 0;
36648f0bdf4SDavid Belanger }
36748f0bdf4SDavid Belanger 
36848f0bdf4SDavid Belanger #endif
36948f0bdf4SDavid Belanger 
mqd_manager_init_v12(enum KFD_MQD_TYPE type,struct kfd_node * dev)37048f0bdf4SDavid Belanger struct mqd_manager *mqd_manager_init_v12(enum KFD_MQD_TYPE type,
37148f0bdf4SDavid Belanger 		struct kfd_node *dev)
37248f0bdf4SDavid Belanger {
37348f0bdf4SDavid Belanger 	struct mqd_manager *mqd;
37448f0bdf4SDavid Belanger 
37548f0bdf4SDavid Belanger 	if (WARN_ON(type >= KFD_MQD_TYPE_MAX))
37648f0bdf4SDavid Belanger 		return NULL;
37748f0bdf4SDavid Belanger 
37848f0bdf4SDavid Belanger 	mqd = kzalloc(sizeof(*mqd), GFP_KERNEL);
37948f0bdf4SDavid Belanger 	if (!mqd)
38048f0bdf4SDavid Belanger 		return NULL;
38148f0bdf4SDavid Belanger 
38248f0bdf4SDavid Belanger 	mqd->dev = dev;
38348f0bdf4SDavid Belanger 
38448f0bdf4SDavid Belanger 	switch (type) {
38548f0bdf4SDavid Belanger 	case KFD_MQD_TYPE_CP:
38648f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
38748f0bdf4SDavid Belanger 		mqd->allocate_mqd = allocate_mqd;
38848f0bdf4SDavid Belanger 		mqd->init_mqd = init_mqd;
38948f0bdf4SDavid Belanger 		mqd->free_mqd = kfd_free_mqd_cp;
39048f0bdf4SDavid Belanger 		mqd->load_mqd = load_mqd;
39148f0bdf4SDavid Belanger 		mqd->update_mqd = update_mqd;
39248f0bdf4SDavid Belanger 		mqd->destroy_mqd = kfd_destroy_mqd_cp;
39348f0bdf4SDavid Belanger 		mqd->is_occupied = kfd_is_occupied_cp;
39448f0bdf4SDavid Belanger 		mqd->mqd_size = sizeof(struct v12_compute_mqd);
39548f0bdf4SDavid Belanger 		mqd->get_wave_state = get_wave_state;
396a921c35aSEric Huang 		mqd->mqd_stride = kfd_mqd_stride;
39748f0bdf4SDavid Belanger #if defined(CONFIG_DEBUG_FS)
39848f0bdf4SDavid Belanger 		mqd->debugfs_show_mqd = debugfs_show_mqd;
39948f0bdf4SDavid Belanger #endif
40048f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
40148f0bdf4SDavid Belanger 		break;
40248f0bdf4SDavid Belanger 	case KFD_MQD_TYPE_HIQ:
40348f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
40448f0bdf4SDavid Belanger 		mqd->allocate_mqd = allocate_hiq_mqd;
40548f0bdf4SDavid Belanger 		mqd->init_mqd = init_mqd_hiq;
40648f0bdf4SDavid Belanger 		mqd->free_mqd = free_mqd_hiq_sdma;
40748f0bdf4SDavid Belanger 		mqd->load_mqd = kfd_hiq_load_mqd_kiq;
40848f0bdf4SDavid Belanger 		mqd->update_mqd = update_mqd;
40948f0bdf4SDavid Belanger 		mqd->destroy_mqd = kfd_destroy_mqd_cp;
41048f0bdf4SDavid Belanger 		mqd->is_occupied = kfd_is_occupied_cp;
41148f0bdf4SDavid Belanger 		mqd->mqd_size = sizeof(struct v12_compute_mqd);
412a921c35aSEric Huang 		mqd->mqd_stride = kfd_mqd_stride;
41348f0bdf4SDavid Belanger #if defined(CONFIG_DEBUG_FS)
41448f0bdf4SDavid Belanger 		mqd->debugfs_show_mqd = debugfs_show_mqd;
41548f0bdf4SDavid Belanger #endif
41648f0bdf4SDavid Belanger 		mqd->check_preemption_failed = check_preemption_failed;
41748f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
41848f0bdf4SDavid Belanger 		break;
41948f0bdf4SDavid Belanger 	case KFD_MQD_TYPE_DIQ:
42048f0bdf4SDavid Belanger 		mqd->allocate_mqd = allocate_mqd;
42148f0bdf4SDavid Belanger 		mqd->init_mqd = init_mqd_hiq;
42248f0bdf4SDavid Belanger 		mqd->free_mqd = kfd_free_mqd_cp;
42348f0bdf4SDavid Belanger 		mqd->load_mqd = load_mqd;
42448f0bdf4SDavid Belanger 		mqd->update_mqd = update_mqd;
42548f0bdf4SDavid Belanger 		mqd->destroy_mqd = kfd_destroy_mqd_cp;
42648f0bdf4SDavid Belanger 		mqd->is_occupied = kfd_is_occupied_cp;
42748f0bdf4SDavid Belanger 		mqd->mqd_size = sizeof(struct v12_compute_mqd);
42848f0bdf4SDavid Belanger #if defined(CONFIG_DEBUG_FS)
42948f0bdf4SDavid Belanger 		mqd->debugfs_show_mqd = debugfs_show_mqd;
43048f0bdf4SDavid Belanger #endif
43148f0bdf4SDavid Belanger 		break;
43248f0bdf4SDavid Belanger 	case KFD_MQD_TYPE_SDMA:
43348f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
43448f0bdf4SDavid Belanger 		mqd->allocate_mqd = allocate_mqd;
43548f0bdf4SDavid Belanger 		mqd->init_mqd = init_mqd_sdma;
43648f0bdf4SDavid Belanger 		mqd->free_mqd = kfd_free_mqd_cp;
43748f0bdf4SDavid Belanger 		mqd->load_mqd = kfd_load_mqd_sdma;
43848f0bdf4SDavid Belanger 		mqd->update_mqd = update_mqd_sdma;
43948f0bdf4SDavid Belanger 		mqd->destroy_mqd = kfd_destroy_mqd_sdma;
44048f0bdf4SDavid Belanger 		mqd->is_occupied = kfd_is_occupied_sdma;
44148f0bdf4SDavid Belanger 		mqd->mqd_size = sizeof(struct v12_sdma_mqd);
442a921c35aSEric Huang 		mqd->mqd_stride = kfd_mqd_stride;
44348f0bdf4SDavid Belanger #if defined(CONFIG_DEBUG_FS)
44448f0bdf4SDavid Belanger 		mqd->debugfs_show_mqd = debugfs_show_mqd_sdma;
44548f0bdf4SDavid Belanger #endif
44648f0bdf4SDavid Belanger 		pr_debug("%s@%i\n", __func__, __LINE__);
44748f0bdf4SDavid Belanger 		break;
44848f0bdf4SDavid Belanger 	default:
44948f0bdf4SDavid Belanger 		kfree(mqd);
45048f0bdf4SDavid Belanger 		return NULL;
45148f0bdf4SDavid Belanger 	}
45248f0bdf4SDavid Belanger 
45348f0bdf4SDavid Belanger 	return mqd;
45448f0bdf4SDavid Belanger }
455