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/include/linux Implement Linux asynchro...
details: https://anonhg.NetBSD.org/src/rev/165cb2fa93d4
branches: trunk
changeset: 992662:165cb2fa93d4
user: riastradh <riastradh%NetBSD.org@localhost>
date: Mon Aug 27 07:24:54 2018 +0000
description:
Implement Linux asynchronous firmware load API.
diffstat:
sys/external/bsd/drm2/include/linux/firmware.h | 85 +++++++++++++++++++++++++-
1 files changed, 84 insertions(+), 1 deletions(-)
diffs (117 lines):
diff -r b88625bc77b4 -r 165cb2fa93d4 sys/external/bsd/drm2/include/linux/firmware.h
--- a/sys/external/bsd/drm2/include/linux/firmware.h Mon Aug 27 07:24:44 2018 +0000
+++ b/sys/external/bsd/drm2/include/linux/firmware.h Mon Aug 27 07:24:54 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: firmware.h,v 1.7 2018/08/27 06:43:24 riastradh Exp $ */
+/* $NetBSD: firmware.h,v 1.8 2018/08/27 07:24:54 riastradh Exp $ */
/*-
* Copyright (c) 2013 The NetBSD Foundation, Inc.
@@ -35,10 +35,15 @@
#include <sys/types.h>
#include <sys/device.h>
#include <sys/kmem.h>
+#include <sys/module.h>
#include <sys/systm.h>
#include <dev/firmload.h>
+#include <linux/slab.h>
+#include <linux/string.h>
+#include <linux/workqueue.h>
+
struct device;
struct firmware {
@@ -46,6 +51,15 @@
size_t size;
};
+struct firmload_work {
+ char *flw_name;
+ void (*flw_callback)(const struct firmware *, void *);
+ void *flw_cookie;
+ struct device *flw_device;
+ struct module *flw_module;
+ struct work_struct flw_work;
+};
+
static inline int
request_firmware(const struct firmware **fwp, const char *image_name,
struct device *dev)
@@ -92,6 +106,75 @@
}
static inline void
+request_firmware_work(struct work_struct *wk)
+{
+ struct firmload_work *work = container_of(wk, struct firmload_work,
+ flw_work);
+ const struct firmware *fw;
+ int ret;
+
+ /* Reqeust the firmware. If it failed, set it to NULL. */
+ ret = request_firmware(&fw, work->flw_name, work->flw_device);
+ if (ret)
+ fw = NULL;
+
+ /* Call the callback. */
+ (*work->flw_callback)(fw, work->flw_cookie);
+
+ /*
+ * Release the device and module references now that we're
+ * done.
+ *
+ * XXX Heh. What if the module gets unloaded _during_
+ * module_rele because it went to zero?
+ */
+ /* XXX device_release */
+ if (work->flw_module)
+ module_rele(work->flw_module);
+}
+
+static inline int
+request_firmware_nowait(struct module *module, bool uevent, const char *name,
+ struct device *device, gfp_t gfp, void *cookie,
+ void (*callback)(const struct firmware *, void *))
+{
+ char *namedup;
+ struct firmload_work *work;
+
+ /* Allocate memory for it, or fail if we can't. */
+ work = kzalloc(sizeof(*work), gfp);
+ if (work == NULL)
+ goto fail0;
+
+ /* Copy the name just in case. */
+ namedup = kstrdup(name, gfp);
+ if (namedup == NULL)
+ goto fail1;
+
+ /* Hold the module and device so they don't go away before callback. */
+ if (module)
+ module_hold(module);
+ /* XXX device_acquire(device) */
+
+ /* Initialize the work. */
+ work->flw_name = namedup;
+ work->flw_callback = callback;
+ work->flw_cookie = callback;
+ work->flw_device = device;
+ work->flw_module = module;
+ INIT_WORK(&work->flw_work, request_firmware_work);
+
+ /* Kick it off. */
+ schedule_work(&work->flw_work);
+
+ /* Success! */
+ return 0;
+
+fail1: kfree(work);
+fail0: return -ENOMEM;
+}
+
+static inline void
release_firmware(const struct firmware *fw)
{
Home |
Main Index |
Thread Index |
Old Index