Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/sdmmc Defer DIOCCACHESYNC to the sdmmc task queue so...
details: https://anonhg.NetBSD.org/src/rev/36e620032d5b
branches: trunk
changeset: 994470:36e620032d5b
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Fri Nov 09 14:39:19 2018 +0000
description:
Defer DIOCCACHESYNC to the sdmmc task queue so they are serialized with other requests.
diffstat:
sys/dev/sdmmc/ld_sdmmc.c | 62 ++++++++++++++++++++++++++++++++++++++++++++---
1 files changed, 57 insertions(+), 5 deletions(-)
diffs (119 lines):
diff -r b0677e03f85e -r 36e620032d5b sys/dev/sdmmc/ld_sdmmc.c
--- a/sys/dev/sdmmc/ld_sdmmc.c Fri Nov 09 14:38:36 2018 +0000
+++ b/sys/dev/sdmmc/ld_sdmmc.c Fri Nov 09 14:39:19 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: ld_sdmmc.c,v 1.34 2017/08/20 15:58:43 mlelstv Exp $ */
+/* $NetBSD: ld_sdmmc.c,v 1.35 2018/11/09 14:39:19 jmcneill Exp $ */
/*
* Copyright (c) 2008 KIYOHARA Takashi
@@ -28,7 +28,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: ld_sdmmc.c,v 1.34 2017/08/20 15:58:43 mlelstv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: ld_sdmmc.c,v 1.35 2018/11/09 14:39:19 jmcneill Exp $");
#ifdef _KERNEL_OPT
#include "opt_sdmmc.h"
@@ -78,6 +78,11 @@
struct buf *task_bp;
int task_retries; /* number of xfer retry */
struct callout task_restart_ch;
+
+ kmutex_t task_lock;
+ kcondvar_t task_cv;
+
+ uintptr_t task_data;
};
struct ld_sdmmc_softc {
@@ -91,6 +96,7 @@
struct evcnt sc_ev_discard; /* discard counter */
struct evcnt sc_ev_discarderr; /* discard error counter */
struct evcnt sc_ev_discardbusy; /* discard busy counter */
+ struct evcnt sc_ev_cachesyncbusy; /* cache sync busy counter */
};
static int ld_sdmmc_match(device_t, cfdata_t, void *);
@@ -153,6 +159,8 @@
task = &sc->sc_task[i];
task->task_sc = sc;
callout_init(&task->task_restart_ch, CALLOUT_MPSAFE);
+ mutex_init(&task->task_lock, MUTEX_DEFAULT, IPL_NONE);
+ cv_init(&task->task_cv, "ldsdmmctask");
pcq_put(sc->sc_freeq, task);
}
@@ -224,8 +232,11 @@
return rv;
ldenddetach(ld);
- for (i = 0; i < __arraycount(sc->sc_task); i++)
+ for (i = 0; i < __arraycount(sc->sc_task); i++) {
callout_destroy(&sc->sc_task[i].task_restart_ch);
+ mutex_destroy(&sc->sc_task[i].task_lock);
+ cv_destroy(&sc->sc_task[i].task_cv);
+ }
pcq_destroy(sc->sc_freeq);
evcnt_detach(&sc->sc_ev_discard);
@@ -379,15 +390,56 @@
return 0;
}
+static void
+ld_sdmmc_docachesync(void *arg)
+{
+ struct ld_sdmmc_task *task = arg;
+ struct ld_sdmmc_softc *sc = task->task_sc;
+ const bool poll = (bool)task->task_data;
+
+ task->task_data = sdmmc_mem_flush_cache(sc->sc_sf, poll);
+
+ mutex_enter(&task->task_lock);
+ cv_signal(&task->task_cv);
+ mutex_exit(&task->task_lock);
+}
+
+static int
+ld_sdmmc_cachesync(struct ld_softc *ld, bool poll)
+{
+ struct ld_sdmmc_softc *sc = device_private(ld->sc_dv);
+ struct ld_sdmmc_task *task = pcq_get(sc->sc_freeq);
+ int error = 0;
+
+ if (task == NULL) {
+ sc->sc_ev_cachesyncbusy.ev_count++;
+ return EBUSY;
+ }
+
+ sdmmc_init_task(&task->task, ld_sdmmc_docachesync, task);
+ task->task_data = poll;
+
+ mutex_enter(&task->task_lock);
+ sdmmc_add_task(sc->sc_sf->sc, &task->task);
+ error = cv_wait_sig(&task->task_cv, &task->task_lock);
+ mutex_exit(&task->task_lock);
+
+ if (error == 0)
+ error = (int)task->task_data;
+
+ pcq_put(sc->sc_freeq, task);
+
+ return error;
+}
+
static int
ld_sdmmc_ioctl(struct ld_softc *ld, u_long cmd, void *addr, int32_t flag,
bool poll)
{
- struct ld_sdmmc_softc *sc = device_private(ld->sc_dv);
switch (cmd) {
case DIOCCACHESYNC:
- return sdmmc_mem_flush_cache(sc->sc_sf, poll);
+ return ld_sdmmc_cachesync(ld, poll);
default:
return EPASSTHROUGH;
}
Home |
Main Index |
Thread Index |
Old Index