From e734712472e061cab50252acaf2310cb97d7bf95 Mon Sep 17 00:00:00 2001 From: Dominik Behr Date: Fri, 29 Aug 2014 20:08:58 -0700 Subject: [PATCH] CHROMIUM: drm: add helpers for fence and reservation based dma-buf sync BUG=chromium:395901 TEST=emerge-veyron chromeos-kernel-3_14, deploy and boot kernel Change-Id: I0cdf6d23e9f4924128d4de77c0f3ed7589766bb8 Signed-off-by: Dominik Behr Reviewed-on: https://chromium-review.googlesource.com/218381 Conflicts: drivers/gpu/drm/Makefile (cherry picked from cros/chromeos-3.14 commit 0adee464da8094c70469514dd96799c1797f77b0) Signed-off-by: ZhengShunQian --- drivers/gpu/drm/Kconfig | 8 + drivers/gpu/drm/Makefile | 3 +- drivers/gpu/drm/drm_sync_helper.c | 314 ++++++++++++++++++++++++++++++ include/drm/drm_sync_helper.h | 156 +++++++++++++++ 4 files changed, 480 insertions(+), 1 deletion(-) create mode 100644 drivers/gpu/drm/drm_sync_helper.c create mode 100644 include/drm/drm_sync_helper.h diff --git a/drivers/gpu/drm/Kconfig b/drivers/gpu/drm/Kconfig index c4bf9a1cf4a6..52d6e890bf85 100644 --- a/drivers/gpu/drm/Kconfig +++ b/drivers/gpu/drm/Kconfig @@ -97,6 +97,14 @@ config DRM_KMS_CMA_HELPER help Choose this if you need the KMS CMA helper functions +config DRM_DMA_SYNC + bool "DRM DMA Sync" + depends on DRM && DMA_SHARED_BUFFER + default n + help + Choose this option if you want to use fences and reservations for + synchronization of shared dma-buf access between different drivers. + source "drivers/gpu/drm/i2c/Kconfig" config DRM_TDFX diff --git a/drivers/gpu/drm/Makefile b/drivers/gpu/drm/Makefile index 1e9ff4c3e3db..475e249898e2 100644 --- a/drivers/gpu/drm/Makefile +++ b/drivers/gpu/drm/Makefile @@ -12,7 +12,8 @@ drm-y := drm_auth.o drm_bufs.o drm_cache.o \ drm_info.o drm_debugfs.o drm_encoder_slave.o \ drm_trace_points.o drm_global.o drm_prime.o \ drm_rect.o drm_vma_manager.o drm_flip_work.o \ - drm_modeset_lock.o drm_atomic.o drm_bridge.o + drm_modeset_lock.o drm_atomic.o drm_bridge.o \ + drm_sync_helper.o drm-$(CONFIG_COMPAT) += drm_ioc32.o drm-$(CONFIG_DRM_GEM_CMA_HELPER) += drm_gem_cma_helper.o diff --git a/drivers/gpu/drm/drm_sync_helper.c b/drivers/gpu/drm/drm_sync_helper.c new file mode 100644 index 000000000000..0c6aa875b4c6 --- /dev/null +++ b/drivers/gpu/drm/drm_sync_helper.c @@ -0,0 +1,314 @@ +/* + * drm_sync_helper.c: software fence and helper functions for fences and + * reservations used for dma buffer access synchronization between drivers. + * + * Copyright 2014 Google, Inc. + * + * This software is licensed under the terms of the GNU General Public + * License version 2, as published by the Free Software Foundation, and + * may be copied, distributed, and modified under those terms. + * + * 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. + */ + +#include +#include +#include +#include + +static DEFINE_SPINLOCK(sw_fence_lock); + +void drm_add_reservation(struct reservation_object *resv, + struct reservation_object **resvs, + unsigned long *excl_resvs_bitmap, + unsigned int *num_resvs, bool exclusive) +{ + unsigned int r; + + for (r = 0; r < *num_resvs; r++) { + if (resvs[r] == resv) + return; + } + resvs[*num_resvs] = resv; + if (exclusive) + set_bit(*num_resvs, excl_resvs_bitmap); + (*num_resvs)++; +} +EXPORT_SYMBOL(drm_add_reservation); + +int drm_lock_reservations(struct reservation_object **resvs, + unsigned int num_resvs, struct ww_acquire_ctx *ctx) +{ + unsigned int r; + struct reservation_object *slow_res = NULL; + + ww_acquire_init(ctx, &reservation_ww_class); + +retry: + for (r = 0; r < num_resvs; r++) { + int ret; + /* skip the resv we locked with slow lock */ + if (resvs[r] == slow_res) { + slow_res = NULL; + continue; + } + ret = ww_mutex_lock(&resvs[r]->lock, ctx); + if (ret < 0) { + unsigned int slow_r = r; + /* + * undo all the locks we already done, + * in reverse order + */ + while (r > 0) { + r--; + ww_mutex_unlock(&resvs[r]->lock); + } + if (slow_res) + ww_mutex_unlock(&slow_res->lock); + if (ret == -EDEADLK) { + slow_res = resvs[slow_r]; + ww_mutex_lock_slow(&slow_res->lock, ctx); + goto retry; + } + ww_acquire_fini(ctx); + return ret; + } + } + + ww_acquire_done(ctx); + return 0; +} +EXPORT_SYMBOL(drm_lock_reservations); + +void drm_unlock_reservations(struct reservation_object **resvs, + unsigned int num_resvs, + struct ww_acquire_ctx *ctx) +{ + unsigned int r; + + for (r = 0; r < num_resvs; r++) + ww_mutex_unlock(&resvs[r]->lock); + + ww_acquire_fini(ctx); +} +EXPORT_SYMBOL(drm_unlock_reservations); + +static void reservation_cb_fence_cb(struct fence *fence, struct fence_cb *cb) +{ + struct drm_reservation_fence_cb *rfcb = + container_of(cb, struct drm_reservation_fence_cb, base); + struct drm_reservation_cb *rcb = rfcb->parent; + + if (atomic_dec_and_test(&rcb->count)) + schedule_work(&rcb->work); +} + +static void +reservation_cb_cleanup(struct drm_reservation_cb *rcb) +{ + unsigned cb; + + for (cb = 0; cb < rcb->num_fence_cbs; cb++) { + if (rcb->fence_cbs[cb]) { + fence_remove_callback(rcb->fence_cbs[cb]->fence, + &rcb->fence_cbs[cb]->base); + fence_put(rcb->fence_cbs[cb]->fence); + kfree(rcb->fence_cbs[cb]); + rcb->fence_cbs[cb] = NULL; + } + } + kfree(rcb->fence_cbs); + rcb->fence_cbs = NULL; + rcb->num_fence_cbs = 0; +} + +static void reservation_cb_work(struct work_struct *pwork) +{ + struct drm_reservation_cb *rcb = + container_of(pwork, struct drm_reservation_cb, work); + /* + * clean up everything before calling the callback, because the callback + * may free structure containing rcb and work_struct + */ + reservation_cb_cleanup(rcb); + rcb->func(rcb, rcb->context); +} + +static int +reservation_cb_add_fence_cb(struct drm_reservation_cb *rcb, struct fence *fence) +{ + int ret = 0; + struct drm_reservation_fence_cb *fence_cb; + struct drm_reservation_fence_cb **new_fence_cbs; + + new_fence_cbs = krealloc(rcb->fence_cbs, + (rcb->num_fence_cbs + 1) + * sizeof(struct drm_reservation_fence_cb *), + GFP_KERNEL); + if (!new_fence_cbs) + return -ENOMEM; + rcb->fence_cbs = new_fence_cbs; + + fence_cb = kzalloc(sizeof(struct drm_reservation_fence_cb), GFP_KERNEL); + if (!fence_cb) + return -ENOMEM; + + /* + * do not want for fence to disappear on us while we are waiting for + * callback and we need it in case we want to remove callbacks + */ + fence_get(fence); + fence_cb->fence = fence; + fence_cb->parent = rcb; + rcb->fence_cbs[rcb->num_fence_cbs] = fence_cb; + atomic_inc(&rcb->count); + ret = fence_add_callback(fence, &fence_cb->base, + reservation_cb_fence_cb); + if (ret == -ENOENT) { + /* already signaled */ + atomic_dec(&rcb->count); + fence_put(fence_cb->fence); + kfree(fence_cb); + ret = 0; + } else if (ret < 0) { + atomic_dec(&rcb->count); + fence_put(fence_cb->fence); + kfree(fence_cb); + return ret; + } else { + rcb->num_fence_cbs++; + } + return ret; +} + +void +drm_reservation_cb_init(struct drm_reservation_cb *rcb, + drm_reservation_cb_func_t func, void *context) +{ + INIT_WORK(&rcb->work, reservation_cb_work); + atomic_set(&rcb->count, 1); + rcb->num_fence_cbs = 0; + rcb->fence_cbs = NULL; + rcb->func = func; + rcb->context = context; +} +EXPORT_SYMBOL(drm_reservation_cb_init); + +int +drm_reservation_cb_add(struct drm_reservation_cb *rcb, + struct reservation_object *resv, bool exclusive) +{ + int ret = 0; + struct fence *fence; + unsigned shared_count = 0, f; + struct fence **shared_fences = NULL; + + /* enum all the fences in the reservation and add callbacks */ + ret = reservation_object_get_fences_rcu(resv, &fence, + &shared_count, &shared_fences); + if (ret < 0) + return ret; + + if (fence) { + ret = reservation_cb_add_fence_cb(rcb, fence); + if (ret < 0) { + reservation_cb_cleanup(rcb); + goto error; + } + } + + if (exclusive) { + for (f = 0; f < shared_count; f++) { + ret = reservation_cb_add_fence_cb(rcb, + shared_fences[f]); + if (ret < 0) { + reservation_cb_cleanup(rcb); + goto error; + } + } + } + +error: + if (fence) + fence_put(fence); + if (shared_fences) { + for (f = 0; f < shared_count; f++) + fence_put(shared_fences[f]); + kfree(shared_fences); + } + return ret; +} +EXPORT_SYMBOL(drm_reservation_cb_add); + +void +drm_reservation_cb_done(struct drm_reservation_cb *rcb) +{ + /* + * we need to decrement from initial 1 + * and trigger the callback in case all the + * fences were already triggered + */ + if (atomic_dec_and_test(&rcb->count)) { + /* + * we could call the callback here directly but in case + * the callback function needs to lock the same mutex + * as our caller it could cause a deadlock, so it is + * safer to call it from a worker + */ + schedule_work(&rcb->work); + } +} +EXPORT_SYMBOL(drm_reservation_cb_done); + +void +drm_reservation_cb_fini(struct drm_reservation_cb *rcb) +{ + /* make sure no work will be triggered */ + atomic_set(&rcb->count, 0); + cancel_work_sync(&rcb->work); + reservation_cb_cleanup(rcb); +} +EXPORT_SYMBOL(drm_reservation_cb_fini); + +static bool sw_fence_enable_signaling(struct fence *f) +{ + return true; +} + +static const char *sw_fence_get_get_driver_name(struct fence *fence) +{ + return "drm_sync_helper"; +} + +static const char *sw_fence_get_timeline_name(struct fence *f) +{ + return "drm_sync.sw"; +} + +static const struct fence_ops sw_fence_ops = { + .get_driver_name = sw_fence_get_get_driver_name, + .get_timeline_name = sw_fence_get_timeline_name, + .enable_signaling = sw_fence_enable_signaling, + .signaled = NULL, + .wait = fence_default_wait, + .release = NULL +}; + +struct fence *drm_sw_fence_new(unsigned int context, unsigned seqno) +{ + struct fence *fence; + + fence = kzalloc(sizeof(*fence), GFP_KERNEL); + if (!fence) + return ERR_PTR(-ENOMEM); + fence_init(fence, + &sw_fence_ops, + &sw_fence_lock, + context, seqno); + + return fence; +} +EXPORT_SYMBOL(drm_sw_fence_new); diff --git a/include/drm/drm_sync_helper.h b/include/drm/drm_sync_helper.h new file mode 100644 index 000000000000..a5586c674fec --- /dev/null +++ b/include/drm/drm_sync_helper.h @@ -0,0 +1,156 @@ +/* + * drm_sync_helper.h: software fence and helper functions for fences and + * reservations used for dma buffer access synchronization between drivers. + * + * Copyright 2014 Google, Inc. + * + * This software is licensed under the terms of the GNU General Public + * License version 2, as published by the Free Software Foundation, and + * may be copied, distributed, and modified under those terms. + * + * 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. + */ + +#ifndef _DRM_SYNC_HELPER_H_ +#define _DRM_SYNC_HELPER_H_ + +#include +#include +#include +#include + +/** + * Create software fence + * @context: execution context + * @seqno: the sequence number of this fence inside the execution context + */ +struct fence *drm_sw_fence_new(unsigned int context, + unsigned seqno); + +/** + * Signal and decrease reference count for a fence if it exists + * @fence: fence to signal + * + * Utility function called when owner access to object associated with fence is + * finished (e.g. GPU done with rendering). + */ +static inline void drm_fence_signal_and_put(struct fence **fence) +{ + if (*fence) { + fence_signal(*fence); + fence_put(*fence); + *fence = NULL; + } +} + +struct drm_reservation_cb; + +struct drm_reservation_fence_cb { + struct fence_cb base; + struct drm_reservation_cb *parent; + struct fence *fence; +}; + +/** + * Callback executed when all fences in reservation callback are signaled + * @rcb: reservation callback structure + * @context: context provided by user at init time + */ +typedef void (*drm_reservation_cb_func_t)(struct drm_reservation_cb *rcb, + void *context); + +/** + * Reservation callback structure + * @work: work context in which func is executed + * @fence_cbs: fence callbacks array + * @num_fence_cbs: number of fence callbacks + * @count: count of signaled fences, when it drops to 0 func is called + * @func: callback to execute when all fences are signaled + * @context: context provided by user during initialization + * + * It is safe and expected that func will destroy this structure before + * returning. + */ +struct drm_reservation_cb { + struct work_struct work; + struct drm_reservation_fence_cb **fence_cbs; + unsigned num_fence_cbs; + atomic_t count; + void *context; + drm_reservation_cb_func_t func; +}; + +/** + * Initialize reservation callback + * @rcb: reservation callback structure to initialize + * @func: function to call when all fences are signaled + * @context: parameter to call func with + */ +void drm_reservation_cb_init(struct drm_reservation_cb *rcb, + drm_reservation_cb_func_t func, + void *context); + +/** + * Add fences from reservation object to callback + * @rcb: reservation callback structure + * @resv: reservation object + * @exclusive: (for exclusive wait) when true add all fences, otherwise only + * exclusive fence + */ +int drm_reservation_cb_add(struct drm_reservation_cb *rcb, + struct reservation_object *resv, + bool exclusive); + +/** + * Finish adding fences + * @rcb: reservation callback structure + * + * It will trigger callback worker if all fences were signaled before. + */ +void drm_reservation_cb_done(struct drm_reservation_cb *rcb); + +/** + * Cleanup reservation callback structure + * @rcb: reservation callback structure + * + * Can be called to cancel primed reservation callback. + */ +void drm_reservation_cb_fini(struct drm_reservation_cb *rcb); + +/** + * Add reservation to array of reservations + * @resv: reservation to add + * @resvs: array of reservations + * @excl_resvs_bitmap: bitmap for exclusive reservations + * @num_resvs: number of reservations in array + * @exclusive: bool to store in excl_resvs_bitmap + */ +void +drm_add_reservation(struct reservation_object *resv, + struct reservation_object **resvs, + unsigned long *excl_resvs_bitmap, + unsigned int *num_resvs, bool exclusive); + +/** + * Acquire ww_mutex lock on all reservations in the array + * @resvs: array of reservations + * @num_resvs: number of reservations in the array + * @ctx: ww mutex context + */ +int drm_lock_reservations(struct reservation_object **resvs, + unsigned int num_resvs, struct ww_acquire_ctx *ctx); + +/** + * Release ww_mutex lock on all reservations in the array + * @resvs: array of reservations + * @num_resvs: number of reservations in the array + * @ctx: ww mutex context + */ +void drm_unlock_reservations(struct reservation_object **resvs, + unsigned int num_resvs, + struct ww_acquire_ctx *ctx); + +#endif -- 2.34.1