xref: /dragonfly/sys/bus/mmc/mmc_subr.c (revision 85ccd313)
1*85ccd313SImre Vadász /*-
2*85ccd313SImre Vadász  * Copyright (c) 2006 Bernd Walter.  All rights reserved.
3*85ccd313SImre Vadász  * Copyright (c) 2006 M. Warner Losh.  All rights reserved.
4*85ccd313SImre Vadász  *
5*85ccd313SImre Vadász  * Redistribution and use in source and binary forms, with or without
6*85ccd313SImre Vadász  * modification, are permitted provided that the following conditions
7*85ccd313SImre Vadász  * are met:
8*85ccd313SImre Vadász  * 1. Redistributions of source code must retain the above copyright
9*85ccd313SImre Vadász  *    notice, this list of conditions and the following disclaimer.
10*85ccd313SImre Vadász  * 2. Redistributions in binary form must reproduce the above copyright
11*85ccd313SImre Vadász  *    notice, this list of conditions and the following disclaimer in the
12*85ccd313SImre Vadász  *    documentation and/or other materials provided with the distribution.
13*85ccd313SImre Vadász  *
14*85ccd313SImre Vadász  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
15*85ccd313SImre Vadász  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
16*85ccd313SImre Vadász  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
17*85ccd313SImre Vadász  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
18*85ccd313SImre Vadász  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
19*85ccd313SImre Vadász  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
20*85ccd313SImre Vadász  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
21*85ccd313SImre Vadász  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22*85ccd313SImre Vadász  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
23*85ccd313SImre Vadász  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24*85ccd313SImre Vadász  *
25*85ccd313SImre Vadász  * Portions of this software may have been developed with reference to
26*85ccd313SImre Vadász  * the SD Simplified Specification.  The following disclaimer may apply:
27*85ccd313SImre Vadász  *
28*85ccd313SImre Vadász  * The following conditions apply to the release of the simplified
29*85ccd313SImre Vadász  * specification ("Simplified Specification") by the SD Card Association and
30*85ccd313SImre Vadász  * the SD Group. The Simplified Specification is a subset of the complete SD
31*85ccd313SImre Vadász  * Specification which is owned by the SD Card Association and the SD
32*85ccd313SImre Vadász  * Group. This Simplified Specification is provided on a non-confidential
33*85ccd313SImre Vadász  * basis subject to the disclaimers below. Any implementation of the
34*85ccd313SImre Vadász  * Simplified Specification may require a license from the SD Card
35*85ccd313SImre Vadász  * Association, SD Group, SD-3C LLC or other third parties.
36*85ccd313SImre Vadász  *
37*85ccd313SImre Vadász  * Disclaimers:
38*85ccd313SImre Vadász  *
39*85ccd313SImre Vadász  * The information contained in the Simplified Specification is presented only
40*85ccd313SImre Vadász  * as a standard specification for SD Cards and SD Host/Ancillary products and
41*85ccd313SImre Vadász  * is provided "AS-IS" without any representations or warranties of any
42*85ccd313SImre Vadász  * kind. No responsibility is assumed by the SD Group, SD-3C LLC or the SD
43*85ccd313SImre Vadász  * Card Association for any damages, any infringements of patents or other
44*85ccd313SImre Vadász  * right of the SD Group, SD-3C LLC, the SD Card Association or any third
45*85ccd313SImre Vadász  * parties, which may result from its use. No license is granted by
46*85ccd313SImre Vadász  * implication, estoppel or otherwise under any patent or other rights of the
47*85ccd313SImre Vadász  * SD Group, SD-3C LLC, the SD Card Association or any third party. Nothing
48*85ccd313SImre Vadász  * herein shall be construed as an obligation by the SD Group, the SD-3C LLC
49*85ccd313SImre Vadász  * or the SD Card Association to disclose or distribute any technical
50*85ccd313SImre Vadász  * information, know-how or other confidential information to any third party.
51*85ccd313SImre Vadász  *
52*85ccd313SImre Vadász  * $FreeBSD: head/sys/dev/mmc/mmc_subr.c 315430 2017-03-16 22:23:04Z marius $
53*85ccd313SImre Vadász  */
54*85ccd313SImre Vadász 
55*85ccd313SImre Vadász #include <sys/param.h>
56*85ccd313SImre Vadász #include <sys/systm.h>
57*85ccd313SImre Vadász #include <sys/kernel.h>
58*85ccd313SImre Vadász #include <sys/lock.h>
59*85ccd313SImre Vadász #include <sys/time.h>
60*85ccd313SImre Vadász #include <sys/bus.h>
61*85ccd313SImre Vadász 
62*85ccd313SImre Vadász #include <bus/mmc/bridge.h>
63*85ccd313SImre Vadász #include <bus/mmc/mmc_private.h>
64*85ccd313SImre Vadász #include <bus/mmc/mmc_subr.h>
65*85ccd313SImre Vadász #include <bus/mmc/mmcreg.h>
66*85ccd313SImre Vadász #include <bus/mmc/mmcbrvar.h>
67*85ccd313SImre Vadász 
68*85ccd313SImre Vadász #include "mmcbus_if.h"
69*85ccd313SImre Vadász 
70*85ccd313SImre Vadász #define	CMD_RETRIES	3
71*85ccd313SImre Vadász #define	LOG_PPS		5 /* Log no more than 5 errors per second. */
72*85ccd313SImre Vadász 
73*85ccd313SImre Vadász int
mmc_wait_for_cmd(device_t brdev,device_t reqdev,struct mmc_command * cmd,int retries)74*85ccd313SImre Vadász mmc_wait_for_cmd(device_t brdev, device_t reqdev, struct mmc_command *cmd,
75*85ccd313SImre Vadász     int retries)
76*85ccd313SImre Vadász {
77*85ccd313SImre Vadász 	struct mmc_request mreq;
78*85ccd313SImre Vadász 	struct mmc_softc *sc;
79*85ccd313SImre Vadász 	int err;
80*85ccd313SImre Vadász 
81*85ccd313SImre Vadász 	do {
82*85ccd313SImre Vadász 		memset(&mreq, 0, sizeof(mreq));
83*85ccd313SImre Vadász 		memset(cmd->resp, 0, sizeof(cmd->resp));
84*85ccd313SImre Vadász 		cmd->retries = 0; /* Retries done here, not in hardware. */
85*85ccd313SImre Vadász 		cmd->mrq = &mreq;
86*85ccd313SImre Vadász 		if (cmd->data != NULL)
87*85ccd313SImre Vadász 			cmd->data->mrq = &mreq;
88*85ccd313SImre Vadász 		mreq.cmd = cmd;
89*85ccd313SImre Vadász 		if (MMCBUS_WAIT_FOR_REQUEST(brdev, reqdev, &mreq) != 0)
90*85ccd313SImre Vadász 			err = MMC_ERR_FAILED;
91*85ccd313SImre Vadász 		else
92*85ccd313SImre Vadász 			err = cmd->error;
93*85ccd313SImre Vadász 	} while (err != MMC_ERR_NONE && retries-- > 0);
94*85ccd313SImre Vadász 
95*85ccd313SImre Vadász 	if (err != MMC_ERR_NONE && brdev == reqdev) {
96*85ccd313SImre Vadász 		sc = device_get_softc(brdev);
97*85ccd313SImre Vadász 		if (sc->squelched == 0 && ppsratecheck(&sc->log_time,
98*85ccd313SImre Vadász 		    &sc->log_count, LOG_PPS)) {
99*85ccd313SImre Vadász 			device_printf(sc->dev, "CMD%d failed, RESULT: %d\n",
100*85ccd313SImre Vadász 			    cmd->opcode, err);
101*85ccd313SImre Vadász 		}
102*85ccd313SImre Vadász 	}
103*85ccd313SImre Vadász 
104*85ccd313SImre Vadász 	return (err);
105*85ccd313SImre Vadász }
106*85ccd313SImre Vadász 
107*85ccd313SImre Vadász int
mmc_wait_for_app_cmd(device_t brdev,device_t reqdev,uint16_t rca,struct mmc_command * cmd,int retries)108*85ccd313SImre Vadász mmc_wait_for_app_cmd(device_t brdev, device_t reqdev, uint16_t rca,
109*85ccd313SImre Vadász     struct mmc_command *cmd, int retries)
110*85ccd313SImre Vadász {
111*85ccd313SImre Vadász 	struct mmc_command appcmd;
112*85ccd313SImre Vadász 	struct mmc_softc *sc;
113*85ccd313SImre Vadász 	int err;
114*85ccd313SImre Vadász 
115*85ccd313SImre Vadász 	sc = device_get_softc(brdev);
116*85ccd313SImre Vadász 
117*85ccd313SImre Vadász 	/* Squelch error reporting at lower levels, we report below. */
118*85ccd313SImre Vadász 	sc->squelched++;
119*85ccd313SImre Vadász 	do {
120*85ccd313SImre Vadász 		memset(&appcmd, 0, sizeof(appcmd));
121*85ccd313SImre Vadász 		appcmd.opcode = MMC_APP_CMD;
122*85ccd313SImre Vadász 		appcmd.arg = (uint32_t)rca << 16;
123*85ccd313SImre Vadász 		appcmd.flags = MMC_RSP_R1 | MMC_CMD_AC;
124*85ccd313SImre Vadász 		if (mmc_wait_for_cmd(brdev, reqdev, &appcmd, 0) != 0)
125*85ccd313SImre Vadász 			err = MMC_ERR_FAILED;
126*85ccd313SImre Vadász 		else
127*85ccd313SImre Vadász 			err = appcmd.error;
128*85ccd313SImre Vadász 		if (err == MMC_ERR_NONE) {
129*85ccd313SImre Vadász 			if (!(appcmd.resp[0] & R1_APP_CMD))
130*85ccd313SImre Vadász 				err = MMC_ERR_FAILED;
131*85ccd313SImre Vadász 			else if (mmc_wait_for_cmd(brdev, reqdev, cmd, 0) != 0)
132*85ccd313SImre Vadász 				err = MMC_ERR_FAILED;
133*85ccd313SImre Vadász 			else
134*85ccd313SImre Vadász 				err = cmd->error;
135*85ccd313SImre Vadász 		}
136*85ccd313SImre Vadász 	} while (err != MMC_ERR_NONE && retries-- > 0);
137*85ccd313SImre Vadász 	sc->squelched--;
138*85ccd313SImre Vadász 
139*85ccd313SImre Vadász 	if (err != MMC_ERR_NONE && brdev == reqdev) {
140*85ccd313SImre Vadász 		sc = device_get_softc(brdev);
141*85ccd313SImre Vadász 		if (sc->squelched == 0 && ppsratecheck(&sc->log_time,
142*85ccd313SImre Vadász 		    &sc->log_count, LOG_PPS)) {
143*85ccd313SImre Vadász 			device_printf(sc->dev, "ACMD%d failed, RESULT: %d\n",
144*85ccd313SImre Vadász 			    cmd->opcode, err);
145*85ccd313SImre Vadász 		}
146*85ccd313SImre Vadász 	}
147*85ccd313SImre Vadász 
148*85ccd313SImre Vadász 	return (err);
149*85ccd313SImre Vadász }
150*85ccd313SImre Vadász 
151*85ccd313SImre Vadász int
mmc_switch(device_t brdev,device_t reqdev,uint16_t rca,uint8_t set,uint8_t index,uint8_t value,u_int timeout,bool status)152*85ccd313SImre Vadász mmc_switch(device_t brdev, device_t reqdev, uint16_t rca, uint8_t set,
153*85ccd313SImre Vadász     uint8_t index, uint8_t value, u_int timeout, bool status)
154*85ccd313SImre Vadász {
155*85ccd313SImre Vadász 	struct mmc_command cmd;
156*85ccd313SImre Vadász 	int err;
157*85ccd313SImre Vadász 
158*85ccd313SImre Vadász 	KASSERT(timeout != 0, ("%s: no timeout", __func__));
159*85ccd313SImre Vadász 
160*85ccd313SImre Vadász 	memset(&cmd, 0, sizeof(cmd));
161*85ccd313SImre Vadász 	cmd.opcode = MMC_SWITCH_FUNC;
162*85ccd313SImre Vadász 	cmd.arg = (MMC_SWITCH_FUNC_WR << 24) | (index << 16) | (value << 8) |
163*85ccd313SImre Vadász 	    set;
164*85ccd313SImre Vadász 	/*
165*85ccd313SImre Vadász 	 * If the hardware supports busy detection but the switch timeout
166*85ccd313SImre Vadász 	 * exceeds the maximum host timeout, use a R1 instead of a R1B
167*85ccd313SImre Vadász 	 * response in order to keep the hardware from timing out.
168*85ccd313SImre Vadász 	 */
169*85ccd313SImre Vadász 	if (mmcbr_get_caps(brdev) & MMC_CAP_WAIT_WHILE_BUSY &&
170*85ccd313SImre Vadász 	    timeout > mmcbr_get_max_busy_timeout(brdev))
171*85ccd313SImre Vadász 		cmd.flags = MMC_RSP_R1 | MMC_CMD_AC;
172*85ccd313SImre Vadász 	else
173*85ccd313SImre Vadász 		cmd.flags = MMC_RSP_R1B | MMC_CMD_AC;
174*85ccd313SImre Vadász 	err = mmc_wait_for_cmd(brdev, reqdev, &cmd, CMD_RETRIES);
175*85ccd313SImre Vadász 	if (err != MMC_ERR_NONE || status == false)
176*85ccd313SImre Vadász 		return (err);
177*85ccd313SImre Vadász 	return (mmc_switch_status(brdev, reqdev, rca, timeout));
178*85ccd313SImre Vadász }
179*85ccd313SImre Vadász 
180*85ccd313SImre Vadász int
mmc_switch_status(device_t brdev,device_t reqdev,uint16_t rca,u_int timeout)181*85ccd313SImre Vadász mmc_switch_status(device_t brdev, device_t reqdev, uint16_t rca, u_int timeout)
182*85ccd313SImre Vadász {
183*85ccd313SImre Vadász 	struct timeval cur, end;
184*85ccd313SImre Vadász 	int err;
185*85ccd313SImre Vadász 	uint32_t status;
186*85ccd313SImre Vadász 
187*85ccd313SImre Vadász 	KASSERT(timeout != 0, ("%s: no timeout", __func__));
188*85ccd313SImre Vadász 
189*85ccd313SImre Vadász 	/*
190*85ccd313SImre Vadász 	 * Note that when using a R1B response in mmc_switch(), bridges of
191*85ccd313SImre Vadász 	 * type MMC_CAP_WAIT_WHILE_BUSY will issue mmc_send_status() only
192*85ccd313SImre Vadász 	 * once and then exit the loop.
193*85ccd313SImre Vadász 	 */
194*85ccd313SImre Vadász 	for (;;) {
195*85ccd313SImre Vadász 		err = mmc_send_status(brdev, reqdev, rca, &status);
196*85ccd313SImre Vadász 		if (err != MMC_ERR_NONE)
197*85ccd313SImre Vadász 			break;
198*85ccd313SImre Vadász 		if (R1_CURRENT_STATE(status) == R1_STATE_TRAN)
199*85ccd313SImre Vadász 			break;
200*85ccd313SImre Vadász 		getmicrouptime(&cur);
201*85ccd313SImre Vadász 		if (end.tv_sec == 0 && end.tv_usec == 0) {
202*85ccd313SImre Vadász 			end.tv_usec = timeout;
203*85ccd313SImre Vadász 			timevaladd(&end, &cur);
204*85ccd313SImre Vadász 		}
205*85ccd313SImre Vadász 		if (timevalcmp(&cur, &end, >)) {
206*85ccd313SImre Vadász 			err = MMC_ERR_TIMEOUT;
207*85ccd313SImre Vadász 			break;
208*85ccd313SImre Vadász 		}
209*85ccd313SImre Vadász 	}
210*85ccd313SImre Vadász 	if (err == MMC_ERR_NONE && R1_CURRENT_STATE(status) == R1_SWITCH_ERROR)
211*85ccd313SImre Vadász 		return (MMC_ERR_FAILED);
212*85ccd313SImre Vadász 	return (err);
213*85ccd313SImre Vadász }
214*85ccd313SImre Vadász 
215*85ccd313SImre Vadász int
mmc_send_ext_csd(device_t brdev,device_t reqdev,uint8_t * rawextcsd)216*85ccd313SImre Vadász mmc_send_ext_csd(device_t brdev, device_t reqdev, uint8_t *rawextcsd)
217*85ccd313SImre Vadász {
218*85ccd313SImre Vadász 	struct mmc_command cmd;
219*85ccd313SImre Vadász 	struct mmc_data data;
220*85ccd313SImre Vadász 	int err;
221*85ccd313SImre Vadász 
222*85ccd313SImre Vadász 	memset(&cmd, 0, sizeof(cmd));
223*85ccd313SImre Vadász 	memset(&data, 0, sizeof(data));
224*85ccd313SImre Vadász 
225*85ccd313SImre Vadász 	memset(rawextcsd, 0, MMC_EXTCSD_SIZE);
226*85ccd313SImre Vadász 	cmd.opcode = MMC_SEND_EXT_CSD;
227*85ccd313SImre Vadász 	cmd.flags = MMC_RSP_R1 | MMC_CMD_ADTC;
228*85ccd313SImre Vadász 	cmd.data = &data;
229*85ccd313SImre Vadász 
230*85ccd313SImre Vadász 	data.data = rawextcsd;
231*85ccd313SImre Vadász 	data.len = MMC_EXTCSD_SIZE;
232*85ccd313SImre Vadász 	data.flags = MMC_DATA_READ;
233*85ccd313SImre Vadász 
234*85ccd313SImre Vadász 	err = mmc_wait_for_cmd(brdev, reqdev, &cmd, CMD_RETRIES);
235*85ccd313SImre Vadász 	return (err);
236*85ccd313SImre Vadász }
237*85ccd313SImre Vadász 
238*85ccd313SImre Vadász int
mmc_send_status(device_t brdev,device_t reqdev,uint16_t rca,uint32_t * status)239*85ccd313SImre Vadász mmc_send_status(device_t brdev, device_t reqdev, uint16_t rca, uint32_t *status)
240*85ccd313SImre Vadász {
241*85ccd313SImre Vadász 	struct mmc_command cmd;
242*85ccd313SImre Vadász 	int err;
243*85ccd313SImre Vadász 
244*85ccd313SImre Vadász 	memset(&cmd, 0, sizeof(cmd));
245*85ccd313SImre Vadász 	cmd.opcode = MMC_SEND_STATUS;
246*85ccd313SImre Vadász 	cmd.arg = (uint32_t)rca << 16;
247*85ccd313SImre Vadász 	cmd.flags = MMC_RSP_R1 | MMC_CMD_AC;
248*85ccd313SImre Vadász 	err = mmc_wait_for_cmd(brdev, reqdev, &cmd, CMD_RETRIES);
249*85ccd313SImre Vadász 	*status = cmd.resp[0];
250*85ccd313SImre Vadász 	return (err);
251*85ccd313SImre Vadász }
252