summaryrefslogtreecommitdiff
path: root/mali_kbase/csf/mali_kbase_csf_cpu_queue.c
diff options
context:
space:
mode:
Diffstat (limited to 'mali_kbase/csf/mali_kbase_csf_cpu_queue.c')
-rw-r--r--mali_kbase/csf/mali_kbase_csf_cpu_queue.c132
1 files changed, 132 insertions, 0 deletions
diff --git a/mali_kbase/csf/mali_kbase_csf_cpu_queue.c b/mali_kbase/csf/mali_kbase_csf_cpu_queue.c
new file mode 100644
index 0000000..eb1e28b
--- /dev/null
+++ b/mali_kbase/csf/mali_kbase_csf_cpu_queue.c
@@ -0,0 +1,132 @@
+// SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note
+/*
+ *
+ * (C) COPYRIGHT 2023 ARM Limited. All rights reserved.
+ *
+ * This program is free software and is provided to you under the terms of the
+ * GNU General Public License version 2 as published by the Free Software
+ * Foundation, and any use by you of this program is subject to the terms
+ * of such GNU license.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, you can access it online at
+ * http://www.gnu.org/licenses/gpl-2.0.html.
+ *
+ */
+
+#include "mali_kbase_csf_cpu_queue.h"
+#include "mali_kbase_csf_util.h"
+#include <mali_kbase.h>
+#include <asm/atomic.h>
+
+void kbase_csf_cpu_queue_init(struct kbase_context *kctx)
+{
+ if (WARN_ON(!kctx))
+ return;
+
+ kctx->csf.cpu_queue.buffer = NULL;
+ kctx->csf.cpu_queue.buffer_size = 0;
+ atomic_set(&kctx->csf.cpu_queue.dump_req_status, BASE_CSF_CPU_QUEUE_DUMP_COMPLETE);
+}
+
+bool kbase_csf_cpu_queue_read_dump_req(struct kbase_context *kctx,
+ struct base_csf_notification *req)
+{
+ if (atomic_cmpxchg(&kctx->csf.cpu_queue.dump_req_status, BASE_CSF_CPU_QUEUE_DUMP_ISSUED,
+ BASE_CSF_CPU_QUEUE_DUMP_PENDING) != BASE_CSF_CPU_QUEUE_DUMP_ISSUED) {
+ return false;
+ }
+
+ req->type = BASE_CSF_NOTIFICATION_CPU_QUEUE_DUMP;
+ return true;
+}
+
+bool kbase_csf_cpu_queue_dump_needed(struct kbase_context *kctx)
+{
+ return (atomic_read(&kctx->csf.cpu_queue.dump_req_status) ==
+ BASE_CSF_CPU_QUEUE_DUMP_ISSUED);
+}
+
+int kbase_csf_cpu_queue_dump_buffer(struct kbase_context *kctx, u64 buffer, size_t buf_size)
+{
+ size_t alloc_size = buf_size;
+ char *dump_buffer;
+
+ if (!buffer || !buf_size)
+ return 0;
+
+ if (alloc_size > KBASE_MEM_ALLOC_MAX_SIZE)
+ return -EINVAL;
+
+ alloc_size = (alloc_size + PAGE_SIZE) & ~(PAGE_SIZE - 1);
+ dump_buffer = kzalloc(alloc_size, GFP_KERNEL);
+ if (!dump_buffer)
+ return -ENOMEM;
+
+ WARN_ON(kctx->csf.cpu_queue.buffer != NULL);
+
+ if (copy_from_user(dump_buffer, u64_to_user_ptr(buffer), buf_size)) {
+ kfree(dump_buffer);
+ return -EFAULT;
+ }
+
+ rt_mutex_lock(&kctx->csf.lock);
+
+ kfree(kctx->csf.cpu_queue.buffer);
+
+ if (atomic_read(&kctx->csf.cpu_queue.dump_req_status) == BASE_CSF_CPU_QUEUE_DUMP_PENDING) {
+ kctx->csf.cpu_queue.buffer = dump_buffer;
+ kctx->csf.cpu_queue.buffer_size = buf_size;
+ complete_all(&kctx->csf.cpu_queue.dump_cmp);
+ } else
+ kfree(dump_buffer);
+
+ rt_mutex_unlock(&kctx->csf.lock);
+
+ return 0;
+}
+
+int kbasep_csf_cpu_queue_dump_print(struct kbase_context *kctx, struct kbasep_printer *kbpr)
+{
+ rt_mutex_lock(&kctx->csf.lock);
+ if (atomic_read(&kctx->csf.cpu_queue.dump_req_status) != BASE_CSF_CPU_QUEUE_DUMP_COMPLETE) {
+ kbasep_print(kbpr, "Dump request already started! (try again)\n");
+ rt_mutex_unlock(&kctx->csf.lock);
+ return -EBUSY;
+ }
+
+ atomic_set(&kctx->csf.cpu_queue.dump_req_status, BASE_CSF_CPU_QUEUE_DUMP_ISSUED);
+ init_completion(&kctx->csf.cpu_queue.dump_cmp);
+ kbase_event_wakeup_nosync(kctx);
+ rt_mutex_unlock(&kctx->csf.lock);
+
+ kbasep_print(kbpr, "CPU Queues table (version:v" __stringify(
+ MALI_CSF_CPU_QUEUE_DUMP_VERSION) "):\n");
+
+ wait_for_completion_timeout(&kctx->csf.cpu_queue.dump_cmp, msecs_to_jiffies(3000));
+
+ rt_mutex_lock(&kctx->csf.lock);
+ if (kctx->csf.cpu_queue.buffer) {
+ WARN_ON(atomic_read(&kctx->csf.cpu_queue.dump_req_status) !=
+ BASE_CSF_CPU_QUEUE_DUMP_PENDING);
+
+ /* The CPU queue dump is returned as a single formatted string */
+ kbasep_puts(kbpr, kctx->csf.cpu_queue.buffer);
+ kbasep_puts(kbpr, "\n");
+
+ kfree(kctx->csf.cpu_queue.buffer);
+ kctx->csf.cpu_queue.buffer = NULL;
+ kctx->csf.cpu_queue.buffer_size = 0;
+ } else
+ kbasep_print(kbpr, "Dump error! (time out)\n");
+
+ atomic_set(&kctx->csf.cpu_queue.dump_req_status, BASE_CSF_CPU_QUEUE_DUMP_COMPLETE);
+
+ rt_mutex_unlock(&kctx->csf.lock);
+ return 0;
+}