From: David Yat Sin <[email protected]>

Enable pc sampling to query system capability.

Co-developed-by: James Zhu <[email protected]>
Signed-off-by: James Zhu <[email protected]>
Signed-off-by: David Yat Sin <[email protected]>
---
 drivers/gpu/drm/amd/amdkfd/kfd_pc_sampling.c | 65 +++++++++++++++++++-
 1 file changed, 64 insertions(+), 1 deletion(-)

diff --git a/drivers/gpu/drm/amd/amdkfd/kfd_pc_sampling.c 
b/drivers/gpu/drm/amd/amdkfd/kfd_pc_sampling.c
index a7e78ff42d07..e9277c9beec7 100644
--- a/drivers/gpu/drm/amd/amdkfd/kfd_pc_sampling.c
+++ b/drivers/gpu/drm/amd/amdkfd/kfd_pc_sampling.c
@@ -25,10 +25,73 @@
 #include "amdgpu_amdkfd.h"
 #include "kfd_pc_sampling.h"
 
+struct supported_pc_sample_info {
+       uint32_t ip_version;
+       const struct kfd_pc_sample_info *sample_info;
+};
+
+const struct kfd_pc_sample_info sample_info_hosttrap_9_0_0 = {
+       0, 1, ~0ULL, 0, KFD_IOCTL_PCS_METHOD_HOSTTRAP, 
KFD_IOCTL_PCS_TYPE_TIME_US };
+
+struct supported_pc_sample_info supported_formats[] = {
+       { IP_VERSION(9, 4, 1), &sample_info_hosttrap_9_0_0 },
+       { IP_VERSION(9, 4, 2), &sample_info_hosttrap_9_0_0 },
+};
+
 static int kfd_pc_sample_query_cap(struct kfd_process_device *pdd,
                                        struct kfd_ioctl_pc_sample_args __user 
*user_args)
 {
-       return -EINVAL;
+       uint64_t sample_offset;
+       int num_method = 0;
+       int ret;
+       int i;
+
+       for (i = 0; i < ARRAY_SIZE(supported_formats); i++)
+               if (KFD_GC_VERSION(pdd->dev) == supported_formats[i].ip_version)
+                       num_method++;
+
+       if (!num_method) {
+               pr_debug("PC Sampling not supported on GC_HWIP:0x%x.",
+                       pdd->dev->adev->ip_versions[GC_HWIP][0]);
+               return -EOPNOTSUPP;
+       }
+
+       ret = 0;
+       mutex_lock(&pdd->dev->pcs_data.mutex);
+       if (user_args->flags != KFD_IOCTL_PCS_QUERY_TYPE_FULL &&
+                       pdd->dev->pcs_data.hosttrap_entry.base.use_count) {
+               /* If we already have a session, restrict returned list to 
current method  */
+               user_args->num_sample_info = 1;
+
+               if (user_args->sample_info_ptr)
+                       ret = copy_to_user((void __user *) 
user_args->sample_info_ptr,
+                               
&pdd->dev->pcs_data.hosttrap_entry.base.pc_sample_info,
+                               sizeof(struct kfd_pc_sample_info));
+               mutex_unlock(&pdd->dev->pcs_data.mutex);
+               return ret ? -EFAULT : 0;
+       }
+       mutex_unlock(&pdd->dev->pcs_data.mutex);
+
+       if (!user_args->sample_info_ptr || user_args->num_sample_info < 
num_method) {
+               user_args->num_sample_info = num_method;
+               pr_debug("ASIC requires space for %d kfd_pc_sample_info 
entries.", num_method);
+               return -ENOSPC;
+       }
+
+       sample_offset = user_args->sample_info_ptr;
+       for (i = 0; i < ARRAY_SIZE(supported_formats); i++) {
+               if (KFD_GC_VERSION(pdd->dev) == 
supported_formats[i].ip_version) {
+                       ret = copy_to_user((void __user *) sample_offset,
+                               supported_formats[i].sample_info, sizeof(struct 
kfd_pc_sample_info));
+                       if (ret) {
+                               pr_debug("Failed to copy PC sampling info to 
user.");
+                               return -EFAULT;
+                       }
+                       sample_offset += sizeof(struct kfd_pc_sample_info);
+               }
+       }
+
+       return 0;
 }
 
 static int kfd_pc_sample_start(struct kfd_process_device *pdd)
-- 
2.25.1

Reply via email to