Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/external/bsd/drm2/linux Implement poll for reservations ...
details: https://anonhg.NetBSD.org/src/rev/f1da39041a8b
branches: trunk
changeset: 993102:f1da39041a8b
user: riastradh <riastradh%NetBSD.org@localhost>
date: Mon Aug 27 15:25:13 2018 +0000
description:
Implement poll for reservations and dma bufs.
diffstat:
sys/external/bsd/drm2/include/linux/dma-buf.h | 4 +-
sys/external/bsd/drm2/include/linux/reservation.h | 25 +-
sys/external/bsd/drm2/linux/linux_dma_buf.c | 22 +-
sys/external/bsd/drm2/linux/linux_reservation.c | 269 +++++++++++++++++++++-
4 files changed, 300 insertions(+), 20 deletions(-)
diffs (truncated from 439 to 300 lines):
diff -r 72cb4c30474c -r f1da39041a8b sys/external/bsd/drm2/include/linux/dma-buf.h
--- a/sys/external/bsd/drm2/include/linux/dma-buf.h Mon Aug 27 15:24:53 2018 +0000
+++ b/sys/external/bsd/drm2/include/linux/dma-buf.h Mon Aug 27 15:25:13 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: dma-buf.h,v 1.3 2018/08/27 15:22:54 riastradh Exp $ */
+/* $NetBSD: dma-buf.h,v 1.4 2018/08/27 15:25:13 riastradh Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -88,7 +88,7 @@
kmutex_t db_lock;
volatile unsigned db_refcnt;
- struct selinfo db_selq;
+ struct reservation_poll db_resv_poll;
struct reservation_object db_resv_int[];
};
diff -r 72cb4c30474c -r f1da39041a8b sys/external/bsd/drm2/include/linux/reservation.h
--- a/sys/external/bsd/drm2/include/linux/reservation.h Mon Aug 27 15:24:53 2018 +0000
+++ b/sys/external/bsd/drm2/include/linux/reservation.h Mon Aug 27 15:25:13 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: reservation.h,v 1.7 2018/08/27 14:01:14 riastradh Exp $ */
+/* $NetBSD: reservation.h,v 1.8 2018/08/27 15:25:13 riastradh Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -32,11 +32,10 @@
#ifndef _LINUX_RESERVATION_H_
#define _LINUX_RESERVATION_H_
+#include <linux/fence.h>
#include <linux/rcupdate.h>
#include <linux/ww_mutex.h>
-struct fence;
-
struct reservation_object {
struct ww_mutex lock;
@@ -54,6 +53,14 @@
struct fence __rcu *shared[];
};
+/* NetBSD addition */
+struct reservation_poll {
+ kmutex_t rp_lock;
+ struct selinfo rp_selq;
+ struct fence_cb rp_fcb;
+ bool rp_claimed;
+};
+
#define reservation_object_add_excl_fence linux_reservation_object_add_excl_fence
#define reservation_object_add_shared_fence linux_reservation_object_add_shared_fence
#define reservation_object_fini linux_reservation_object_fini
@@ -62,9 +69,13 @@
#define reservation_object_get_list linux_reservation_object_get_list
#define reservation_object_held linux_reservation_object_held
#define reservation_object_init linux_reservation_object_init
+#define reservation_object_kqfilter linux_reservation_object_kqfilter
+#define reservation_object_poll linux_reservation_object_poll
#define reservation_object_reserve_shared linux_reservation_object_reserve_shared
#define reservation_object_test_signaled_rcu linux_reservation_object_test_signaled_rcu
#define reservation_object_wait_timeout_rcu linux_reservation_object_wait_timeout_rcu
+#define reservation_poll_fini linux_reservation_poll_fini
+#define reservation_poll_init linux_reservation_poll_init
#define reservation_ww_class linux_reservation_ww_class
extern struct ww_class reservation_ww_class;
@@ -90,4 +101,12 @@
long reservation_object_wait_timeout_rcu(struct reservation_object *,
bool, bool, unsigned long);
+/* NetBSD additions */
+void reservation_poll_init(struct reservation_poll *);
+void reservation_poll_fini(struct reservation_poll *);
+int reservation_object_poll(struct reservation_object *, int,
+ struct reservation_poll *);
+int reservation_object_kqfilter(struct reservation_object *,
+ struct knote *, struct reservation_poll *);
+
#endif /* _LINUX_RESERVATION_H_ */
diff -r 72cb4c30474c -r f1da39041a8b sys/external/bsd/drm2/linux/linux_dma_buf.c
--- a/sys/external/bsd/drm2/linux/linux_dma_buf.c Mon Aug 27 15:24:53 2018 +0000
+++ b/sys/external/bsd/drm2/linux/linux_dma_buf.c Mon Aug 27 15:25:13 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: linux_dma_buf.c,v 1.3 2018/08/27 15:24:40 riastradh Exp $ */
+/* $NetBSD: linux_dma_buf.c,v 1.4 2018/08/27 15:25:13 riastradh Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: linux_dma_buf.c,v 1.3 2018/08/27 15:24:40 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_dma_buf.c,v 1.4 2018/08/27 15:25:13 riastradh Exp $");
#include <sys/types.h>
#include <sys/atomic.h>
@@ -87,6 +87,7 @@
mutex_init(&dmabuf->db_lock, MUTEX_DEFAULT, IPL_NONE);
dmabuf->db_refcnt = 1;
+ reservation_poll_init(&dmabuf->db_resv_poll);
if (dmabuf->resv == NULL) {
dmabuf->resv = &dmabuf->db_resv_int[0];
@@ -167,6 +168,7 @@
if (atomic_dec_uint_nv(&dmabuf->db_refcnt) != 0)
return;
+ reservation_poll_fini(&dmabuf->db_resv_poll);
mutex_destroy(&dmabuf->db_lock);
if (dmabuf->resv == &dmabuf->db_resv_int[0]) {
reservation_object_fini(dmabuf->resv);
@@ -239,27 +241,21 @@
static int
dmabuf_fop_poll(struct file *file, int events)
{
-#ifdef notyet
struct dma_buf_file *dbf = file->f_data;
struct dma_buf *dmabuf = dbf->dbf_dmabuf;
+ struct reservation_poll *rpoll = &dmabuf->db_resv_poll;
- return reservation_object_poll(dmabuf->resv, events);
-#else
- return -ENOSYS;
-#endif
+ return reservation_object_poll(dmabuf->resv, events, rpoll);
}
static int
-dmabuf_fop_kqfilter(struct file *file, struct knote *knote)
+dmabuf_fop_kqfilter(struct file *file, struct knote *kn)
{
-#ifdef notyet
struct dma_buf_file *dbf = file->f_data;
struct dma_buf *dmabuf = dbf->dbf_dmabuf;
+ struct reservation_poll *rpoll = &dmabuf->db_resv_poll;
- return reservation_object_kqfilter(dmabuf->resv, knote);
-#else
- return -ENOSYS;
-#endif
+ return reservation_object_kqfilter(dmabuf->resv, kn, rpoll);
}
static int
diff -r 72cb4c30474c -r f1da39041a8b sys/external/bsd/drm2/linux/linux_reservation.c
--- a/sys/external/bsd/drm2/linux/linux_reservation.c Mon Aug 27 15:24:53 2018 +0000
+++ b/sys/external/bsd/drm2/linux/linux_reservation.c Mon Aug 27 15:25:13 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: linux_reservation.c,v 1.7 2018/08/27 14:14:13 riastradh Exp $ */
+/* $NetBSD: linux_reservation.c,v 1.8 2018/08/27 15:25:14 riastradh Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -30,10 +30,15 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: linux_reservation.c,v 1.7 2018/08/27 14:14:13 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: linux_reservation.c,v 1.8 2018/08/27 15:25:14 riastradh Exp $");
+
+#include <sys/param.h>
+#include <sys/poll.h>
+#include <sys/select.h>
#include <linux/fence.h>
#include <linux/reservation.h>
+#include <linux/ww_mutex.h>
DEFINE_WW_CLASS(reservation_ww_class __cacheline_aligned);
@@ -792,3 +797,263 @@
return timeout;
goto top;
}
+
+/*
+ * reservation_poll_init(rpoll, lock)
+ *
+ * Initialize reservation poll state.
+ */
+void
+reservation_poll_init(struct reservation_poll *rpoll)
+{
+
+ mutex_init(&rpoll->rp_lock, MUTEX_DEFAULT, IPL_VM);
+ selinit(&rpoll->rp_selq);
+ rpoll->rp_claimed = 0;
+}
+
+/*
+ * reservation_poll_fini(rpoll)
+ *
+ * Release any resource associated with reservation poll state.
+ */
+void
+reservation_poll_fini(struct reservation_poll *rpoll)
+{
+
+ KASSERT(rpoll->rp_claimed == 0);
+ seldestroy(&rpoll->rp_selq);
+ mutex_destroy(&rpoll->rp_lock);
+}
+
+/*
+ * reservation_poll_cb(fence, fcb)
+ *
+ * Callback to notify a reservation poll that a fence has
+ * completed. Notify any waiters and allow the next poller to
+ * claim the callback.
+ *
+ * If one thread is waiting for the exclusive fence only, and we
+ * spuriously notify them about a shared fence, tough.
+ */
+static void
+reservation_poll_cb(struct fence *fence, struct fence_cb *fcb)
+{
+ struct reservation_poll *rpoll = container_of(fcb,
+ struct reservation_poll, rp_fcb);
+
+ mutex_enter(&rpoll->rp_lock);
+ selnotify(&rpoll->rp_selq, 0, NOTE_SUBMIT);
+ rpoll->rp_claimed = 0;
+ mutex_exit(&rpoll->rp_lock);
+}
+
+/*
+ * reservation_object_poll(robj, events, rpoll)
+ *
+ * POLLOUT wait for all fences shared and exclusive
+ * POLLIN wait for the exclusive fence
+ */
+int
+reservation_object_poll(struct reservation_object *robj, int events,
+ struct reservation_poll *rpoll)
+{
+ struct reservation_object_read_ticket ticket;
+ struct reservation_object_list *list;
+ struct fence *fence;
+ uint32_t i, shared_count;
+ int revents;
+ bool recorded = false; /* curlwp is on the selq */
+ bool claimed = false; /* we claimed the callback */
+ bool callback = false; /* we requested a callback */
+
+ /*
+ * Start with the maximal set of events that could be ready.
+ * We will eliminate the events that are definitely not ready
+ * as we go at the same time as we add callbacks to notify us
+ * that they may be ready.
+ */
+ revents = events & (POLLIN|POLLOUT);
+ if (revents == 0)
+ return 0;
+
+top:
+ /* Enter an RCU read section and get a read ticket. */
+ rcu_read_lock();
+ reservation_object_read_begin(robj, &ticket);
+
+ /* If we want to wait for all fences, get the shared list. */
+ if ((events & POLLOUT) && (list = robj->robj_list) != NULL) do {
+ /* Make sure the content of the list has been published. */
+ membar_datadep_consumer();
+
+ /* Find out how long it is. */
+ shared_count = list->shared_count;
+
+ /*
+ * Make sure we saw a consistent snapshot of the list
+ * pointer and length.
+ */
+ if (!reservation_object_read_valid(robj, &ticket))
+ goto restart;
+
+ /*
+ * For each fence, if it is going away, restart.
+ * Otherwise, acquire a reference to it to test whether
+ * it is signalled. Stop and request a callback if we
+ * find any that is not signalled.
+ */
+ for (i = 0; i < shared_count; i++) {
+ fence = fence_get_rcu(list->shared[i]);
+ if (fence == NULL)
+ goto restart;
+ if (!fence_is_signaled(fence)) {
+ fence_put(fence);
+ break;
+ }
+ fence_put(fence);
+ }
+
+ /* If all shared fences have been signalled, move on. */
+ if (i == shared_count)
+ break;
+
Home |
Main Index |
Thread Index |
Old Index