Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/sdmmc After setting HS_TIMING value for HS200 or lat...
details: https://anonhg.NetBSD.org/src/rev/77e20433a9fb
branches: trunk
changeset: 341294:77e20433a9fb
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Thu Oct 29 22:37:15 2015 +0000
description:
After setting HS_TIMING value for HS200 or later, send repeated SEND_STATUS
command until the device is no longer busy or the SWITCH_ERROR bit is set.
diffstat:
sys/dev/sdmmc/sdmmc_mem.c | 36 +++++++++++++++++++++++++++++++++---
sys/dev/sdmmc/sdmmcreg.h | 3 ++-
2 files changed, 35 insertions(+), 4 deletions(-)
diffs (81 lines):
diff -r a3a870aeabe2 -r 77e20433a9fb sys/dev/sdmmc/sdmmc_mem.c
--- a/sys/dev/sdmmc/sdmmc_mem.c Thu Oct 29 21:07:48 2015 +0000
+++ b/sys/dev/sdmmc/sdmmc_mem.c Thu Oct 29 22:37:15 2015 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sdmmc_mem.c,v 1.47 2015/10/06 14:32:51 mlelstv Exp $ */
+/* $NetBSD: sdmmc_mem.c,v 1.48 2015/10/29 22:37:15 jmcneill Exp $ */
/* $OpenBSD: sdmmc_mem.c,v 1.10 2009/01/09 10:55:22 jsg Exp $ */
/*
@@ -45,7 +45,7 @@
/* Routines for SD/MMC memory cards. */
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sdmmc_mem.c,v 1.47 2015/10/06 14:32:51 mlelstv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sdmmc_mem.c,v 1.48 2015/10/29 22:37:15 jmcneill Exp $");
#ifdef _KERNEL_OPT
#include "opt_sdmmc.h"
@@ -1414,6 +1414,7 @@
{
struct sdmmc_softc *sc = sf->sc;
struct sdmmc_command cmd;
+ int error;
memset(&cmd, 0, sizeof(cmd));
cmd.c_opcode = MMC_SWITCH;
@@ -1421,7 +1422,36 @@
(index << 16) | (value << 8) | set;
cmd.c_flags = SCF_RSP_SPI_R1B | SCF_RSP_R1B | SCF_CMD_AC;
- return sdmmc_mmc_command(sc, &cmd);
+ error = sdmmc_mmc_command(sc, &cmd);
+ if (error)
+ return error;
+
+ if (index == EXT_CSD_HS_TIMING && value >= 2) {
+ do {
+ memset(&cmd, 0, sizeof(cmd));
+ cmd.c_opcode = MMC_SEND_STATUS;
+ if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE))
+ cmd.c_arg = MMC_ARG_RCA(sf->rca);
+ cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1 | SCF_RSP_SPI_R2;
+ error = sdmmc_mmc_command(sc, &cmd);
+ if (error)
+ break;
+ if (ISSET(MMC_R1(cmd.c_resp), MMC_R1_SWITCH_ERROR)) {
+ aprint_error_dev(sc->sc_dev, "switch error\n");
+ return EINVAL;
+ }
+ /* XXX time out */
+ } while (!ISSET(MMC_R1(cmd.c_resp), MMC_R1_READY_FOR_DATA));
+
+ if (error) {
+ aprint_error_dev(sc->sc_dev,
+ "error waiting for high speed switch: %d\n",
+ error);
+ return error;
+ }
+ }
+
+ return 0;
}
/*
diff -r a3a870aeabe2 -r 77e20433a9fb sys/dev/sdmmc/sdmmcreg.h
--- a/sys/dev/sdmmc/sdmmcreg.h Thu Oct 29 21:07:48 2015 +0000
+++ b/sys/dev/sdmmc/sdmmcreg.h Thu Oct 29 22:37:15 2015 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sdmmcreg.h,v 1.20 2015/08/08 10:50:55 jmcneill Exp $ */
+/* $NetBSD: sdmmcreg.h,v 1.21 2015/10/29 22:37:15 jmcneill Exp $ */
/* $OpenBSD: sdmmcreg.h,v 1.4 2009/01/09 10:55:22 jsg Exp $ */
/*
@@ -98,6 +98,7 @@
/* R1 response type bits */
#define MMC_R1_READY_FOR_DATA (1<<8) /* ready for next transfer */
+#define MMC_R1_SWITCH_ERROR (1<<7) /* switch command failed */
#define MMC_R1_APP_CMD (1<<5) /* app. commands supported */
/* 48-bit response decoding (32 bits w/o CRC) */
Home |
Main Index |
Thread Index |
Old Index