| // SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note |
| /* |
| * |
| * (C) COPYRIGHT 2011-2016, 2020-2021 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_dma_fence.h before checking for CONFIG_MALI_DMA_FENCE as |
| * it will be set there. |
| */ |
| #include "mali_kbase_dma_fence.h" |
| #include <linux/atomic.h> |
| #include <linux/list.h> |
| #include <linux/lockdep.h> |
| #include <linux/mutex.h> |
| #include <linux/version.h> |
| #include <linux/slab.h> |
| #include <linux/spinlock.h> |
| #include <linux/workqueue.h> |
| #include <linux/ww_mutex.h> |
| #include <mali_kbase.h> |
| |
| static void |
| kbase_dma_fence_work(struct work_struct *pwork); |
| |
| static void |
| kbase_dma_fence_waiters_add(struct kbase_jd_atom *katom) |
| { |
| struct kbase_context *kctx = katom->kctx; |
| |
| list_add_tail(&katom->queue, &kctx->dma_fence.waiting_resource); |
| } |
| |
| static void |
| kbase_dma_fence_waiters_remove(struct kbase_jd_atom *katom) |
| { |
| list_del(&katom->queue); |
| } |
| |
| static int |
| kbase_dma_fence_lock_reservations(struct kbase_dma_fence_resv_info *info, |
| struct ww_acquire_ctx *ctx) |
| { |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| struct reservation_object *content_res = NULL; |
| #else |
| struct dma_resv *content_res = NULL; |
| #endif |
| unsigned int content_res_idx = 0; |
| unsigned int r; |
| int err = 0; |
| |
| ww_acquire_init(ctx, &reservation_ww_class); |
| |
| retry: |
| for (r = 0; r < info->dma_fence_resv_count; r++) { |
| if (info->resv_objs[r] == content_res) { |
| content_res = NULL; |
| continue; |
| } |
| |
| err = ww_mutex_lock(&info->resv_objs[r]->lock, ctx); |
| if (err) |
| goto error; |
| } |
| |
| ww_acquire_done(ctx); |
| return err; |
| |
| error: |
| content_res_idx = r; |
| |
| /* Unlock the locked one ones */ |
| while (r--) |
| ww_mutex_unlock(&info->resv_objs[r]->lock); |
| |
| if (content_res) |
| ww_mutex_unlock(&content_res->lock); |
| |
| /* If we deadlock try with lock_slow and retry */ |
| if (err == -EDEADLK) { |
| content_res = info->resv_objs[content_res_idx]; |
| ww_mutex_lock_slow(&content_res->lock, ctx); |
| goto retry; |
| } |
| |
| /* If we are here the function failed */ |
| ww_acquire_fini(ctx); |
| return err; |
| } |
| |
| static void |
| kbase_dma_fence_unlock_reservations(struct kbase_dma_fence_resv_info *info, |
| struct ww_acquire_ctx *ctx) |
| { |
| unsigned int r; |
| |
| for (r = 0; r < info->dma_fence_resv_count; r++) |
| ww_mutex_unlock(&info->resv_objs[r]->lock); |
| ww_acquire_fini(ctx); |
| } |
| |
| |
| |
| /** |
| * kbase_dma_fence_queue_work() - Queue work to handle @katom |
| * @katom: Pointer to atom for which to queue work |
| * |
| * Queue kbase_dma_fence_work() for @katom to clean up the fence callbacks and |
| * submit the atom. |
| */ |
| static void |
| kbase_dma_fence_queue_work(struct kbase_jd_atom *katom) |
| { |
| struct kbase_context *kctx = katom->kctx; |
| bool ret; |
| |
| INIT_WORK(&katom->work, kbase_dma_fence_work); |
| ret = queue_work(kctx->dma_fence.wq, &katom->work); |
| /* Warn if work was already queued, that should not happen. */ |
| WARN_ON(!ret); |
| } |
| |
| /** |
| * kbase_dma_fence_cancel_atom() - Cancels waiting on an atom |
| * @katom: Katom to cancel |
| * |
| * Locking: katom->dma_fence.callbacks list assumes jctx.lock is held. |
| */ |
| static void |
| kbase_dma_fence_cancel_atom(struct kbase_jd_atom *katom) |
| { |
| lockdep_assert_held(&katom->kctx->jctx.lock); |
| |
| /* Cancel callbacks and clean up. */ |
| kbase_fence_free_callbacks(katom); |
| |
| /* Mark the atom as handled in case all fences signaled just before |
| * canceling the callbacks and the worker was queued. |
| */ |
| kbase_fence_dep_count_set(katom, -1); |
| |
| /* Prevent job_done_nolock from being called twice on an atom when |
| * there is a race between job completion and cancellation. |
| */ |
| |
| if (katom->status == KBASE_JD_ATOM_STATE_QUEUED) { |
| /* Wait was cancelled - zap the atom */ |
| katom->event_code = BASE_JD_EVENT_JOB_CANCELLED; |
| if (jd_done_nolock(katom, NULL)) |
| kbase_js_sched_all(katom->kctx->kbdev); |
| } |
| } |
| |
| /** |
| * kbase_dma_fence_work() - Worker thread called when a fence is signaled |
| * @pwork: work_struct containing a pointer to a katom |
| * |
| * This function will clean and mark all dependencies as satisfied |
| */ |
| static void |
| kbase_dma_fence_work(struct work_struct *pwork) |
| { |
| struct kbase_jd_atom *katom; |
| struct kbase_jd_context *ctx; |
| |
| katom = container_of(pwork, struct kbase_jd_atom, work); |
| ctx = &katom->kctx->jctx; |
| |
| mutex_lock(&ctx->lock); |
| if (kbase_fence_dep_count_read(katom) != 0) |
| goto out; |
| |
| kbase_fence_dep_count_set(katom, -1); |
| |
| /* Remove atom from list of dma-fence waiting atoms. */ |
| kbase_dma_fence_waiters_remove(katom); |
| /* Cleanup callbacks. */ |
| kbase_fence_free_callbacks(katom); |
| /* |
| * Queue atom on GPU, unless it has already completed due to a failing |
| * dependency. Run jd_done_nolock() on the katom if it is completed. |
| */ |
| if (unlikely(katom->status == KBASE_JD_ATOM_STATE_COMPLETED)) |
| jd_done_nolock(katom, NULL); |
| else |
| kbase_jd_dep_clear_locked(katom); |
| |
| out: |
| mutex_unlock(&ctx->lock); |
| } |
| |
| static void |
| #if (KERNEL_VERSION(4, 10, 0) > LINUX_VERSION_CODE) |
| kbase_dma_fence_cb(struct fence *fence, struct fence_cb *cb) |
| #else |
| kbase_dma_fence_cb(struct dma_fence *fence, struct dma_fence_cb *cb) |
| #endif |
| { |
| struct kbase_fence_cb *kcb = container_of(cb, |
| struct kbase_fence_cb, |
| fence_cb); |
| struct kbase_jd_atom *katom = kcb->katom; |
| |
| /* If the atom is zapped dep_count will be forced to a negative number |
| * preventing this callback from ever scheduling work. Which in turn |
| * would reschedule the atom. |
| */ |
| |
| if (kbase_fence_dep_count_dec_and_test(katom)) |
| kbase_dma_fence_queue_work(katom); |
| } |
| |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| static int |
| kbase_dma_fence_add_reservation_callback(struct kbase_jd_atom *katom, |
| struct reservation_object *resv, |
| bool exclusive) |
| #else |
| static int |
| kbase_dma_fence_add_reservation_callback(struct kbase_jd_atom *katom, |
| struct dma_resv *resv, |
| bool exclusive) |
| #endif |
| { |
| #if (KERNEL_VERSION(4, 10, 0) > LINUX_VERSION_CODE) |
| struct fence *excl_fence = NULL; |
| struct fence **shared_fences = NULL; |
| #else |
| struct dma_fence *excl_fence = NULL; |
| struct dma_fence **shared_fences = NULL; |
| #endif |
| unsigned int shared_count = 0; |
| int err, i; |
| |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| err = reservation_object_get_fences_rcu( |
| #else |
| err = dma_resv_get_fences_rcu( |
| #endif |
| resv, |
| &excl_fence, |
| &shared_count, |
| &shared_fences); |
| if (err) |
| return err; |
| |
| if (excl_fence) { |
| err = kbase_fence_add_callback(katom, |
| excl_fence, |
| kbase_dma_fence_cb); |
| |
| /* Release our reference, taken by reservation_object_get_fences_rcu(), |
| * to the fence. We have set up our callback (if that was possible), |
| * and it's the fence's owner is responsible for singling the fence |
| * before allowing it to disappear. |
| */ |
| dma_fence_put(excl_fence); |
| |
| if (err) |
| goto out; |
| } |
| |
| if (exclusive) { |
| for (i = 0; i < shared_count; i++) { |
| err = kbase_fence_add_callback(katom, |
| shared_fences[i], |
| kbase_dma_fence_cb); |
| if (err) |
| goto out; |
| } |
| } |
| |
| /* Release all our references to the shared fences, taken by |
| * reservation_object_get_fences_rcu(). We have set up our callback (if |
| * that was possible), and it's the fence's owner is responsible for |
| * signaling the fence before allowing it to disappear. |
| */ |
| out: |
| for (i = 0; i < shared_count; i++) |
| dma_fence_put(shared_fences[i]); |
| kfree(shared_fences); |
| |
| if (err) { |
| /* |
| * On error, cancel and clean up all callbacks that was set up |
| * before the error. |
| */ |
| kbase_fence_free_callbacks(katom); |
| } |
| |
| return err; |
| } |
| |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| void kbase_dma_fence_add_reservation(struct reservation_object *resv, |
| struct kbase_dma_fence_resv_info *info, |
| bool exclusive) |
| #else |
| void kbase_dma_fence_add_reservation(struct dma_resv *resv, |
| struct kbase_dma_fence_resv_info *info, |
| bool exclusive) |
| #endif |
| { |
| unsigned int i; |
| |
| for (i = 0; i < info->dma_fence_resv_count; i++) { |
| /* Duplicate resource, ignore */ |
| if (info->resv_objs[i] == resv) |
| return; |
| } |
| |
| info->resv_objs[info->dma_fence_resv_count] = resv; |
| if (exclusive) |
| set_bit(info->dma_fence_resv_count, |
| info->dma_fence_excl_bitmap); |
| (info->dma_fence_resv_count)++; |
| } |
| |
| int kbase_dma_fence_wait(struct kbase_jd_atom *katom, |
| struct kbase_dma_fence_resv_info *info) |
| { |
| int err, i; |
| #if (KERNEL_VERSION(4, 10, 0) > LINUX_VERSION_CODE) |
| struct fence *fence; |
| #else |
| struct dma_fence *fence; |
| #endif |
| struct ww_acquire_ctx ww_ctx; |
| |
| lockdep_assert_held(&katom->kctx->jctx.lock); |
| |
| fence = kbase_fence_out_new(katom); |
| if (!fence) { |
| err = -ENOMEM; |
| dev_err(katom->kctx->kbdev->dev, |
| "Error %d creating fence.\n", err); |
| return err; |
| } |
| |
| kbase_fence_dep_count_set(katom, 1); |
| |
| err = kbase_dma_fence_lock_reservations(info, &ww_ctx); |
| if (err) { |
| dev_err(katom->kctx->kbdev->dev, |
| "Error %d locking reservations.\n", err); |
| kbase_fence_dep_count_set(katom, -1); |
| kbase_fence_out_remove(katom); |
| return err; |
| } |
| |
| for (i = 0; i < info->dma_fence_resv_count; i++) { |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| struct reservation_object *obj = info->resv_objs[i]; |
| #else |
| struct dma_resv *obj = info->resv_objs[i]; |
| #endif |
| if (!test_bit(i, info->dma_fence_excl_bitmap)) { |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| err = reservation_object_reserve_shared(obj); |
| #else |
| err = dma_resv_reserve_shared(obj, 0); |
| #endif |
| if (err) { |
| dev_err(katom->kctx->kbdev->dev, |
| "Error %d reserving space for shared fence.\n", err); |
| goto end; |
| } |
| |
| err = kbase_dma_fence_add_reservation_callback(katom, obj, false); |
| if (err) { |
| dev_err(katom->kctx->kbdev->dev, |
| "Error %d adding reservation to callback.\n", err); |
| goto end; |
| } |
| |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| reservation_object_add_shared_fence(obj, fence); |
| #else |
| dma_resv_add_shared_fence(obj, fence); |
| #endif |
| } else { |
| err = kbase_dma_fence_add_reservation_callback(katom, obj, true); |
| if (err) { |
| dev_err(katom->kctx->kbdev->dev, |
| "Error %d adding reservation to callback.\n", err); |
| goto end; |
| } |
| |
| #if (KERNEL_VERSION(5, 4, 0) > LINUX_VERSION_CODE) |
| reservation_object_add_excl_fence(obj, fence); |
| #else |
| dma_resv_add_excl_fence(obj, fence); |
| #endif |
| } |
| } |
| |
| end: |
| kbase_dma_fence_unlock_reservations(info, &ww_ctx); |
| |
| if (likely(!err)) { |
| /* Test if the callbacks are already triggered */ |
| if (kbase_fence_dep_count_dec_and_test(katom)) { |
| kbase_fence_dep_count_set(katom, -1); |
| kbase_fence_free_callbacks(katom); |
| } else { |
| /* Add katom to the list of dma-buf fence waiting atoms |
| * only if it is still waiting. |
| */ |
| kbase_dma_fence_waiters_add(katom); |
| } |
| } else { |
| /* There was an error, cancel callbacks, set dep_count to -1 to |
| * indicate that the atom has been handled (the caller will |
| * kill it for us), signal the fence, free callbacks and the |
| * fence. |
| */ |
| kbase_fence_free_callbacks(katom); |
| kbase_fence_dep_count_set(katom, -1); |
| kbase_dma_fence_signal(katom); |
| } |
| |
| return err; |
| } |
| |
| void kbase_dma_fence_cancel_all_atoms(struct kbase_context *kctx) |
| { |
| struct list_head *list = &kctx->dma_fence.waiting_resource; |
| |
| while (!list_empty(list)) { |
| struct kbase_jd_atom *katom; |
| |
| katom = list_first_entry(list, struct kbase_jd_atom, queue); |
| kbase_dma_fence_waiters_remove(katom); |
| kbase_dma_fence_cancel_atom(katom); |
| } |
| } |
| |
| void kbase_dma_fence_cancel_callbacks(struct kbase_jd_atom *katom) |
| { |
| /* Cancel callbacks and clean up. */ |
| if (kbase_fence_free_callbacks(katom)) |
| kbase_dma_fence_queue_work(katom); |
| } |
| |
| void kbase_dma_fence_signal(struct kbase_jd_atom *katom) |
| { |
| if (!katom->dma_fence.fence) |
| return; |
| |
| /* Signal the atom's fence. */ |
| dma_fence_signal(katom->dma_fence.fence); |
| |
| kbase_fence_out_remove(katom); |
| |
| kbase_fence_free_callbacks(katom); |
| } |
| |
| void kbase_dma_fence_term(struct kbase_context *kctx) |
| { |
| destroy_workqueue(kctx->dma_fence.wq); |
| kctx->dma_fence.wq = NULL; |
| } |
| |
| int kbase_dma_fence_init(struct kbase_context *kctx) |
| { |
| INIT_LIST_HEAD(&kctx->dma_fence.waiting_resource); |
| |
| kctx->dma_fence.wq = alloc_workqueue("mali-fence-%d", |
| WQ_UNBOUND, 1, kctx->pid); |
| if (!kctx->dma_fence.wq) |
| return -ENOMEM; |
| |
| return 0; |
| } |