Commit fb910658 authored by Philip Yang's avatar Philip Yang Committed by Alex Deucher

drm/amdkfd: Refactor queue wptr_bo GART mapping

Add helper function kfd_queue_acquire_buffers to get queue wptr_bo
reference from queue write_ptr if it is mapped to the KFD node with
expected size.

Add wptr_bo to structure queue_properties because structure queue is
allocated after queue buffers are validated, then we can remove wptr_bo
parameter from pqm_create_queue.

Rename structure queue wptr_bo_gart to hold wptr_bo reference for GART
mapping and umapping. Move MES wptr_bo_gart mapping to init_user_queue,
the same location with queue ctx_bo GART mapping.
Signed-off-by: default avatarPhilip Yang <Philip.Yang@amd.com>
Reviewed-by: default avatarFelix Kuehling <felix.kuehling@amd.com>
Acked-by: default avatarChristian König <christian.koenig@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent db54a725
...@@ -322,7 +322,7 @@ int amdgpu_amdkfd_gpuvm_map_gtt_bo_to_kernel(struct kgd_mem *mem, ...@@ -322,7 +322,7 @@ int amdgpu_amdkfd_gpuvm_map_gtt_bo_to_kernel(struct kgd_mem *mem,
void **kptr, uint64_t *size); void **kptr, uint64_t *size);
void amdgpu_amdkfd_gpuvm_unmap_gtt_bo_from_kernel(struct kgd_mem *mem); void amdgpu_amdkfd_gpuvm_unmap_gtt_bo_from_kernel(struct kgd_mem *mem);
int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo); int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo, struct amdgpu_bo **bo_gart);
int amdgpu_amdkfd_gpuvm_restore_process_bos(void *process_info, int amdgpu_amdkfd_gpuvm_restore_process_bos(void *process_info,
struct dma_fence __rcu **ef); struct dma_fence __rcu **ef);
......
...@@ -2226,11 +2226,12 @@ int amdgpu_amdkfd_gpuvm_sync_memory( ...@@ -2226,11 +2226,12 @@ int amdgpu_amdkfd_gpuvm_sync_memory(
/** /**
* amdgpu_amdkfd_map_gtt_bo_to_gart - Map BO to GART and increment reference count * amdgpu_amdkfd_map_gtt_bo_to_gart - Map BO to GART and increment reference count
* @bo: Buffer object to be mapped * @bo: Buffer object to be mapped
* @bo_gart: Return bo reference
* *
* Before return, bo reference count is incremented. To release the reference and unpin/ * Before return, bo reference count is incremented. To release the reference and unpin/
* unmap the BO, call amdgpu_amdkfd_free_gtt_mem. * unmap the BO, call amdgpu_amdkfd_free_gtt_mem.
*/ */
int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo) int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo, struct amdgpu_bo **bo_gart)
{ {
int ret; int ret;
...@@ -2257,7 +2258,7 @@ int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo) ...@@ -2257,7 +2258,7 @@ int amdgpu_amdkfd_map_gtt_bo_to_gart(struct amdgpu_bo *bo)
amdgpu_bo_unreserve(bo); amdgpu_bo_unreserve(bo);
bo = amdgpu_bo_ref(bo); *bo_gart = amdgpu_bo_ref(bo);
return 0; return 0;
......
...@@ -247,8 +247,8 @@ static int set_queue_properties_from_user(struct queue_properties *q_properties, ...@@ -247,8 +247,8 @@ static int set_queue_properties_from_user(struct queue_properties *q_properties,
q_properties->priority = args->queue_priority; q_properties->priority = args->queue_priority;
q_properties->queue_address = args->ring_base_address; q_properties->queue_address = args->ring_base_address;
q_properties->queue_size = args->ring_size; q_properties->queue_size = args->ring_size;
q_properties->read_ptr = (uint32_t *) args->read_pointer_address; q_properties->read_ptr = (void __user *)args->read_pointer_address;
q_properties->write_ptr = (uint32_t *) args->write_pointer_address; q_properties->write_ptr = (void __user *)args->write_pointer_address;
q_properties->eop_ring_buffer_address = args->eop_buffer_address; q_properties->eop_ring_buffer_address = args->eop_buffer_address;
q_properties->eop_ring_buffer_size = args->eop_buffer_size; q_properties->eop_ring_buffer_size = args->eop_buffer_size;
q_properties->ctx_save_restore_area_address = q_properties->ctx_save_restore_area_address =
...@@ -306,7 +306,6 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p, ...@@ -306,7 +306,6 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p,
struct kfd_process_device *pdd; struct kfd_process_device *pdd;
struct queue_properties q_properties; struct queue_properties q_properties;
uint32_t doorbell_offset_in_process = 0; uint32_t doorbell_offset_in_process = 0;
struct amdgpu_bo *wptr_bo = NULL;
memset(&q_properties, 0, sizeof(struct queue_properties)); memset(&q_properties, 0, sizeof(struct queue_properties));
...@@ -342,53 +341,17 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p, ...@@ -342,53 +341,17 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p,
} }
} }
/* Starting with GFX11, wptr BOs must be mapped to GART for MES to determine work err = kfd_queue_acquire_buffers(pdd, &q_properties);
* on unmapped queues for usermode queue oversubscription (no aggregated doorbell)
*/
if (dev->kfd->shared_resources.enable_mes &&
((dev->adev->mes.sched_version & AMDGPU_MES_API_VERSION_MASK)
>> AMDGPU_MES_API_VERSION_SHIFT) >= 2) {
struct amdgpu_bo_va_mapping *wptr_mapping;
struct amdgpu_vm *wptr_vm;
wptr_vm = drm_priv_to_vm(pdd->drm_priv);
err = amdgpu_bo_reserve(wptr_vm->root.bo, false);
if (err)
goto err_wptr_map_gart;
wptr_mapping = amdgpu_vm_bo_lookup_mapping(
wptr_vm, args->write_pointer_address >> PAGE_SHIFT);
amdgpu_bo_unreserve(wptr_vm->root.bo);
if (!wptr_mapping) {
pr_err("Failed to lookup wptr bo\n");
err = -EINVAL;
goto err_wptr_map_gart;
}
wptr_bo = wptr_mapping->bo_va->base.bo;
if (wptr_bo->tbo.base.size > PAGE_SIZE) {
pr_err("Requested GART mapping for wptr bo larger than one page\n");
err = -EINVAL;
goto err_wptr_map_gart;
}
if (dev->adev != amdgpu_ttm_adev(wptr_bo->tbo.bdev)) {
pr_err("Queue memory allocated to wrong device\n");
err = -EINVAL;
goto err_wptr_map_gart;
}
err = amdgpu_amdkfd_map_gtt_bo_to_gart(wptr_bo);
if (err) { if (err) {
pr_err("Failed to map wptr bo to GART\n"); pr_debug("failed to acquire user queue buffers\n");
goto err_wptr_map_gart; goto err_acquire_queue_buf;
}
} }
pr_debug("Creating queue for PASID 0x%x on gpu 0x%x\n", pr_debug("Creating queue for PASID 0x%x on gpu 0x%x\n",
p->pasid, p->pasid,
dev->id); dev->id);
err = pqm_create_queue(&p->pqm, dev, filep, &q_properties, &queue_id, wptr_bo, err = pqm_create_queue(&p->pqm, dev, filep, &q_properties, &queue_id,
NULL, NULL, NULL, &doorbell_offset_in_process); NULL, NULL, NULL, &doorbell_offset_in_process);
if (err != 0) if (err != 0)
goto err_create_queue; goto err_create_queue;
...@@ -422,9 +385,8 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p, ...@@ -422,9 +385,8 @@ static int kfd_ioctl_create_queue(struct file *filep, struct kfd_process *p,
return 0; return 0;
err_create_queue: err_create_queue:
if (wptr_bo) kfd_queue_release_buffers(pdd, &q_properties);
amdgpu_amdkfd_free_gtt_mem(dev->adev, (void **)&wptr_bo); err_acquire_queue_buf:
err_wptr_map_gart:
err_bind_process: err_bind_process:
err_pdd: err_pdd:
mutex_unlock(&p->mutex); mutex_unlock(&p->mutex);
......
...@@ -208,10 +208,8 @@ static int add_queue_mes(struct device_queue_manager *dqm, struct queue *q, ...@@ -208,10 +208,8 @@ static int add_queue_mes(struct device_queue_manager *dqm, struct queue *q,
queue_input.mqd_addr = q->gart_mqd_addr; queue_input.mqd_addr = q->gart_mqd_addr;
queue_input.wptr_addr = (uint64_t)q->properties.write_ptr; queue_input.wptr_addr = (uint64_t)q->properties.write_ptr;
if (q->wptr_bo) {
wptr_addr_off = (uint64_t)q->properties.write_ptr & (PAGE_SIZE - 1); wptr_addr_off = (uint64_t)q->properties.write_ptr & (PAGE_SIZE - 1);
queue_input.wptr_mc_addr = amdgpu_bo_gpu_offset(q->wptr_bo) + wptr_addr_off; queue_input.wptr_mc_addr = amdgpu_bo_gpu_offset(q->properties.wptr_bo) + wptr_addr_off;
}
queue_input.is_kfd_process = 1; queue_input.is_kfd_process = 1;
queue_input.is_aql_queue = (q->properties.format == KFD_QUEUE_FORMAT_AQL); queue_input.is_aql_queue = (q->properties.format == KFD_QUEUE_FORMAT_AQL);
......
...@@ -494,8 +494,8 @@ struct queue_properties { ...@@ -494,8 +494,8 @@ struct queue_properties {
uint64_t queue_size; uint64_t queue_size;
uint32_t priority; uint32_t priority;
uint32_t queue_percent; uint32_t queue_percent;
uint32_t *read_ptr; void __user *read_ptr;
uint32_t *write_ptr; void __user *write_ptr;
void __iomem *doorbell_ptr; void __iomem *doorbell_ptr;
uint32_t doorbell_off; uint32_t doorbell_off;
bool is_interop; bool is_interop;
...@@ -522,6 +522,8 @@ struct queue_properties { ...@@ -522,6 +522,8 @@ struct queue_properties {
uint64_t tba_addr; uint64_t tba_addr;
uint64_t tma_addr; uint64_t tma_addr;
uint64_t exception_status; uint64_t exception_status;
struct amdgpu_bo *wptr_bo;
}; };
#define QUEUE_IS_ACTIVE(q) ((q).queue_size > 0 && \ #define QUEUE_IS_ACTIVE(q) ((q).queue_size > 0 && \
...@@ -604,7 +606,7 @@ struct queue { ...@@ -604,7 +606,7 @@ struct queue {
uint64_t gang_ctx_gpu_addr; uint64_t gang_ctx_gpu_addr;
void *gang_ctx_cpu_ptr; void *gang_ctx_cpu_ptr;
struct amdgpu_bo *wptr_bo; struct amdgpu_bo *wptr_bo_gart;
}; };
enum KFD_MQD_TYPE { enum KFD_MQD_TYPE {
...@@ -1284,6 +1286,10 @@ int init_queue(struct queue **q, const struct queue_properties *properties); ...@@ -1284,6 +1286,10 @@ int init_queue(struct queue **q, const struct queue_properties *properties);
void uninit_queue(struct queue *q); void uninit_queue(struct queue *q);
void print_queue_properties(struct queue_properties *q); void print_queue_properties(struct queue_properties *q);
void print_queue(struct queue *q); void print_queue(struct queue *q);
int kfd_queue_buffer_get(struct amdgpu_vm *vm, void __user *addr, struct amdgpu_bo **pbo,
u64 expected_size);
int kfd_queue_acquire_buffers(struct kfd_process_device *pdd, struct queue_properties *properties);
int kfd_queue_release_buffers(struct kfd_process_device *pdd, struct queue_properties *properties);
struct mqd_manager *mqd_manager_init_cik(enum KFD_MQD_TYPE type, struct mqd_manager *mqd_manager_init_cik(enum KFD_MQD_TYPE type,
struct kfd_node *dev); struct kfd_node *dev);
...@@ -1320,7 +1326,6 @@ int pqm_create_queue(struct process_queue_manager *pqm, ...@@ -1320,7 +1326,6 @@ int pqm_create_queue(struct process_queue_manager *pqm,
struct file *f, struct file *f,
struct queue_properties *properties, struct queue_properties *properties,
unsigned int *qid, unsigned int *qid,
struct amdgpu_bo *wptr_bo,
const struct kfd_criu_queue_priv_data *q_data, const struct kfd_criu_queue_priv_data *q_data,
const void *restore_mqd, const void *restore_mqd,
const void *restore_ctl_stack, const void *restore_ctl_stack,
......
...@@ -205,18 +205,21 @@ static void pqm_clean_queue_resource(struct process_queue_manager *pqm, ...@@ -205,18 +205,21 @@ static void pqm_clean_queue_resource(struct process_queue_manager *pqm,
if (dev->kfd->shared_resources.enable_mes) { if (dev->kfd->shared_resources.enable_mes) {
amdgpu_amdkfd_free_gtt_mem(dev->adev, &pqn->q->gang_ctx_bo); amdgpu_amdkfd_free_gtt_mem(dev->adev, &pqn->q->gang_ctx_bo);
if (pqn->q->wptr_bo) amdgpu_amdkfd_free_gtt_mem(dev->adev, (void **)&pqn->q->wptr_bo_gart);
amdgpu_amdkfd_free_gtt_mem(dev->adev, (void **)&pqn->q->wptr_bo);
} }
} }
void pqm_uninit(struct process_queue_manager *pqm) void pqm_uninit(struct process_queue_manager *pqm)
{ {
struct process_queue_node *pqn, *next; struct process_queue_node *pqn, *next;
struct kfd_process_device *pdd;
list_for_each_entry_safe(pqn, next, &pqm->queues, process_queue_list) { list_for_each_entry_safe(pqn, next, &pqm->queues, process_queue_list) {
if (pqn->q) if (pqn->q) {
pdd = kfd_get_process_device_data(pqn->q->device, pqm->process);
kfd_queue_release_buffers(pdd, &pqn->q->properties);
pqm_clean_queue_resource(pqm, pqn); pqm_clean_queue_resource(pqm, pqn);
}
kfd_procfs_del_queue(pqn->q); kfd_procfs_del_queue(pqn->q);
uninit_queue(pqn->q); uninit_queue(pqn->q);
...@@ -231,8 +234,7 @@ void pqm_uninit(struct process_queue_manager *pqm) ...@@ -231,8 +234,7 @@ void pqm_uninit(struct process_queue_manager *pqm)
static int init_user_queue(struct process_queue_manager *pqm, static int init_user_queue(struct process_queue_manager *pqm,
struct kfd_node *dev, struct queue **q, struct kfd_node *dev, struct queue **q,
struct queue_properties *q_properties, struct queue_properties *q_properties,
struct file *f, struct amdgpu_bo *wptr_bo, struct file *f, unsigned int qid)
unsigned int qid)
{ {
int retval; int retval;
...@@ -263,12 +265,32 @@ static int init_user_queue(struct process_queue_manager *pqm, ...@@ -263,12 +265,32 @@ static int init_user_queue(struct process_queue_manager *pqm,
goto cleanup; goto cleanup;
} }
memset((*q)->gang_ctx_cpu_ptr, 0, AMDGPU_MES_GANG_CTX_SIZE); memset((*q)->gang_ctx_cpu_ptr, 0, AMDGPU_MES_GANG_CTX_SIZE);
(*q)->wptr_bo = wptr_bo;
/* Starting with GFX11, wptr BOs must be mapped to GART for MES to determine work
* on unmapped queues for usermode queue oversubscription (no aggregated doorbell)
*/
if (((dev->adev->mes.sched_version & AMDGPU_MES_API_VERSION_MASK)
>> AMDGPU_MES_API_VERSION_SHIFT) >= 2) {
if (dev->adev != amdgpu_ttm_adev(q_properties->wptr_bo->tbo.bdev)) {
pr_err("Queue memory allocated to wrong device\n");
retval = -EINVAL;
goto free_gang_ctx_bo;
}
retval = amdgpu_amdkfd_map_gtt_bo_to_gart(q_properties->wptr_bo,
&(*q)->wptr_bo_gart);
if (retval) {
pr_err("Failed to map wptr bo to GART\n");
goto free_gang_ctx_bo;
}
}
} }
pr_debug("PQM After init queue"); pr_debug("PQM After init queue");
return 0; return 0;
free_gang_ctx_bo:
amdgpu_amdkfd_free_gtt_mem(dev->adev, (*q)->gang_ctx_bo);
cleanup: cleanup:
uninit_queue(*q); uninit_queue(*q);
*q = NULL; *q = NULL;
...@@ -280,7 +302,6 @@ int pqm_create_queue(struct process_queue_manager *pqm, ...@@ -280,7 +302,6 @@ int pqm_create_queue(struct process_queue_manager *pqm,
struct file *f, struct file *f,
struct queue_properties *properties, struct queue_properties *properties,
unsigned int *qid, unsigned int *qid,
struct amdgpu_bo *wptr_bo,
const struct kfd_criu_queue_priv_data *q_data, const struct kfd_criu_queue_priv_data *q_data,
const void *restore_mqd, const void *restore_mqd,
const void *restore_ctl_stack, const void *restore_ctl_stack,
...@@ -351,7 +372,7 @@ int pqm_create_queue(struct process_queue_manager *pqm, ...@@ -351,7 +372,7 @@ int pqm_create_queue(struct process_queue_manager *pqm,
* allocate_sdma_queue() in create_queue() has the * allocate_sdma_queue() in create_queue() has the
* corresponding check logic. * corresponding check logic.
*/ */
retval = init_user_queue(pqm, dev, &q, properties, f, wptr_bo, *qid); retval = init_user_queue(pqm, dev, &q, properties, f, *qid);
if (retval != 0) if (retval != 0)
goto err_create_queue; goto err_create_queue;
pqn->q = q; pqn->q = q;
...@@ -372,7 +393,7 @@ int pqm_create_queue(struct process_queue_manager *pqm, ...@@ -372,7 +393,7 @@ int pqm_create_queue(struct process_queue_manager *pqm,
goto err_create_queue; goto err_create_queue;
} }
retval = init_user_queue(pqm, dev, &q, properties, f, wptr_bo, *qid); retval = init_user_queue(pqm, dev, &q, properties, f, *qid);
if (retval != 0) if (retval != 0)
goto err_create_queue; goto err_create_queue;
pqn->q = q; pqn->q = q;
...@@ -490,6 +511,10 @@ int pqm_destroy_queue(struct process_queue_manager *pqm, unsigned int qid) ...@@ -490,6 +511,10 @@ int pqm_destroy_queue(struct process_queue_manager *pqm, unsigned int qid)
} }
if (pqn->q) { if (pqn->q) {
retval = kfd_queue_release_buffers(pdd, &pqn->q->properties);
if (retval)
goto err_destroy_queue;
kfd_procfs_del_queue(pqn->q); kfd_procfs_del_queue(pqn->q);
dqm = pqn->q->device->dqm; dqm = pqn->q->device->dqm;
retval = dqm->ops.destroy_queue(dqm, &pdd->qpd, pqn->q); retval = dqm->ops.destroy_queue(dqm, &pdd->qpd, pqn->q);
...@@ -971,7 +996,7 @@ int kfd_criu_restore_queue(struct kfd_process *p, ...@@ -971,7 +996,7 @@ int kfd_criu_restore_queue(struct kfd_process *p,
print_queue_properties(&qp); print_queue_properties(&qp);
ret = pqm_create_queue(&p->pqm, pdd->dev, NULL, &qp, &queue_id, NULL, q_data, mqd, ctl_stack, ret = pqm_create_queue(&p->pqm, pdd->dev, NULL, &qp, &queue_id, q_data, mqd, ctl_stack,
NULL); NULL);
if (ret) { if (ret) {
pr_err("Failed to create new queue err:%d\n", ret); pr_err("Failed to create new queue err:%d\n", ret);
......
...@@ -82,3 +82,60 @@ void uninit_queue(struct queue *q) ...@@ -82,3 +82,60 @@ void uninit_queue(struct queue *q)
{ {
kfree(q); kfree(q);
} }
int kfd_queue_buffer_get(struct amdgpu_vm *vm, void __user *addr, struct amdgpu_bo **pbo,
u64 expected_size)
{
struct amdgpu_bo_va_mapping *mapping;
u64 user_addr;
u64 size;
user_addr = (u64)addr >> AMDGPU_GPU_PAGE_SHIFT;
size = expected_size >> AMDGPU_GPU_PAGE_SHIFT;
mapping = amdgpu_vm_bo_lookup_mapping(vm, user_addr);
if (!mapping)
goto out_err;
if (user_addr != mapping->start || user_addr + size - 1 != mapping->last) {
pr_debug("expected size 0x%llx not equal to mapping addr 0x%llx size 0x%llx\n",
expected_size, mapping->start << AMDGPU_GPU_PAGE_SHIFT,
(mapping->last - mapping->start + 1) << AMDGPU_GPU_PAGE_SHIFT);
goto out_err;
}
*pbo = amdgpu_bo_ref(mapping->bo_va->base.bo);
return 0;
out_err:
*pbo = NULL;
return -EINVAL;
}
int kfd_queue_acquire_buffers(struct kfd_process_device *pdd, struct queue_properties *properties)
{
struct amdgpu_vm *vm;
int err;
vm = drm_priv_to_vm(pdd->drm_priv);
err = amdgpu_bo_reserve(vm->root.bo, false);
if (err)
return err;
err = kfd_queue_buffer_get(vm, properties->write_ptr, &properties->wptr_bo, PAGE_SIZE);
if (err)
goto out_unreserve;
amdgpu_bo_unreserve(vm->root.bo);
return 0;
out_unreserve:
amdgpu_bo_unreserve(vm->root.bo);
return err;
}
int kfd_queue_release_buffers(struct kfd_process_device *pdd, struct queue_properties *properties)
{
amdgpu_bo_unref(&properties->wptr_bo);
return 0;
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment