xref: /freebsd/sys/dev/sdio/sdiob.c (revision 5b56413d04e608379c9a306373554a8e4d321bc0)
167ca7330SBjoern A. Zeeb /*-
267ca7330SBjoern A. Zeeb  * Copyright (c) 2017 Ilya Bakulin.  All rights reserved.
367ca7330SBjoern A. Zeeb  * Copyright (c) 2018-2019 The FreeBSD Foundation
467ca7330SBjoern A. Zeeb  *
567ca7330SBjoern A. Zeeb  * Portions of this software were developed by Björn Zeeb
667ca7330SBjoern A. Zeeb  * under sponsorship from the FreeBSD Foundation.
767ca7330SBjoern A. Zeeb  *
867ca7330SBjoern A. Zeeb  * Redistribution and use in source and binary forms, with or without
967ca7330SBjoern A. Zeeb  * modification, are permitted provided that the following conditions
1067ca7330SBjoern A. Zeeb  * are met:
1167ca7330SBjoern A. Zeeb  * 1. Redistributions of source code must retain the above copyright
1267ca7330SBjoern A. Zeeb  *    notice, this list of conditions and the following disclaimer.
1367ca7330SBjoern A. Zeeb  * 2. Redistributions in binary form must reproduce the above copyright
1467ca7330SBjoern A. Zeeb  *    notice, this list of conditions and the following disclaimer in the
1567ca7330SBjoern A. Zeeb  *    documentation and/or other materials provided with the distribution.
1667ca7330SBjoern A. Zeeb  *
1767ca7330SBjoern A. Zeeb  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
1867ca7330SBjoern A. Zeeb  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
1967ca7330SBjoern A. Zeeb  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
2067ca7330SBjoern A. Zeeb  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
2167ca7330SBjoern A. Zeeb  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
2267ca7330SBjoern A. Zeeb  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
2367ca7330SBjoern A. Zeeb  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
2467ca7330SBjoern A. Zeeb  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
2567ca7330SBjoern A. Zeeb  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
2667ca7330SBjoern A. Zeeb  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2767ca7330SBjoern A. Zeeb  *
2867ca7330SBjoern A. Zeeb  *
2967ca7330SBjoern A. Zeeb  * Portions of this software may have been developed with reference to
3067ca7330SBjoern A. Zeeb  * the SD Simplified Specification.  The following disclaimer may apply:
3167ca7330SBjoern A. Zeeb  *
3267ca7330SBjoern A. Zeeb  * The following conditions apply to the release of the simplified
3367ca7330SBjoern A. Zeeb  * specification ("Simplified Specification") by the SD Card Association and
3467ca7330SBjoern A. Zeeb  * the SD Group. The Simplified Specification is a subset of the complete SD
3567ca7330SBjoern A. Zeeb  * Specification which is owned by the SD Card Association and the SD
3667ca7330SBjoern A. Zeeb  * Group. This Simplified Specification is provided on a non-confidential
3767ca7330SBjoern A. Zeeb  * basis subject to the disclaimers below. Any implementation of the
3867ca7330SBjoern A. Zeeb  * Simplified Specification may require a license from the SD Card
3967ca7330SBjoern A. Zeeb  * Association, SD Group, SD-3C LLC or other third parties.
4067ca7330SBjoern A. Zeeb  *
4167ca7330SBjoern A. Zeeb  * Disclaimers:
4267ca7330SBjoern A. Zeeb  *
4367ca7330SBjoern A. Zeeb  * The information contained in the Simplified Specification is presented only
4467ca7330SBjoern A. Zeeb  * as a standard specification for SD Cards and SD Host/Ancillary products and
4567ca7330SBjoern A. Zeeb  * is provided "AS-IS" without any representations or warranties of any
4667ca7330SBjoern A. Zeeb  * kind. No responsibility is assumed by the SD Group, SD-3C LLC or the SD
4767ca7330SBjoern A. Zeeb  * Card Association for any damages, any infringements of patents or other
4867ca7330SBjoern A. Zeeb  * right of the SD Group, SD-3C LLC, the SD Card Association or any third
4967ca7330SBjoern A. Zeeb  * parties, which may result from its use. No license is granted by
5067ca7330SBjoern A. Zeeb  * implication, estoppel or otherwise under any patent or other rights of the
5167ca7330SBjoern A. Zeeb  * SD Group, SD-3C LLC, the SD Card Association or any third party. Nothing
5267ca7330SBjoern A. Zeeb  * herein shall be construed as an obligation by the SD Group, the SD-3C LLC
5367ca7330SBjoern A. Zeeb  * or the SD Card Association to disclose or distribute any technical
5467ca7330SBjoern A. Zeeb  * information, know-how or other confidential information to any third party.
5567ca7330SBjoern A. Zeeb  */
5667ca7330SBjoern A. Zeeb /*
5767ca7330SBjoern A. Zeeb  * Implements the (kernel specific) SDIO parts.
5867ca7330SBjoern A. Zeeb  * This will hide all cam(4) functionality from the SDIO driver implementations
5967ca7330SBjoern A. Zeeb  * which will just be newbus/device(9) and hence look like any other driver for,
6067ca7330SBjoern A. Zeeb  * e.g., PCI.
6167ca7330SBjoern A. Zeeb  * The sdiob(4) parts effetively "translate" between the two worlds "bridging"
6267ca7330SBjoern A. Zeeb  * messages from MMCCAM to newbus and back.
6367ca7330SBjoern A. Zeeb  */
6467ca7330SBjoern A. Zeeb 
6567ca7330SBjoern A. Zeeb #include <sys/param.h>
6667ca7330SBjoern A. Zeeb #include <sys/systm.h>
6767ca7330SBjoern A. Zeeb #include <sys/types.h>
6867ca7330SBjoern A. Zeeb #include <sys/kernel.h>
6967ca7330SBjoern A. Zeeb #include <sys/bus.h>
7067ca7330SBjoern A. Zeeb #include <sys/endian.h>
7167ca7330SBjoern A. Zeeb #include <sys/lock.h>
7267ca7330SBjoern A. Zeeb #include <sys/malloc.h>
7367ca7330SBjoern A. Zeeb #include <sys/module.h>
7467ca7330SBjoern A. Zeeb #include <sys/mutex.h>
7567ca7330SBjoern A. Zeeb 
7667ca7330SBjoern A. Zeeb #include <cam/cam.h>
7767ca7330SBjoern A. Zeeb #include <cam/cam_ccb.h>
7867ca7330SBjoern A. Zeeb #include <cam/cam_queue.h>
7967ca7330SBjoern A. Zeeb #include <cam/cam_periph.h>
8067ca7330SBjoern A. Zeeb #include <cam/cam_xpt.h>
8167ca7330SBjoern A. Zeeb #include <cam/cam_xpt_periph.h>
8267ca7330SBjoern A. Zeeb #include <cam/cam_xpt_internal.h> /* for cam_path */
8367ca7330SBjoern A. Zeeb #include <cam/cam_debug.h>
8467ca7330SBjoern A. Zeeb 
8567ca7330SBjoern A. Zeeb #include <dev/mmc/mmcreg.h>
8667ca7330SBjoern A. Zeeb 
8767ca7330SBjoern A. Zeeb #include <dev/sdio/sdiob.h>
8867ca7330SBjoern A. Zeeb #include <dev/sdio/sdio_subr.h>
8967ca7330SBjoern A. Zeeb 
9067ca7330SBjoern A. Zeeb #include "sdio_if.h"
9167ca7330SBjoern A. Zeeb 
9267ca7330SBjoern A. Zeeb #ifdef DEBUG
9367ca7330SBjoern A. Zeeb #define	DPRINTF(...)		printf(__VA_ARGS__)
9467ca7330SBjoern A. Zeeb #define	DPRINTFDEV(_dev, ...)	device_printf((_dev), __VA_ARGS__)
9567ca7330SBjoern A. Zeeb #else
9667ca7330SBjoern A. Zeeb #define	DPRINTF(...)
9767ca7330SBjoern A. Zeeb #define	DPRINTFDEV(_dev, ...)
9867ca7330SBjoern A. Zeeb #endif
9967ca7330SBjoern A. Zeeb 
10067ca7330SBjoern A. Zeeb struct sdiob_softc {
10167ca7330SBjoern A. Zeeb 	uint32_t			sdio_state;
10267ca7330SBjoern A. Zeeb #define	SDIO_STATE_DEAD			0x0001
10367ca7330SBjoern A. Zeeb #define	SDIO_STATE_INITIALIZING		0x0002
10467ca7330SBjoern A. Zeeb #define	SDIO_STATE_READY		0x0004
10567ca7330SBjoern A. Zeeb 	uint32_t			nb_state;
10667ca7330SBjoern A. Zeeb #define	NB_STATE_DEAD			0x0001
10767ca7330SBjoern A. Zeeb #define	NB_STATE_SIM_ADDED		0x0002
10867ca7330SBjoern A. Zeeb #define	NB_STATE_READY			0x0004
10967ca7330SBjoern A. Zeeb 
110fdd60a97SWarner Losh 	/* CAM side. */
11167ca7330SBjoern A. Zeeb 	struct card_info		cardinfo;
11267ca7330SBjoern A. Zeeb 	struct cam_periph		*periph;
11367ca7330SBjoern A. Zeeb 	union ccb			*ccb;
11467ca7330SBjoern A. Zeeb 	struct task			discover_task;
11567ca7330SBjoern A. Zeeb 
11667ca7330SBjoern A. Zeeb 	/* Newbus side. */
11767ca7330SBjoern A. Zeeb 	device_t			dev;	/* Ourselves. */
11867ca7330SBjoern A. Zeeb 	device_t			child[8];
11967ca7330SBjoern A. Zeeb };
12067ca7330SBjoern A. Zeeb 
12167ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
12267ca7330SBjoern A. Zeeb /*
12367ca7330SBjoern A. Zeeb  * SDIO CMD52 and CM53 implementations along with wrapper functions for
12467ca7330SBjoern A. Zeeb  * read/write and a CAM periph helper function.
12567ca7330SBjoern A. Zeeb  * These are the backend implementations of the sdio_if.m framework talking
12667ca7330SBjoern A. Zeeb  * through CAM to sdhci.
12767ca7330SBjoern A. Zeeb  * Note: these functions are also called during early discovery stage when
12867ca7330SBjoern A. Zeeb  * we are not a device(9) yet. Hence they cannot always use device_printf()
12967ca7330SBjoern A. Zeeb  * to log errors and have to call CAM_DEBUG() during these early stages.
13067ca7330SBjoern A. Zeeb  */
13167ca7330SBjoern A. Zeeb 
13267ca7330SBjoern A. Zeeb static int
13367ca7330SBjoern A. Zeeb sdioerror(union ccb *ccb, u_int32_t cam_flags, u_int32_t sense_flags)
13467ca7330SBjoern A. Zeeb {
13567ca7330SBjoern A. Zeeb 
13667ca7330SBjoern A. Zeeb 	return (cam_periph_error(ccb, cam_flags, sense_flags));
13767ca7330SBjoern A. Zeeb }
13867ca7330SBjoern A. Zeeb 
13967ca7330SBjoern A. Zeeb /* CMD52: direct byte access. */
14067ca7330SBjoern A. Zeeb static int
14167ca7330SBjoern A. Zeeb sdiob_rw_direct_sc(struct sdiob_softc *sc, uint8_t fn, uint32_t addr, bool wr,
14267ca7330SBjoern A. Zeeb     uint8_t *val)
14367ca7330SBjoern A. Zeeb {
14467ca7330SBjoern A. Zeeb 	uint32_t arg, flags;
14567ca7330SBjoern A. Zeeb 	int error;
14667ca7330SBjoern A. Zeeb 
14767ca7330SBjoern A. Zeeb 	KASSERT((val != NULL), ("%s val passed as NULL\n", __func__));
14867ca7330SBjoern A. Zeeb 
14967ca7330SBjoern A. Zeeb 	if (sc->ccb == NULL)
15067ca7330SBjoern A. Zeeb 		sc->ccb = xpt_alloc_ccb();
15167ca7330SBjoern A. Zeeb 	else
15267ca7330SBjoern A. Zeeb 		memset(sc->ccb, 0, sizeof(*sc->ccb));
15367ca7330SBjoern A. Zeeb 	xpt_setup_ccb(&sc->ccb->ccb_h, sc->periph->path, CAM_PRIORITY_NONE);
15467ca7330SBjoern A. Zeeb 	CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_TRACE,
15567ca7330SBjoern A. Zeeb 	    ("%s(fn=%d, addr=%#02x, wr=%d, *val=%#02x)\n", __func__,
15667ca7330SBjoern A. Zeeb 	    fn, addr, wr, *val));
15767ca7330SBjoern A. Zeeb 
15867ca7330SBjoern A. Zeeb 	flags = MMC_RSP_R5 | MMC_CMD_AC;
15967ca7330SBjoern A. Zeeb 	arg = SD_IO_RW_FUNC(fn) | SD_IO_RW_ADR(addr);
16067ca7330SBjoern A. Zeeb 	if (wr)
16167ca7330SBjoern A. Zeeb 		arg |= SD_IO_RW_WR | SD_IO_RW_RAW | SD_IO_RW_DAT(*val);
16267ca7330SBjoern A. Zeeb 
16367ca7330SBjoern A. Zeeb 	cam_fill_mmcio(&sc->ccb->mmcio,
16467ca7330SBjoern A. Zeeb 		/*retries*/ 0,
16567ca7330SBjoern A. Zeeb 		/*cbfcnp*/ NULL,
16667ca7330SBjoern A. Zeeb 		/*flags*/ CAM_DIR_NONE,
16767ca7330SBjoern A. Zeeb 		/*mmc_opcode*/ SD_IO_RW_DIRECT,
16867ca7330SBjoern A. Zeeb 		/*mmc_arg*/ arg,
16967ca7330SBjoern A. Zeeb 		/*mmc_flags*/ flags,
17067ca7330SBjoern A. Zeeb 		/*mmc_data*/ 0,
17167ca7330SBjoern A. Zeeb 		/*timeout*/ sc->cardinfo.f[fn].timeout);
17267ca7330SBjoern A. Zeeb 	error = cam_periph_runccb(sc->ccb, sdioerror, CAM_FLAG_NONE, 0, NULL);
17367ca7330SBjoern A. Zeeb 	if (error != 0) {
17467ca7330SBjoern A. Zeeb 		if (sc->dev != NULL)
17567ca7330SBjoern A. Zeeb 			device_printf(sc->dev,
17667ca7330SBjoern A. Zeeb 			    "%s: Failed to %s address %#10x error=%d\n",
17767ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write" : "read", addr, error);
17867ca7330SBjoern A. Zeeb 		else
17967ca7330SBjoern A. Zeeb 			CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_INFO,
18067ca7330SBjoern A. Zeeb 			    ("%s: Failed to %s address: %#10x error=%d\n",
18167ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write" : "read", addr, error));
18267ca7330SBjoern A. Zeeb 		return (error);
18367ca7330SBjoern A. Zeeb 	}
18467ca7330SBjoern A. Zeeb 
18567ca7330SBjoern A. Zeeb 	/* TODO: Add handling of MMC errors */
18667ca7330SBjoern A. Zeeb 	/* ccb->mmcio.cmd.error ? */
18767ca7330SBjoern A. Zeeb 	if (wr == false)
18867ca7330SBjoern A. Zeeb 		*val = sc->ccb->mmcio.cmd.resp[0] & 0xff;
18967ca7330SBjoern A. Zeeb 
19067ca7330SBjoern A. Zeeb 	return (0);
19167ca7330SBjoern A. Zeeb }
19267ca7330SBjoern A. Zeeb 
19367ca7330SBjoern A. Zeeb static int
19467ca7330SBjoern A. Zeeb sdio_rw_direct(device_t dev, uint8_t fn, uint32_t addr, bool wr,
19567ca7330SBjoern A. Zeeb     uint8_t *val)
19667ca7330SBjoern A. Zeeb {
19767ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
19867ca7330SBjoern A. Zeeb 	int error;
19967ca7330SBjoern A. Zeeb 
20067ca7330SBjoern A. Zeeb 	sc = device_get_softc(dev);
20167ca7330SBjoern A. Zeeb 	cam_periph_lock(sc->periph);
20267ca7330SBjoern A. Zeeb 	error = sdiob_rw_direct_sc(sc, fn, addr, wr, val);
20367ca7330SBjoern A. Zeeb 	cam_periph_unlock(sc->periph);
20467ca7330SBjoern A. Zeeb 	return (error);
20567ca7330SBjoern A. Zeeb }
20667ca7330SBjoern A. Zeeb 
20767ca7330SBjoern A. Zeeb static int
20867ca7330SBjoern A. Zeeb sdiob_read_direct(device_t dev, uint8_t fn, uint32_t addr, uint8_t *val)
20967ca7330SBjoern A. Zeeb {
21067ca7330SBjoern A. Zeeb 	int error;
21167ca7330SBjoern A. Zeeb 	uint8_t v;
21267ca7330SBjoern A. Zeeb 
21367ca7330SBjoern A. Zeeb 	error = sdio_rw_direct(dev, fn, addr, false, &v);
21467ca7330SBjoern A. Zeeb 	/* Be polite and do not touch the value on read error. */
21567ca7330SBjoern A. Zeeb 	if (error == 0 && val != NULL)
21667ca7330SBjoern A. Zeeb 		*val = v;
21767ca7330SBjoern A. Zeeb 	return (error);
21867ca7330SBjoern A. Zeeb }
21967ca7330SBjoern A. Zeeb 
22067ca7330SBjoern A. Zeeb static int
22167ca7330SBjoern A. Zeeb sdiob_write_direct(device_t dev, uint8_t fn, uint32_t addr, uint8_t val)
22267ca7330SBjoern A. Zeeb {
22367ca7330SBjoern A. Zeeb 
22467ca7330SBjoern A. Zeeb 	return (sdio_rw_direct(dev, fn, addr, true, &val));
22567ca7330SBjoern A. Zeeb }
22667ca7330SBjoern A. Zeeb 
22767ca7330SBjoern A. Zeeb /*
22867ca7330SBjoern A. Zeeb  * CMD53: IO_RW_EXTENDED, read and write multiple I/O registers.
22967ca7330SBjoern A. Zeeb  * Increment false gets FIFO mode (single register address).
23067ca7330SBjoern A. Zeeb  */
23167ca7330SBjoern A. Zeeb /*
23267ca7330SBjoern A. Zeeb  * A b_count of 0 means byte mode, b_count > 0 gets block mode.
23367ca7330SBjoern A. Zeeb  * A b_count of >= 512 would mean infinitive block transfer, which would become
23467ca7330SBjoern A. Zeeb  * b_count = 0, is not yet supported.
23567ca7330SBjoern A. Zeeb  * For b_count == 0, blksz is the len of bytes, otherwise it is the amount of
23667ca7330SBjoern A. Zeeb  * full sized blocks (you must not round the blocks up and leave the last one
23767ca7330SBjoern A. Zeeb  * partial!)
23867ca7330SBjoern A. Zeeb  * For byte mode, the maximum of blksz is the functions cur_blksize.
23967ca7330SBjoern A. Zeeb  * This function should ever only be called by sdio_rw_extended_sc()!
24067ca7330SBjoern A. Zeeb  */
24167ca7330SBjoern A. Zeeb static int
24267ca7330SBjoern A. Zeeb sdiob_rw_extended_cam(struct sdiob_softc *sc, uint8_t fn, uint32_t addr,
24367ca7330SBjoern A. Zeeb     bool wr, uint8_t *buffer, bool incaddr, uint32_t b_count, uint16_t blksz)
24467ca7330SBjoern A. Zeeb {
24567ca7330SBjoern A. Zeeb 	struct mmc_data mmcd;
24667ca7330SBjoern A. Zeeb 	uint32_t arg, cam_flags, flags, len;
24767ca7330SBjoern A. Zeeb 	int error;
24867ca7330SBjoern A. Zeeb 
24967ca7330SBjoern A. Zeeb 	if (sc->ccb == NULL)
25067ca7330SBjoern A. Zeeb 		sc->ccb = xpt_alloc_ccb();
25167ca7330SBjoern A. Zeeb 	else
25267ca7330SBjoern A. Zeeb 		memset(sc->ccb, 0, sizeof(*sc->ccb));
25367ca7330SBjoern A. Zeeb 	xpt_setup_ccb(&sc->ccb->ccb_h, sc->periph->path, CAM_PRIORITY_NONE);
25467ca7330SBjoern A. Zeeb 	CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_TRACE,
25567ca7330SBjoern A. Zeeb 	    ("%s(fn=%d addr=%#0x wr=%d b_count=%u blksz=%u buf=%p incr=%d)\n",
25667ca7330SBjoern A. Zeeb 	    __func__, fn, addr, wr, b_count, blksz, buffer, incaddr));
25767ca7330SBjoern A. Zeeb 
25867ca7330SBjoern A. Zeeb 	KASSERT((b_count <= 511), ("%s: infinitive block transfer not yet "
25967ca7330SBjoern A. Zeeb 	    "supported: b_count %u blksz %u, sc %p, fn %u, addr %#10x, %s, "
26067ca7330SBjoern A. Zeeb 	    "buffer %p, %s\n", __func__, b_count, blksz, sc, fn, addr,
26167ca7330SBjoern A. Zeeb 	    wr ? "wr" : "rd", buffer, incaddr ? "incaddr" : "fifo"));
26267ca7330SBjoern A. Zeeb 	/* Blksz needs to be within bounds for both byte and block mode! */
26367ca7330SBjoern A. Zeeb 	KASSERT((blksz <= sc->cardinfo.f[fn].cur_blksize), ("%s: blksz "
26467ca7330SBjoern A. Zeeb 	    "%u > bur_blksize %u, sc %p, fn %u, addr %#10x, %s, "
26567ca7330SBjoern A. Zeeb 	    "buffer %p, %s, b_count %u\n", __func__, blksz,
26667ca7330SBjoern A. Zeeb 	    sc->cardinfo.f[fn].cur_blksize, sc, fn, addr,
26767ca7330SBjoern A. Zeeb 	    wr ? "wr" : "rd", buffer, incaddr ? "incaddr" : "fifo",
26867ca7330SBjoern A. Zeeb 	    b_count));
26967ca7330SBjoern A. Zeeb 	if (b_count == 0) {
27067ca7330SBjoern A. Zeeb 		/* Byte mode */
27167ca7330SBjoern A. Zeeb 		len = blksz;
27267ca7330SBjoern A. Zeeb 		if (blksz == 512)
27367ca7330SBjoern A. Zeeb 			blksz = 0;
27467ca7330SBjoern A. Zeeb 		arg = SD_IOE_RW_LEN(blksz);
27567ca7330SBjoern A. Zeeb 	} else {
27667ca7330SBjoern A. Zeeb 		/* Block mode. */
27767ca7330SBjoern A. Zeeb #ifdef __notyet__
27867ca7330SBjoern A. Zeeb 		if (b_count > 511) {
27967ca7330SBjoern A. Zeeb 			/* Infinitive block transfer. */
28067ca7330SBjoern A. Zeeb 			b_count = 0;
28167ca7330SBjoern A. Zeeb 		}
28267ca7330SBjoern A. Zeeb #endif
28367ca7330SBjoern A. Zeeb 		len = b_count * blksz;
28467ca7330SBjoern A. Zeeb 		arg = SD_IOE_RW_BLK | SD_IOE_RW_LEN(b_count);
28567ca7330SBjoern A. Zeeb 	}
28667ca7330SBjoern A. Zeeb 
28767ca7330SBjoern A. Zeeb 	flags = MMC_RSP_R5 | MMC_CMD_ADTC;
28867ca7330SBjoern A. Zeeb 	arg |= SD_IOE_RW_FUNC(fn) | SD_IOE_RW_ADR(addr);
28967ca7330SBjoern A. Zeeb 	if (incaddr)
29067ca7330SBjoern A. Zeeb 		arg |= SD_IOE_RW_INCR;
29167ca7330SBjoern A. Zeeb 
29267ca7330SBjoern A. Zeeb 	memset(&mmcd, 0, sizeof(mmcd));
29367ca7330SBjoern A. Zeeb 	mmcd.data = buffer;
29467ca7330SBjoern A. Zeeb 	mmcd.len = len;
29567ca7330SBjoern A. Zeeb 	if (arg & SD_IOE_RW_BLK) {
29667ca7330SBjoern A. Zeeb 		/* XXX both should be known from elsewhere, aren't they? */
29767ca7330SBjoern A. Zeeb 		mmcd.block_size = blksz;
29867ca7330SBjoern A. Zeeb 		mmcd.block_count = b_count;
29967ca7330SBjoern A. Zeeb 	}
30067ca7330SBjoern A. Zeeb 
30167ca7330SBjoern A. Zeeb 	if (wr) {
30267ca7330SBjoern A. Zeeb 		arg |= SD_IOE_RW_WR;
30367ca7330SBjoern A. Zeeb 		cam_flags = CAM_DIR_OUT;
30467ca7330SBjoern A. Zeeb 		mmcd.flags = MMC_DATA_WRITE;
30567ca7330SBjoern A. Zeeb 	} else {
30667ca7330SBjoern A. Zeeb 		cam_flags = CAM_DIR_IN;
30767ca7330SBjoern A. Zeeb 		mmcd.flags = MMC_DATA_READ;
30867ca7330SBjoern A. Zeeb 	}
30967ca7330SBjoern A. Zeeb #ifdef __notyet__
31067ca7330SBjoern A. Zeeb 	if (b_count == 0) {
31167ca7330SBjoern A. Zeeb 		/* XXX-BZ TODO FIXME.  Cancel I/O: CCCR -> ASx */
31267ca7330SBjoern A. Zeeb 		/* Stop cmd. */
31367ca7330SBjoern A. Zeeb 	}
31467ca7330SBjoern A. Zeeb #endif
31567ca7330SBjoern A. Zeeb 	cam_fill_mmcio(&sc->ccb->mmcio,
31667ca7330SBjoern A. Zeeb 		/*retries*/ 0,
31767ca7330SBjoern A. Zeeb 		/*cbfcnp*/ NULL,
31867ca7330SBjoern A. Zeeb 		/*flags*/ cam_flags,
31967ca7330SBjoern A. Zeeb 		/*mmc_opcode*/ SD_IO_RW_EXTENDED,
32067ca7330SBjoern A. Zeeb 		/*mmc_arg*/ arg,
32167ca7330SBjoern A. Zeeb 		/*mmc_flags*/ flags,
32267ca7330SBjoern A. Zeeb 		/*mmc_data*/ &mmcd,
32367ca7330SBjoern A. Zeeb 		/*timeout*/ sc->cardinfo.f[fn].timeout);
32467ca7330SBjoern A. Zeeb 	if (arg & SD_IOE_RW_BLK) {
32567ca7330SBjoern A. Zeeb 		mmcd.flags |= MMC_DATA_BLOCK_SIZE;
32667ca7330SBjoern A. Zeeb 		if (b_count != 1)
32767ca7330SBjoern A. Zeeb 			sc->ccb->mmcio.cmd.data->flags |= MMC_DATA_MULTI;
32867ca7330SBjoern A. Zeeb 	}
32967ca7330SBjoern A. Zeeb 
33067ca7330SBjoern A. Zeeb 	/* Execute. */
33167ca7330SBjoern A. Zeeb 	error = cam_periph_runccb(sc->ccb, sdioerror, CAM_FLAG_NONE, 0, NULL);
33267ca7330SBjoern A. Zeeb 	if (error != 0) {
33367ca7330SBjoern A. Zeeb 		if (sc->dev != NULL)
33467ca7330SBjoern A. Zeeb 			device_printf(sc->dev,
33567ca7330SBjoern A. Zeeb 			    "%s: Failed to %s address %#10x buffer %p size %u "
33667ca7330SBjoern A. Zeeb 			    "%s b_count %u blksz %u error=%d\n",
33767ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write to" : "read from", addr,
33867ca7330SBjoern A. Zeeb 			    buffer, len, (incaddr) ? "incr" : "fifo",
33967ca7330SBjoern A. Zeeb 			    b_count, blksz, error);
34067ca7330SBjoern A. Zeeb 		else
34167ca7330SBjoern A. Zeeb 			CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_INFO,
34267ca7330SBjoern A. Zeeb 			    ("%s: Failed to %s address %#10x buffer %p size %u "
34367ca7330SBjoern A. Zeeb 			    "%s b_count %u blksz %u error=%d\n",
34467ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write to" : "read from", addr,
34567ca7330SBjoern A. Zeeb 			    buffer, len, (incaddr) ? "incr" : "fifo",
34667ca7330SBjoern A. Zeeb 			    b_count, blksz, error));
34767ca7330SBjoern A. Zeeb 		return (error);
34867ca7330SBjoern A. Zeeb 	}
34967ca7330SBjoern A. Zeeb 
35067ca7330SBjoern A. Zeeb 	/* TODO: Add handling of MMC errors */
35167ca7330SBjoern A. Zeeb 	/* ccb->mmcio.cmd.error ? */
35267ca7330SBjoern A. Zeeb 	error = sc->ccb->mmcio.cmd.resp[0] & 0xff;
35367ca7330SBjoern A. Zeeb 	if (error != 0) {
35467ca7330SBjoern A. Zeeb 		if (sc->dev != NULL)
35567ca7330SBjoern A. Zeeb 			device_printf(sc->dev,
35667ca7330SBjoern A. Zeeb 			    "%s: Failed to %s address %#10x buffer %p size %u "
35767ca7330SBjoern A. Zeeb 			    "%s b_count %u blksz %u mmcio resp error=%d\n",
35867ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write to" : "read from", addr,
35967ca7330SBjoern A. Zeeb 			    buffer, len, (incaddr) ? "incr" : "fifo",
36067ca7330SBjoern A. Zeeb 			    b_count, blksz, error);
36167ca7330SBjoern A. Zeeb 		else
36267ca7330SBjoern A. Zeeb 			CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_INFO,
36367ca7330SBjoern A. Zeeb 			    ("%s: Failed to %s address %#10x buffer %p size %u "
36467ca7330SBjoern A. Zeeb 			    "%s b_count %u blksz %u mmcio resp error=%d\n",
36567ca7330SBjoern A. Zeeb 			    __func__, (wr) ? "write to" : "read from", addr,
36667ca7330SBjoern A. Zeeb 			    buffer, len, (incaddr) ? "incr" : "fifo",
36767ca7330SBjoern A. Zeeb 			    b_count, blksz, error));
36867ca7330SBjoern A. Zeeb 	}
36967ca7330SBjoern A. Zeeb 	return (error);
37067ca7330SBjoern A. Zeeb }
37167ca7330SBjoern A. Zeeb 
37267ca7330SBjoern A. Zeeb static int
37367ca7330SBjoern A. Zeeb sdiob_rw_extended_sc(struct sdiob_softc *sc, uint8_t fn, uint32_t addr,
37467ca7330SBjoern A. Zeeb     bool wr, uint32_t size, uint8_t *buffer, bool incaddr)
37567ca7330SBjoern A. Zeeb {
37667ca7330SBjoern A. Zeeb 	int error;
37767ca7330SBjoern A. Zeeb 	uint32_t len;
37867ca7330SBjoern A. Zeeb 	uint32_t b_count;
37967ca7330SBjoern A. Zeeb 
38067ca7330SBjoern A. Zeeb 	/*
38167ca7330SBjoern A. Zeeb 	 * If block mode is supported and we have at least 4 bytes to write and
38267ca7330SBjoern A. Zeeb 	 * the size is at least one block, then start doing blk transfers.
38367ca7330SBjoern A. Zeeb 	 */
38467ca7330SBjoern A. Zeeb 	while (sc->cardinfo.support_multiblk &&
38567ca7330SBjoern A. Zeeb 	    size > 4 && size >= sc->cardinfo.f[fn].cur_blksize) {
38667ca7330SBjoern A. Zeeb 		b_count = size / sc->cardinfo.f[fn].cur_blksize;
38767ca7330SBjoern A. Zeeb 		KASSERT(b_count >= 1, ("%s: block count too small %u size %u "
38867ca7330SBjoern A. Zeeb 		    "cur_blksize %u\n", __func__, b_count, size,
38967ca7330SBjoern A. Zeeb 		    sc->cardinfo.f[fn].cur_blksize));
39067ca7330SBjoern A. Zeeb 
39167ca7330SBjoern A. Zeeb #ifdef __notyet__
39267ca7330SBjoern A. Zeeb 		/* XXX support inifinite transfer with b_count = 0. */
39367ca7330SBjoern A. Zeeb #else
39467ca7330SBjoern A. Zeeb 		if (b_count > 511)
39567ca7330SBjoern A. Zeeb 			b_count = 511;
39667ca7330SBjoern A. Zeeb #endif
39767ca7330SBjoern A. Zeeb 		len = b_count * sc->cardinfo.f[fn].cur_blksize;
39867ca7330SBjoern A. Zeeb 		error = sdiob_rw_extended_cam(sc, fn, addr, wr, buffer, incaddr,
39967ca7330SBjoern A. Zeeb 		    b_count, sc->cardinfo.f[fn].cur_blksize);
40067ca7330SBjoern A. Zeeb 		if (error != 0)
40167ca7330SBjoern A. Zeeb 			return (error);
40267ca7330SBjoern A. Zeeb 
40367ca7330SBjoern A. Zeeb 		size -= len;
40467ca7330SBjoern A. Zeeb 		buffer += len;
40567ca7330SBjoern A. Zeeb 		if (incaddr)
40667ca7330SBjoern A. Zeeb 			addr += len;
40767ca7330SBjoern A. Zeeb 	}
40867ca7330SBjoern A. Zeeb 
40967ca7330SBjoern A. Zeeb 	while (size > 0) {
41067ca7330SBjoern A. Zeeb 		len = MIN(size, sc->cardinfo.f[fn].cur_blksize);
41167ca7330SBjoern A. Zeeb 
41267ca7330SBjoern A. Zeeb 		error = sdiob_rw_extended_cam(sc, fn, addr, wr, buffer, incaddr,
41367ca7330SBjoern A. Zeeb 		    0, len);
41467ca7330SBjoern A. Zeeb 		if (error != 0)
41567ca7330SBjoern A. Zeeb 			return (error);
41667ca7330SBjoern A. Zeeb 
41767ca7330SBjoern A. Zeeb 		/* Prepare for next iteration. */
41867ca7330SBjoern A. Zeeb 		size -= len;
41967ca7330SBjoern A. Zeeb 		buffer += len;
42067ca7330SBjoern A. Zeeb 		if (incaddr)
42167ca7330SBjoern A. Zeeb 			addr += len;
42267ca7330SBjoern A. Zeeb 	}
42367ca7330SBjoern A. Zeeb 
42467ca7330SBjoern A. Zeeb 	return (0);
42567ca7330SBjoern A. Zeeb }
42667ca7330SBjoern A. Zeeb 
42767ca7330SBjoern A. Zeeb static int
42867ca7330SBjoern A. Zeeb sdiob_rw_extended(device_t dev, uint8_t fn, uint32_t addr, bool wr,
42967ca7330SBjoern A. Zeeb     uint32_t size, uint8_t *buffer, bool incaddr)
43067ca7330SBjoern A. Zeeb {
43167ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
43267ca7330SBjoern A. Zeeb 	int error;
43367ca7330SBjoern A. Zeeb 
43467ca7330SBjoern A. Zeeb 	sc = device_get_softc(dev);
43567ca7330SBjoern A. Zeeb 	cam_periph_lock(sc->periph);
43667ca7330SBjoern A. Zeeb 	error = sdiob_rw_extended_sc(sc, fn, addr, wr, size, buffer, incaddr);
43767ca7330SBjoern A. Zeeb 	cam_periph_unlock(sc->periph);
43867ca7330SBjoern A. Zeeb 	return (error);
43967ca7330SBjoern A. Zeeb }
44067ca7330SBjoern A. Zeeb 
44167ca7330SBjoern A. Zeeb static int
44267ca7330SBjoern A. Zeeb sdiob_read_extended(device_t dev, uint8_t fn, uint32_t addr, uint32_t size,
44367ca7330SBjoern A. Zeeb     uint8_t *buffer, bool incaddr)
44467ca7330SBjoern A. Zeeb {
44567ca7330SBjoern A. Zeeb 
44667ca7330SBjoern A. Zeeb 	return (sdiob_rw_extended(dev, fn, addr, false, size, buffer, incaddr));
44767ca7330SBjoern A. Zeeb }
44867ca7330SBjoern A. Zeeb 
44967ca7330SBjoern A. Zeeb static int
45067ca7330SBjoern A. Zeeb sdiob_write_extended(device_t dev, uint8_t fn, uint32_t addr, uint32_t size,
45167ca7330SBjoern A. Zeeb     uint8_t *buffer, bool incaddr)
45267ca7330SBjoern A. Zeeb {
45367ca7330SBjoern A. Zeeb 
45467ca7330SBjoern A. Zeeb 	return (sdiob_rw_extended(dev, fn, addr, true, size, buffer, incaddr));
45567ca7330SBjoern A. Zeeb }
45667ca7330SBjoern A. Zeeb 
45767ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
45867ca7330SBjoern A. Zeeb /* Bus interface, ivars handling. */
45967ca7330SBjoern A. Zeeb 
46067ca7330SBjoern A. Zeeb static int
46167ca7330SBjoern A. Zeeb sdiob_read_ivar(device_t dev, device_t child, int which, uintptr_t *result)
46267ca7330SBjoern A. Zeeb {
46367ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
46467ca7330SBjoern A. Zeeb 	struct sdio_func *f;
46567ca7330SBjoern A. Zeeb 
46667ca7330SBjoern A. Zeeb 	f = device_get_ivars(child);
46767ca7330SBjoern A. Zeeb 	KASSERT(f != NULL, ("%s: dev %p child %p which %d, child ivars NULL\n",
46867ca7330SBjoern A. Zeeb 	    __func__, dev, child, which));
46967ca7330SBjoern A. Zeeb 
47067ca7330SBjoern A. Zeeb 	switch (which) {
47167ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_SUPPORT_MULTIBLK:
47267ca7330SBjoern A. Zeeb 		sc = device_get_softc(dev);
47367ca7330SBjoern A. Zeeb 		KASSERT(sc != NULL, ("%s: dev %p child %p which %d, sc NULL\n",
47467ca7330SBjoern A. Zeeb 		    __func__, dev, child, which));
47567ca7330SBjoern A. Zeeb 		*result = sc->cardinfo.support_multiblk;
47667ca7330SBjoern A. Zeeb 		break;
47767ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_FUNCTION:
47867ca7330SBjoern A. Zeeb 		*result = (uintptr_t)f;
47967ca7330SBjoern A. Zeeb 		break;
48067ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_FUNCNUM:
48167ca7330SBjoern A. Zeeb 		*result = f->fn;
48267ca7330SBjoern A. Zeeb 		break;
48367ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_CLASS:
48467ca7330SBjoern A. Zeeb 		*result = f->class;
48567ca7330SBjoern A. Zeeb 		break;
48667ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_VENDOR:
48767ca7330SBjoern A. Zeeb 		*result = f->vendor;
48867ca7330SBjoern A. Zeeb 		break;
48967ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_DEVICE:
49067ca7330SBjoern A. Zeeb 		*result = f->device;
49167ca7330SBjoern A. Zeeb 		break;
49267ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_DRVDATA:
49367ca7330SBjoern A. Zeeb 		*result = f->drvdata;
49467ca7330SBjoern A. Zeeb 		break;
49567ca7330SBjoern A. Zeeb 	default:
49667ca7330SBjoern A. Zeeb 		return (ENOENT);
49767ca7330SBjoern A. Zeeb 	}
49867ca7330SBjoern A. Zeeb 	return (0);
49967ca7330SBjoern A. Zeeb }
50067ca7330SBjoern A. Zeeb 
50167ca7330SBjoern A. Zeeb static int
50267ca7330SBjoern A. Zeeb sdiob_write_ivar(device_t dev, device_t child, int which, uintptr_t value)
50367ca7330SBjoern A. Zeeb {
50467ca7330SBjoern A. Zeeb 	struct sdio_func *f;
50567ca7330SBjoern A. Zeeb 
50667ca7330SBjoern A. Zeeb 	f = device_get_ivars(child);
50767ca7330SBjoern A. Zeeb 	KASSERT(f != NULL, ("%s: dev %p child %p which %d, child ivars NULL\n",
50867ca7330SBjoern A. Zeeb 	    __func__, dev, child, which));
50967ca7330SBjoern A. Zeeb 
51067ca7330SBjoern A. Zeeb 	switch (which) {
51167ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_SUPPORT_MULTIBLK:
51267ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_FUNCTION:
51367ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_FUNCNUM:
51467ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_CLASS:
51567ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_VENDOR:
51667ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_DEVICE:
51767ca7330SBjoern A. Zeeb 		return (EINVAL);	/* Disallowed. */
51867ca7330SBjoern A. Zeeb 	case SDIOB_IVAR_DRVDATA:
51967ca7330SBjoern A. Zeeb 		f->drvdata = value;
52067ca7330SBjoern A. Zeeb 		break;
52167ca7330SBjoern A. Zeeb 	default:
52267ca7330SBjoern A. Zeeb 		return (ENOENT);
52367ca7330SBjoern A. Zeeb 	}
52467ca7330SBjoern A. Zeeb 
52567ca7330SBjoern A. Zeeb 	return (0);
52667ca7330SBjoern A. Zeeb }
52767ca7330SBjoern A. Zeeb 
52867ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
52967ca7330SBjoern A. Zeeb /*
53067ca7330SBjoern A. Zeeb  * Newbus functions for ourselves to probe/attach/detach and become a proper
53167ca7330SBjoern A. Zeeb  * device(9).  Attach will also probe for child devices (another driver
53267ca7330SBjoern A. Zeeb  * implementing SDIO).
53367ca7330SBjoern A. Zeeb  */
53467ca7330SBjoern A. Zeeb 
53567ca7330SBjoern A. Zeeb static int
53667ca7330SBjoern A. Zeeb sdiob_probe(device_t dev)
53767ca7330SBjoern A. Zeeb {
53867ca7330SBjoern A. Zeeb 
53967ca7330SBjoern A. Zeeb 	device_set_desc(dev, "SDIO CAM-Newbus bridge");
54067ca7330SBjoern A. Zeeb 	return (BUS_PROBE_DEFAULT);
54167ca7330SBjoern A. Zeeb }
54267ca7330SBjoern A. Zeeb 
54367ca7330SBjoern A. Zeeb static int
54467ca7330SBjoern A. Zeeb sdiob_attach(device_t dev)
54567ca7330SBjoern A. Zeeb {
54667ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
54767ca7330SBjoern A. Zeeb 	int error, i;
54867ca7330SBjoern A. Zeeb 
54967ca7330SBjoern A. Zeeb 	sc = device_get_softc(dev);
55067ca7330SBjoern A. Zeeb 	if (sc == NULL)
55167ca7330SBjoern A. Zeeb 		return (ENXIO);
55267ca7330SBjoern A. Zeeb 
55367ca7330SBjoern A. Zeeb 	/*
55467ca7330SBjoern A. Zeeb 	 * Now that we are a dev, create one child device per function,
55567ca7330SBjoern A. Zeeb 	 * initialize the backpointer, so we can pass them around and
55667ca7330SBjoern A. Zeeb 	 * call CAM operations on the parent, and also set the function
55767ca7330SBjoern A. Zeeb 	 * itself as ivars, so that we can query/update them.
55867ca7330SBjoern A. Zeeb 	 * Do this before any child gets a chance to attach.
55967ca7330SBjoern A. Zeeb 	 */
56067ca7330SBjoern A. Zeeb 	for (i = 0; i < sc->cardinfo.num_funcs; i++) {
561*5b56413dSWarner Losh 		sc->child[i] = device_add_child(dev, NULL, DEVICE_UNIT_ANY);
56267ca7330SBjoern A. Zeeb 		if (sc->child[i] == NULL) {
56367ca7330SBjoern A. Zeeb 			device_printf(dev, "%s: failed to add child\n", __func__);
56467ca7330SBjoern A. Zeeb 			return (ENXIO);
56567ca7330SBjoern A. Zeeb 		}
56667ca7330SBjoern A. Zeeb 		sc->cardinfo.f[i].dev = sc->child[i];
56767ca7330SBjoern A. Zeeb 
56867ca7330SBjoern A. Zeeb 		/* Set the function as ivar to the child device. */
56967ca7330SBjoern A. Zeeb 		device_set_ivars(sc->child[i], &sc->cardinfo.f[i]);
57067ca7330SBjoern A. Zeeb 	}
57167ca7330SBjoern A. Zeeb 
57267ca7330SBjoern A. Zeeb 	/*
57367ca7330SBjoern A. Zeeb 	 * No one will ever attach to F0; we do the above to have a "device"
57467ca7330SBjoern A. Zeeb 	 * to talk to in a general way in the code.
57567ca7330SBjoern A. Zeeb 	 * Also do the probe/attach in a 2nd loop, so that all devices are
57667ca7330SBjoern A. Zeeb 	 * present as we do have drivers consuming more than one device/func
57767ca7330SBjoern A. Zeeb 	 * and might play "tricks" in order to do that assuming devices and
57867ca7330SBjoern A. Zeeb 	 * ivars are available for all.
57967ca7330SBjoern A. Zeeb 	 */
58067ca7330SBjoern A. Zeeb 	for (i = 1; i < sc->cardinfo.num_funcs; i++) {
58167ca7330SBjoern A. Zeeb 		error = device_probe_and_attach(sc->child[i]);
58267ca7330SBjoern A. Zeeb 		if (error != 0 && bootverbose)
58367ca7330SBjoern A. Zeeb 			device_printf(dev, "%s: device_probe_and_attach(%p %s) "
58467ca7330SBjoern A. Zeeb 			    "failed %d for function %d, no child yet\n",
58567ca7330SBjoern A. Zeeb 			     __func__,
58667ca7330SBjoern A. Zeeb 			     sc->child, device_get_nameunit(sc->child[i]),
58767ca7330SBjoern A. Zeeb 			     error, i);
58867ca7330SBjoern A. Zeeb 	}
58967ca7330SBjoern A. Zeeb 
59067ca7330SBjoern A. Zeeb 	sc->nb_state = NB_STATE_READY;
59167ca7330SBjoern A. Zeeb 
59267ca7330SBjoern A. Zeeb 	cam_periph_lock(sc->periph);
59367ca7330SBjoern A. Zeeb 	xpt_announce_periph(sc->periph, NULL);
59467ca7330SBjoern A. Zeeb 	cam_periph_unlock(sc->periph);
59567ca7330SBjoern A. Zeeb 
59667ca7330SBjoern A. Zeeb 	return (0);
59767ca7330SBjoern A. Zeeb }
59867ca7330SBjoern A. Zeeb 
59967ca7330SBjoern A. Zeeb static int
60067ca7330SBjoern A. Zeeb sdiob_detach(device_t dev)
60167ca7330SBjoern A. Zeeb {
60267ca7330SBjoern A. Zeeb 
60367ca7330SBjoern A. Zeeb 	/* XXX TODO? */
60467ca7330SBjoern A. Zeeb 	return (EOPNOTSUPP);
60567ca7330SBjoern A. Zeeb }
60667ca7330SBjoern A. Zeeb 
60767ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
60867ca7330SBjoern A. Zeeb /*
60967ca7330SBjoern A. Zeeb  * driver(9) and device(9) "control plane".
61067ca7330SBjoern A. Zeeb  * This is what we use when we are making ourselves a device(9) in order to
61167ca7330SBjoern A. Zeeb  * provide a newbus interface again, as well as the implementation of the
61267ca7330SBjoern A. Zeeb  * SDIO interface.
61367ca7330SBjoern A. Zeeb  */
61467ca7330SBjoern A. Zeeb 
61567ca7330SBjoern A. Zeeb static device_method_t sdiob_methods[] = {
61667ca7330SBjoern A. Zeeb 	/* Device interface. */
61767ca7330SBjoern A. Zeeb 	DEVMETHOD(device_probe,		sdiob_probe),
61867ca7330SBjoern A. Zeeb 	DEVMETHOD(device_attach,	sdiob_attach),
61967ca7330SBjoern A. Zeeb 	DEVMETHOD(device_detach,	sdiob_detach),
62067ca7330SBjoern A. Zeeb 
62167ca7330SBjoern A. Zeeb 	/* Bus interface. */
62267ca7330SBjoern A. Zeeb 	DEVMETHOD(bus_add_child,	bus_generic_add_child),
62367ca7330SBjoern A. Zeeb 	DEVMETHOD(bus_driver_added,	bus_generic_driver_added),
62467ca7330SBjoern A. Zeeb 	DEVMETHOD(bus_read_ivar,	sdiob_read_ivar),
62567ca7330SBjoern A. Zeeb 	DEVMETHOD(bus_write_ivar,	sdiob_write_ivar),
62667ca7330SBjoern A. Zeeb 
62767ca7330SBjoern A. Zeeb 	/* SDIO interface. */
62867ca7330SBjoern A. Zeeb 	DEVMETHOD(sdio_read_direct,	sdiob_read_direct),
62967ca7330SBjoern A. Zeeb 	DEVMETHOD(sdio_write_direct,	sdiob_write_direct),
63067ca7330SBjoern A. Zeeb 	DEVMETHOD(sdio_read_extended,	sdiob_read_extended),
63167ca7330SBjoern A. Zeeb 	DEVMETHOD(sdio_write_extended,	sdiob_write_extended),
63267ca7330SBjoern A. Zeeb 
63367ca7330SBjoern A. Zeeb 	DEVMETHOD_END
63467ca7330SBjoern A. Zeeb };
63567ca7330SBjoern A. Zeeb 
63667ca7330SBjoern A. Zeeb static driver_t sdiob_driver = {
63767ca7330SBjoern A. Zeeb 	SDIOB_NAME_S,
63867ca7330SBjoern A. Zeeb 	sdiob_methods,
63967ca7330SBjoern A. Zeeb 	0
64067ca7330SBjoern A. Zeeb };
64167ca7330SBjoern A. Zeeb 
64267ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
64367ca7330SBjoern A. Zeeb /*
64467ca7330SBjoern A. Zeeb  * CIS related.
64567ca7330SBjoern A. Zeeb  * Read card and function information and populate the cardinfo structure.
64667ca7330SBjoern A. Zeeb  */
64767ca7330SBjoern A. Zeeb 
64867ca7330SBjoern A. Zeeb static int
64967ca7330SBjoern A. Zeeb sdio_read_direct_sc(struct sdiob_softc *sc, uint8_t fn, uint32_t addr,
65067ca7330SBjoern A. Zeeb     uint8_t *val)
65167ca7330SBjoern A. Zeeb {
65267ca7330SBjoern A. Zeeb 	int error;
65367ca7330SBjoern A. Zeeb 	uint8_t v;
65467ca7330SBjoern A. Zeeb 
65567ca7330SBjoern A. Zeeb 	error = sdiob_rw_direct_sc(sc, fn, addr, false, &v);
65667ca7330SBjoern A. Zeeb 	if (error == 0 && val != NULL)
65767ca7330SBjoern A. Zeeb 		*val = v;
65867ca7330SBjoern A. Zeeb 	return (error);
65967ca7330SBjoern A. Zeeb }
66067ca7330SBjoern A. Zeeb 
66167ca7330SBjoern A. Zeeb static int
66267ca7330SBjoern A. Zeeb sdio_func_read_cis(struct sdiob_softc *sc, uint8_t fn, uint32_t cis_addr)
66367ca7330SBjoern A. Zeeb {
66467ca7330SBjoern A. Zeeb 	char cis1_info_buf[256];
66567ca7330SBjoern A. Zeeb 	char *cis1_info[4];
66667ca7330SBjoern A. Zeeb 	int start, i, count, ret;
66767ca7330SBjoern A. Zeeb 	uint32_t addr;
66867ca7330SBjoern A. Zeeb 	uint8_t ch, tuple_id, tuple_len, tuple_count, v;
66967ca7330SBjoern A. Zeeb 
67067ca7330SBjoern A. Zeeb 	/* If we encounter any read errors, abort and return. */
67167ca7330SBjoern A. Zeeb #define	ERR_OUT(ret)							\
67267ca7330SBjoern A. Zeeb 	if (ret != 0)							\
67367ca7330SBjoern A. Zeeb 		goto err;
67467ca7330SBjoern A. Zeeb 	ret = 0;
67567ca7330SBjoern A. Zeeb 	/* Use to prevent infinite loop in case of parse errors. */
67667ca7330SBjoern A. Zeeb 	tuple_count = 0;
67767ca7330SBjoern A. Zeeb 	memset(cis1_info_buf, 0, 256);
67867ca7330SBjoern A. Zeeb 	do {
67967ca7330SBjoern A. Zeeb 		addr = cis_addr;
68067ca7330SBjoern A. Zeeb 		ret = sdio_read_direct_sc(sc, 0, addr++, &tuple_id);
68167ca7330SBjoern A. Zeeb 		ERR_OUT(ret);
68267ca7330SBjoern A. Zeeb 		if (tuple_id == SD_IO_CISTPL_END)
68367ca7330SBjoern A. Zeeb 			break;
68467ca7330SBjoern A. Zeeb 		if (tuple_id == 0) {
68567ca7330SBjoern A. Zeeb 			cis_addr++;
68667ca7330SBjoern A. Zeeb 			continue;
68767ca7330SBjoern A. Zeeb 		}
68867ca7330SBjoern A. Zeeb 		ret = sdio_read_direct_sc(sc, 0, addr++, &tuple_len);
68967ca7330SBjoern A. Zeeb 		ERR_OUT(ret);
69067ca7330SBjoern A. Zeeb 		if (tuple_len == 0) {
69167ca7330SBjoern A. Zeeb 			CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_PERIPH,
69267ca7330SBjoern A. Zeeb 			    ("%s: parse error: 0-length tuple %#02x\n",
69367ca7330SBjoern A. Zeeb 			    __func__, tuple_id));
69467ca7330SBjoern A. Zeeb 			return (EIO);
69567ca7330SBjoern A. Zeeb 		}
69667ca7330SBjoern A. Zeeb 
69767ca7330SBjoern A. Zeeb 		switch (tuple_id) {
69867ca7330SBjoern A. Zeeb 		case SD_IO_CISTPL_VERS_1:
69967ca7330SBjoern A. Zeeb 			addr += 2;
70067ca7330SBjoern A. Zeeb 			for (count = 0, start = 0, i = 0;
70167ca7330SBjoern A. Zeeb 			     (count < 4) && ((i + 4) < 256); i++) {
70267ca7330SBjoern A. Zeeb 				ret = sdio_read_direct_sc(sc, 0, addr + i, &ch);
70367ca7330SBjoern A. Zeeb 				ERR_OUT(ret);
70467ca7330SBjoern A. Zeeb 				DPRINTF("%s: count=%d, start=%d, i=%d, got "
70567ca7330SBjoern A. Zeeb 				    "(%#02x)\n", __func__, count, start, i, ch);
70667ca7330SBjoern A. Zeeb 				if (ch == 0xff)
70767ca7330SBjoern A. Zeeb 					break;
70867ca7330SBjoern A. Zeeb 				cis1_info_buf[i] = ch;
70967ca7330SBjoern A. Zeeb 				if (ch == 0) {
71067ca7330SBjoern A. Zeeb 					cis1_info[count] =
71167ca7330SBjoern A. Zeeb 					    cis1_info_buf + start;
71267ca7330SBjoern A. Zeeb 					start = i + 1;
71367ca7330SBjoern A. Zeeb 					count++;
71467ca7330SBjoern A. Zeeb 				}
71567ca7330SBjoern A. Zeeb 			}
71667ca7330SBjoern A. Zeeb 			DPRINTF("Card info: ");
71767ca7330SBjoern A. Zeeb 			for (i=0; i < 4; i++)
71867ca7330SBjoern A. Zeeb 				if (cis1_info[i])
71967ca7330SBjoern A. Zeeb 					DPRINTF(" %s", cis1_info[i]);
72067ca7330SBjoern A. Zeeb 			DPRINTF("\n");
72167ca7330SBjoern A. Zeeb 			break;
72267ca7330SBjoern A. Zeeb 		case SD_IO_CISTPL_MANFID:
72367ca7330SBjoern A. Zeeb 			/* TPLMID_MANF */
72467ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr++, &v);
72567ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
72667ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].vendor = v;
72767ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr++, &v);
72867ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
72967ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].vendor |= (v << 8);
73067ca7330SBjoern A. Zeeb 			/* TPLMID_CARD */
73167ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr++, &v);
73267ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
73367ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].device = v;
73467ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr, &v);
73567ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
73667ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].device |= (v << 8);
73767ca7330SBjoern A. Zeeb 			break;
73867ca7330SBjoern A. Zeeb 		case SD_IO_CISTPL_FUNCID:
73967ca7330SBjoern A. Zeeb 			/* Not sure if we need to parse it? */
74067ca7330SBjoern A. Zeeb 			break;
74167ca7330SBjoern A. Zeeb 		case SD_IO_CISTPL_FUNCE:
74267ca7330SBjoern A. Zeeb 			if (tuple_len < 4) {
74367ca7330SBjoern A. Zeeb 				printf("%s: FUNCE is too short: %d\n",
74467ca7330SBjoern A. Zeeb 				    __func__, tuple_len);
74567ca7330SBjoern A. Zeeb 				break;
74667ca7330SBjoern A. Zeeb 			}
74767ca7330SBjoern A. Zeeb 			/* TPLFE_TYPE (Extended Data) */
74867ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr++, &v);
74967ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
75067ca7330SBjoern A. Zeeb 			if (fn == 0) {
75167ca7330SBjoern A. Zeeb 				if (v != 0x00)
75267ca7330SBjoern A. Zeeb 					break;
75367ca7330SBjoern A. Zeeb 			} else {
75467ca7330SBjoern A. Zeeb 				if (v != 0x01)
75567ca7330SBjoern A. Zeeb 					break;
75667ca7330SBjoern A. Zeeb 				addr += 0x0b;
75767ca7330SBjoern A. Zeeb 			}
75867ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr, &v);
75967ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
76067ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].max_blksize = v;
76167ca7330SBjoern A. Zeeb 			ret = sdio_read_direct_sc(sc, 0, addr+1, &v);
76267ca7330SBjoern A. Zeeb 			ERR_OUT(ret);
76367ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].max_blksize |= (v << 8);
76467ca7330SBjoern A. Zeeb 			break;
76567ca7330SBjoern A. Zeeb 		default:
76667ca7330SBjoern A. Zeeb 			CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_PERIPH,
76767ca7330SBjoern A. Zeeb 			    ("%s: Skipping fn %d tuple %d ID %#02x "
76867ca7330SBjoern A. Zeeb 			    "len %#02x\n", __func__, fn, tuple_count,
76967ca7330SBjoern A. Zeeb 			    tuple_id, tuple_len));
77067ca7330SBjoern A. Zeeb 		}
77167ca7330SBjoern A. Zeeb 		if (tuple_len == 0xff) {
77267ca7330SBjoern A. Zeeb 			/* Also marks the end of a tuple chain (E1 16.2) */
77367ca7330SBjoern A. Zeeb 			/* The tuple is valid, hence this going at the end. */
77467ca7330SBjoern A. Zeeb 			break;
77567ca7330SBjoern A. Zeeb 		}
77667ca7330SBjoern A. Zeeb 		cis_addr += 2 + tuple_len;
77767ca7330SBjoern A. Zeeb 		tuple_count++;
77867ca7330SBjoern A. Zeeb 	} while (tuple_count < 20);
77967ca7330SBjoern A. Zeeb err:
78067ca7330SBjoern A. Zeeb #undef ERR_OUT
78167ca7330SBjoern A. Zeeb 	return (ret);
78267ca7330SBjoern A. Zeeb }
78367ca7330SBjoern A. Zeeb 
78467ca7330SBjoern A. Zeeb static int
78567ca7330SBjoern A. Zeeb sdio_get_common_cis_addr(struct sdiob_softc *sc, uint32_t *addr)
78667ca7330SBjoern A. Zeeb {
78767ca7330SBjoern A. Zeeb 	int error;
78867ca7330SBjoern A. Zeeb 	uint32_t a;
78967ca7330SBjoern A. Zeeb 	uint8_t val;
79067ca7330SBjoern A. Zeeb 
79167ca7330SBjoern A. Zeeb 	error = sdio_read_direct_sc(sc, 0, SD_IO_CCCR_CISPTR + 0, &val);
79267ca7330SBjoern A. Zeeb 	if (error != 0)
79367ca7330SBjoern A. Zeeb 		goto err;
79467ca7330SBjoern A. Zeeb 	a = val;
79567ca7330SBjoern A. Zeeb 	error = sdio_read_direct_sc(sc, 0, SD_IO_CCCR_CISPTR + 1, &val);
79667ca7330SBjoern A. Zeeb 	if (error != 0)
79767ca7330SBjoern A. Zeeb 		goto err;
79867ca7330SBjoern A. Zeeb 	a |= (val << 8);
79967ca7330SBjoern A. Zeeb 	error = sdio_read_direct_sc(sc, 0, SD_IO_CCCR_CISPTR + 2, &val);
80067ca7330SBjoern A. Zeeb 	if (error != 0)
80167ca7330SBjoern A. Zeeb 		goto err;
80267ca7330SBjoern A. Zeeb 	a |= (val << 16);
80367ca7330SBjoern A. Zeeb 
80467ca7330SBjoern A. Zeeb 	if (a < SD_IO_CIS_START || a > SD_IO_CIS_START + SD_IO_CIS_SIZE) {
80567ca7330SBjoern A. Zeeb err:
80667ca7330SBjoern A. Zeeb 		CAM_DEBUG(sc->ccb->ccb_h.path, CAM_DEBUG_PERIPH,
80767ca7330SBjoern A. Zeeb 		    ("%s: bad CIS address: %#04x, error %d\n", __func__, a,
80867ca7330SBjoern A. Zeeb 		    error));
80967ca7330SBjoern A. Zeeb 	} else if (error == 0 && addr != NULL)
81067ca7330SBjoern A. Zeeb 		*addr = a;
81167ca7330SBjoern A. Zeeb 
81267ca7330SBjoern A. Zeeb 	return (error);
81367ca7330SBjoern A. Zeeb }
81467ca7330SBjoern A. Zeeb 
81567ca7330SBjoern A. Zeeb static int
81667ca7330SBjoern A. Zeeb sdiob_get_card_info(struct sdiob_softc *sc)
81767ca7330SBjoern A. Zeeb {
81867ca7330SBjoern A. Zeeb 	struct mmc_params *mmcp;
81967ca7330SBjoern A. Zeeb 	uint32_t cis_addr, fbr_addr;
82067ca7330SBjoern A. Zeeb 	int fn, error;
82167ca7330SBjoern A. Zeeb 	uint8_t fn_max, val;
82267ca7330SBjoern A. Zeeb 
82367ca7330SBjoern A. Zeeb 	error = sdio_get_common_cis_addr(sc, &cis_addr);
82467ca7330SBjoern A. Zeeb 	if (error != 0)
82567ca7330SBjoern A. Zeeb 		return (-1);
82667ca7330SBjoern A. Zeeb 
82767ca7330SBjoern A. Zeeb 	memset(&sc->cardinfo, 0, sizeof(sc->cardinfo));
82867ca7330SBjoern A. Zeeb 
82967ca7330SBjoern A. Zeeb 	/* F0 must always be present. */
83067ca7330SBjoern A. Zeeb 	fn = 0;
83167ca7330SBjoern A. Zeeb 	error = sdio_func_read_cis(sc, fn, cis_addr);
83267ca7330SBjoern A. Zeeb 	if (error != 0)
83367ca7330SBjoern A. Zeeb 		return (error);
83467ca7330SBjoern A. Zeeb 	sc->cardinfo.num_funcs++;
83567ca7330SBjoern A. Zeeb 	/* Read CCCR Card Capability. */
83667ca7330SBjoern A. Zeeb 	error = sdio_read_direct_sc(sc, 0, SD_IO_CCCR_CARDCAP, &val);
83767ca7330SBjoern A. Zeeb 	if (error != 0)
83867ca7330SBjoern A. Zeeb 		return (error);
83967ca7330SBjoern A. Zeeb 	sc->cardinfo.support_multiblk = (val & CCCR_CC_SMB) ? true : false;
84067ca7330SBjoern A. Zeeb 	DPRINTF("%s: F%d: Vendor %#04x product %#04x max block size %d bytes "
84167ca7330SBjoern A. Zeeb 	    "support_multiblk %s\n",
84267ca7330SBjoern A. Zeeb 	    __func__, fn, sc->cardinfo.f[fn].vendor, sc->cardinfo.f[fn].device,
84367ca7330SBjoern A. Zeeb 	    sc->cardinfo.f[fn].max_blksize,
84467ca7330SBjoern A. Zeeb 	    sc->cardinfo.support_multiblk ? "yes" : "no");
84567ca7330SBjoern A. Zeeb 
84667ca7330SBjoern A. Zeeb 	/* mmcp->sdio_func_count contains the number of functions w/o F0. */
84767ca7330SBjoern A. Zeeb 	mmcp = &sc->ccb->ccb_h.path->device->mmc_ident_data;
84867ca7330SBjoern A. Zeeb 	fn_max = MIN(mmcp->sdio_func_count + 1, nitems(sc->cardinfo.f));
84967ca7330SBjoern A. Zeeb 	for (fn = 1; fn < fn_max; fn++) {
85067ca7330SBjoern A. Zeeb 		fbr_addr = SD_IO_FBR_START * fn + SD_IO_FBR_CIS_OFFSET;
85167ca7330SBjoern A. Zeeb 
85267ca7330SBjoern A. Zeeb 		error = sdio_read_direct_sc(sc, 0, fbr_addr++, &val);
85367ca7330SBjoern A. Zeeb 		if (error != 0)
85467ca7330SBjoern A. Zeeb 			break;
85567ca7330SBjoern A. Zeeb 		cis_addr = val;
85667ca7330SBjoern A. Zeeb 		error = sdio_read_direct_sc(sc, 0, fbr_addr++, &val);
85767ca7330SBjoern A. Zeeb 		if (error != 0)
85867ca7330SBjoern A. Zeeb 			break;
85967ca7330SBjoern A. Zeeb 		cis_addr |= (val << 8);
86067ca7330SBjoern A. Zeeb 		error = sdio_read_direct_sc(sc, 0, fbr_addr++, &val);
86167ca7330SBjoern A. Zeeb 		if (error != 0)
86267ca7330SBjoern A. Zeeb 			break;
86367ca7330SBjoern A. Zeeb 		cis_addr |= (val << 16);
86467ca7330SBjoern A. Zeeb 
86567ca7330SBjoern A. Zeeb 		error = sdio_func_read_cis(sc, fn, cis_addr);
86667ca7330SBjoern A. Zeeb 		if (error != 0)
86767ca7330SBjoern A. Zeeb 			break;
86867ca7330SBjoern A. Zeeb 
86967ca7330SBjoern A. Zeeb 		/* Read the Standard SDIO Function Interface Code. */
87067ca7330SBjoern A. Zeeb 		fbr_addr = SD_IO_FBR_START * fn;
87167ca7330SBjoern A. Zeeb 		error = sdio_read_direct_sc(sc, 0, fbr_addr++, &val);
87267ca7330SBjoern A. Zeeb 		if (error != 0)
87367ca7330SBjoern A. Zeeb 			break;
87467ca7330SBjoern A. Zeeb 		sc->cardinfo.f[fn].class = (val & 0x0f);
87567ca7330SBjoern A. Zeeb 		if (sc->cardinfo.f[fn].class == 0x0f) {
87667ca7330SBjoern A. Zeeb 			error = sdio_read_direct_sc(sc, 0, fbr_addr, &val);
87767ca7330SBjoern A. Zeeb 			if (error != 0)
87867ca7330SBjoern A. Zeeb 				break;
87967ca7330SBjoern A. Zeeb 			sc->cardinfo.f[fn].class = val;
88067ca7330SBjoern A. Zeeb 		}
88167ca7330SBjoern A. Zeeb 
88267ca7330SBjoern A. Zeeb 		sc->cardinfo.f[fn].fn = fn;
88367ca7330SBjoern A. Zeeb 		sc->cardinfo.f[fn].cur_blksize = sc->cardinfo.f[fn].max_blksize;
88467ca7330SBjoern A. Zeeb 		sc->cardinfo.f[fn].retries = 0;
88567ca7330SBjoern A. Zeeb 		sc->cardinfo.f[fn].timeout = 5000;
88667ca7330SBjoern A. Zeeb 
88767ca7330SBjoern A. Zeeb 		DPRINTF("%s: F%d: Class %d Vendor %#04x product %#04x "
88867ca7330SBjoern A. Zeeb 		    "max_blksize %d bytes\n", __func__, fn,
88967ca7330SBjoern A. Zeeb 		    sc->cardinfo.f[fn].class,
89067ca7330SBjoern A. Zeeb 		    sc->cardinfo.f[fn].vendor, sc->cardinfo.f[fn].device,
89167ca7330SBjoern A. Zeeb 		    sc->cardinfo.f[fn].max_blksize);
89267ca7330SBjoern A. Zeeb 		if (sc->cardinfo.f[fn].vendor == 0) {
89367ca7330SBjoern A. Zeeb 			DPRINTF("%s: F%d doesn't exist\n", __func__, fn);
89467ca7330SBjoern A. Zeeb 			break;
89567ca7330SBjoern A. Zeeb 		}
89667ca7330SBjoern A. Zeeb 		sc->cardinfo.num_funcs++;
89767ca7330SBjoern A. Zeeb 	}
89867ca7330SBjoern A. Zeeb 	return (error);
89967ca7330SBjoern A. Zeeb }
90067ca7330SBjoern A. Zeeb 
90167ca7330SBjoern A. Zeeb /* -------------------------------------------------------------------------- */
90267ca7330SBjoern A. Zeeb /*
90367ca7330SBjoern A. Zeeb  * CAM periph registration, allocation, and detached from that a discovery
90467ca7330SBjoern A. Zeeb  * task, which goes off reads cardinfo, and then adds ourselves to our SIM's
90567ca7330SBjoern A. Zeeb  * device adding the devclass and registering the driver.  This keeps the
90667ca7330SBjoern A. Zeeb  * newbus chain connected though we will talk CAM in the middle (until one
90767ca7330SBjoern A. Zeeb  * day CAM might be newbusyfied).
90867ca7330SBjoern A. Zeeb  */
90967ca7330SBjoern A. Zeeb 
91067ca7330SBjoern A. Zeeb static int
91167ca7330SBjoern A. Zeeb sdio_newbus_sim_add(struct sdiob_softc *sc)
91267ca7330SBjoern A. Zeeb {
91367ca7330SBjoern A. Zeeb 	device_t pdev;
91467ca7330SBjoern A. Zeeb 	devclass_t bus_devclass;
91567ca7330SBjoern A. Zeeb 	int error;
91667ca7330SBjoern A. Zeeb 
91767ca7330SBjoern A. Zeeb 	/* Add ourselves to our parent (SIM) device. */
91867ca7330SBjoern A. Zeeb 
91967ca7330SBjoern A. Zeeb 	/* Add ourselves to our parent. That way we can become a parent. */
920fdd60a97SWarner Losh 	pdev = xpt_path_sim_device(sc->periph->path);
921fdd60a97SWarner Losh 	KASSERT(pdev != NULL,
922fdd60a97SWarner Losh 	    ("%s: pdev is NULL, sc %p periph %p sim %p\n",
923fdd60a97SWarner Losh 	    __func__, sc, sc->periph, sc->periph->sim));
92467ca7330SBjoern A. Zeeb 
92567ca7330SBjoern A. Zeeb 	if (sc->dev == NULL)
926fdd60a97SWarner Losh 		sc->dev = BUS_ADD_CHILD(pdev, 0, SDIOB_NAME_S, -1);
92767ca7330SBjoern A. Zeeb 	if (sc->dev == NULL)
92867ca7330SBjoern A. Zeeb 		return (ENXIO);
92967ca7330SBjoern A. Zeeb 	device_set_softc(sc->dev, sc);
930fdd60a97SWarner Losh 
93167ca7330SBjoern A. Zeeb 	/*
93267ca7330SBjoern A. Zeeb 	 * Don't set description here; devclass_add_driver() ->
93367ca7330SBjoern A. Zeeb 	 * device_probe_child() -> device_set_driver() will nuke it again.
93467ca7330SBjoern A. Zeeb 	 */
93567ca7330SBjoern A. Zeeb 	bus_devclass = device_get_devclass(pdev);
93667ca7330SBjoern A. Zeeb 	if (bus_devclass == NULL) {
93767ca7330SBjoern A. Zeeb 		printf("%s: Failed to get devclass from %s.\n", __func__,
93867ca7330SBjoern A. Zeeb 		    device_get_nameunit(pdev));
93967ca7330SBjoern A. Zeeb 		return (ENXIO);
94067ca7330SBjoern A. Zeeb 	}
94167ca7330SBjoern A. Zeeb 
942c6df6f53SWarner Losh 	bus_topo_lock();
94367ca7330SBjoern A. Zeeb 	error = devclass_add_driver(bus_devclass, &sdiob_driver,
9440018a304SJohn Baldwin 	    BUS_PASS_DEFAULT, NULL);
945c6df6f53SWarner Losh 	bus_topo_unlock();
94667ca7330SBjoern A. Zeeb 	if (error != 0) {
94767ca7330SBjoern A. Zeeb 		printf("%s: Failed to add driver to devclass: %d.\n",
94867ca7330SBjoern A. Zeeb 		    __func__, error);
94967ca7330SBjoern A. Zeeb 		return (error);
95067ca7330SBjoern A. Zeeb 	}
95167ca7330SBjoern A. Zeeb 
95267ca7330SBjoern A. Zeeb 	/* Done. */
95367ca7330SBjoern A. Zeeb 	sc->nb_state = NB_STATE_SIM_ADDED;
95467ca7330SBjoern A. Zeeb 
95567ca7330SBjoern A. Zeeb 	return (0);
95667ca7330SBjoern A. Zeeb }
95767ca7330SBjoern A. Zeeb 
95867ca7330SBjoern A. Zeeb static void
95967ca7330SBjoern A. Zeeb sdiobdiscover(void *context, int pending)
96067ca7330SBjoern A. Zeeb {
96167ca7330SBjoern A. Zeeb 	struct cam_periph *periph;
96267ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
96367ca7330SBjoern A. Zeeb 	int error;
96467ca7330SBjoern A. Zeeb 
96567ca7330SBjoern A. Zeeb 	KASSERT(context != NULL, ("%s: context is NULL\n", __func__));
96667ca7330SBjoern A. Zeeb 	periph = (struct cam_periph *)context;
96767ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s\n", __func__));
96867ca7330SBjoern A. Zeeb 
96967ca7330SBjoern A. Zeeb 	/* Periph was held for us when this task was enqueued. */
97067ca7330SBjoern A. Zeeb 	if ((periph->flags & CAM_PERIPH_INVALID) != 0) {
97167ca7330SBjoern A. Zeeb 		cam_periph_release(periph);
97267ca7330SBjoern A. Zeeb 		return;
97367ca7330SBjoern A. Zeeb 	}
97467ca7330SBjoern A. Zeeb 
97567ca7330SBjoern A. Zeeb 	sc = periph->softc;
97667ca7330SBjoern A. Zeeb 	sc->sdio_state = SDIO_STATE_INITIALIZING;
97767ca7330SBjoern A. Zeeb 
97867ca7330SBjoern A. Zeeb 	if (sc->ccb == NULL)
97967ca7330SBjoern A. Zeeb 		sc->ccb = xpt_alloc_ccb();
98067ca7330SBjoern A. Zeeb 	else
98167ca7330SBjoern A. Zeeb 		memset(sc->ccb, 0, sizeof(*sc->ccb));
98267ca7330SBjoern A. Zeeb 	xpt_setup_ccb(&sc->ccb->ccb_h, periph->path, CAM_PRIORITY_NONE);
98367ca7330SBjoern A. Zeeb 
98467ca7330SBjoern A. Zeeb 	/*
98567ca7330SBjoern A. Zeeb 	 * Read CCCR and FBR of each function, get manufacturer and device IDs,
98667ca7330SBjoern A. Zeeb 	 * max block size, and whatever else we deem necessary.
98767ca7330SBjoern A. Zeeb 	 */
98867ca7330SBjoern A. Zeeb 	cam_periph_lock(periph);
98967ca7330SBjoern A. Zeeb 	error = sdiob_get_card_info(sc);
99067ca7330SBjoern A. Zeeb 	if  (error == 0)
99167ca7330SBjoern A. Zeeb 		sc->sdio_state = SDIO_STATE_READY;
99267ca7330SBjoern A. Zeeb 	else
99367ca7330SBjoern A. Zeeb 		sc->sdio_state = SDIO_STATE_DEAD;
99467ca7330SBjoern A. Zeeb 	cam_periph_unlock(periph);
99567ca7330SBjoern A. Zeeb 
99667ca7330SBjoern A. Zeeb 	if (error)
99767ca7330SBjoern A. Zeeb 		return;
99867ca7330SBjoern A. Zeeb 
99967ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s: num_func %d\n",
100067ca7330SBjoern A. Zeeb 	    __func__, sc->cardinfo.num_funcs));
100167ca7330SBjoern A. Zeeb 
100267ca7330SBjoern A. Zeeb 	/*
100367ca7330SBjoern A. Zeeb 	 * Now CAM portion of the driver has been initialized and
100467ca7330SBjoern A. Zeeb 	 * we know VID/PID of all the functions on the card.
100567ca7330SBjoern A. Zeeb 	 * Time to hook into the newbus.
100667ca7330SBjoern A. Zeeb 	 */
100767ca7330SBjoern A. Zeeb 	error = sdio_newbus_sim_add(sc);
100867ca7330SBjoern A. Zeeb 	if (error != 0)
100967ca7330SBjoern A. Zeeb 		sc->nb_state = NB_STATE_DEAD;
101067ca7330SBjoern A. Zeeb 
101167ca7330SBjoern A. Zeeb 	return;
101267ca7330SBjoern A. Zeeb }
101367ca7330SBjoern A. Zeeb 
101467ca7330SBjoern A. Zeeb /* Called at the end of cam_periph_alloc() for us to finish allocation. */
101567ca7330SBjoern A. Zeeb static cam_status
101667ca7330SBjoern A. Zeeb sdiobregister(struct cam_periph *periph, void *arg)
101767ca7330SBjoern A. Zeeb {
101867ca7330SBjoern A. Zeeb 	struct sdiob_softc *sc;
101967ca7330SBjoern A. Zeeb 	int error;
102067ca7330SBjoern A. Zeeb 
102167ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s: arg %p\n", __func__, arg));
102267ca7330SBjoern A. Zeeb 	if (arg == NULL) {
102367ca7330SBjoern A. Zeeb 		printf("%s: no getdev CCB, can't register device pariph %p\n",
102467ca7330SBjoern A. Zeeb 		    __func__, periph);
102567ca7330SBjoern A. Zeeb 		return(CAM_REQ_CMP_ERR);
102667ca7330SBjoern A. Zeeb 	}
1027fdd60a97SWarner Losh 	if (xpt_path_sim_device(periph->path) == NULL) {
1028fdd60a97SWarner Losh 		printf("%s: no device_t for sim %p\n", __func__, periph->sim);
102967ca7330SBjoern A. Zeeb 		return(CAM_REQ_CMP_ERR);
103067ca7330SBjoern A. Zeeb 	}
103167ca7330SBjoern A. Zeeb 
103267ca7330SBjoern A. Zeeb 	sc = (struct sdiob_softc *) malloc(sizeof(*sc), M_DEVBUF,
103367ca7330SBjoern A. Zeeb 	    M_NOWAIT|M_ZERO);
103467ca7330SBjoern A. Zeeb 	if (sc == NULL) {
103567ca7330SBjoern A. Zeeb 		printf("%s: unable to allocate sc\n", __func__);
103667ca7330SBjoern A. Zeeb 		return (CAM_REQ_CMP_ERR);
103767ca7330SBjoern A. Zeeb 	}
103867ca7330SBjoern A. Zeeb 	sc->sdio_state = SDIO_STATE_DEAD;
103967ca7330SBjoern A. Zeeb 	sc->nb_state = NB_STATE_DEAD;
104067ca7330SBjoern A. Zeeb 	TASK_INIT(&sc->discover_task, 0, sdiobdiscover, periph);
104167ca7330SBjoern A. Zeeb 
104267ca7330SBjoern A. Zeeb 	/* Refcount until we are setup.  Can't block. */
104367ca7330SBjoern A. Zeeb 	error = cam_periph_hold(periph, PRIBIO);
104467ca7330SBjoern A. Zeeb 	if (error != 0) {
104567ca7330SBjoern A. Zeeb 		printf("%s: lost periph during registration!\n", __func__);
104667ca7330SBjoern A. Zeeb 		free(sc, M_DEVBUF);
104767ca7330SBjoern A. Zeeb 		return(CAM_REQ_CMP_ERR);
104867ca7330SBjoern A. Zeeb 	}
104967ca7330SBjoern A. Zeeb 	periph->softc = sc;
105067ca7330SBjoern A. Zeeb 	sc->periph = periph;
105167ca7330SBjoern A. Zeeb 	cam_periph_unlock(periph);
105267ca7330SBjoern A. Zeeb 
105367ca7330SBjoern A. Zeeb 	error = taskqueue_enqueue(taskqueue_thread, &sc->discover_task);
105467ca7330SBjoern A. Zeeb 
105567ca7330SBjoern A. Zeeb 	cam_periph_lock(periph);
105667ca7330SBjoern A. Zeeb 	/* We will continue to hold a refcount for discover_task. */
105767ca7330SBjoern A. Zeeb 	/* cam_periph_unhold(periph); */
105867ca7330SBjoern A. Zeeb 
105967ca7330SBjoern A. Zeeb 	xpt_schedule(periph, CAM_PRIORITY_XPT);
106067ca7330SBjoern A. Zeeb 
106167ca7330SBjoern A. Zeeb 	return (CAM_REQ_CMP);
106267ca7330SBjoern A. Zeeb }
106367ca7330SBjoern A. Zeeb 
106467ca7330SBjoern A. Zeeb static void
106567ca7330SBjoern A. Zeeb sdioboninvalidate(struct cam_periph *periph)
106667ca7330SBjoern A. Zeeb {
106767ca7330SBjoern A. Zeeb 
106867ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s:\n", __func__));
106967ca7330SBjoern A. Zeeb 
107067ca7330SBjoern A. Zeeb 	return;
107167ca7330SBjoern A. Zeeb }
107267ca7330SBjoern A. Zeeb 
107367ca7330SBjoern A. Zeeb static void
107467ca7330SBjoern A. Zeeb sdiobcleanup(struct cam_periph *periph)
107567ca7330SBjoern A. Zeeb {
107667ca7330SBjoern A. Zeeb 
107767ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s:\n", __func__));
107867ca7330SBjoern A. Zeeb 
107967ca7330SBjoern A. Zeeb 	return;
108067ca7330SBjoern A. Zeeb }
108167ca7330SBjoern A. Zeeb 
108267ca7330SBjoern A. Zeeb static void
108367ca7330SBjoern A. Zeeb sdiobstart(struct cam_periph *periph, union ccb *ccb)
108467ca7330SBjoern A. Zeeb {
108567ca7330SBjoern A. Zeeb 
108667ca7330SBjoern A. Zeeb 	CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, ("%s: ccb %p\n", __func__, ccb));
108767ca7330SBjoern A. Zeeb 
108867ca7330SBjoern A. Zeeb 	return;
108967ca7330SBjoern A. Zeeb }
109067ca7330SBjoern A. Zeeb 
109167ca7330SBjoern A. Zeeb static void
109267ca7330SBjoern A. Zeeb sdiobasync(void *softc, uint32_t code, struct cam_path *path, void *arg)
109367ca7330SBjoern A. Zeeb {
109467ca7330SBjoern A. Zeeb 	struct cam_periph *periph;
109567ca7330SBjoern A. Zeeb 	struct ccb_getdev *cgd;
109667ca7330SBjoern A. Zeeb 	cam_status status;
109767ca7330SBjoern A. Zeeb 
109867ca7330SBjoern A. Zeeb 	periph = (struct cam_periph *)softc;
109967ca7330SBjoern A. Zeeb 
110067ca7330SBjoern A. Zeeb 	CAM_DEBUG(path, CAM_DEBUG_TRACE, ("%s(code=%d)\n", __func__, code));
110167ca7330SBjoern A. Zeeb 	switch (code) {
110267ca7330SBjoern A. Zeeb 	case AC_FOUND_DEVICE:
110367ca7330SBjoern A. Zeeb 		if (arg == NULL)
110467ca7330SBjoern A. Zeeb 			break;
110567ca7330SBjoern A. Zeeb 		cgd = (struct ccb_getdev *)arg;
110667ca7330SBjoern A. Zeeb 		if (cgd->protocol != PROTO_MMCSD)
110767ca7330SBjoern A. Zeeb 			break;
110867ca7330SBjoern A. Zeeb 
110967ca7330SBjoern A. Zeeb 		/* We do not support SD memory (Combo) Cards. */
111067ca7330SBjoern A. Zeeb 		if ((path->device->mmc_ident_data.card_features &
111167ca7330SBjoern A. Zeeb 		    CARD_FEATURE_MEMORY)) {
111267ca7330SBjoern A. Zeeb 			CAM_DEBUG(path, CAM_DEBUG_TRACE,
111367ca7330SBjoern A. Zeeb 			     ("Memory card, not interested\n"));
111467ca7330SBjoern A. Zeeb 			break;
111567ca7330SBjoern A. Zeeb 		}
111667ca7330SBjoern A. Zeeb 
111767ca7330SBjoern A. Zeeb 		/*
111867ca7330SBjoern A. Zeeb 		 * Allocate a peripheral instance for this device which starts
111967ca7330SBjoern A. Zeeb 		 * the probe process.
112067ca7330SBjoern A. Zeeb 		 */
112167ca7330SBjoern A. Zeeb 		status = cam_periph_alloc(sdiobregister, sdioboninvalidate,
112267ca7330SBjoern A. Zeeb 		    sdiobcleanup, sdiobstart, SDIOB_NAME_S, CAM_PERIPH_BIO, path,
112367ca7330SBjoern A. Zeeb 		    sdiobasync, AC_FOUND_DEVICE, cgd);
112467ca7330SBjoern A. Zeeb 		if (status != CAM_REQ_CMP && status != CAM_REQ_INPROG)
112567ca7330SBjoern A. Zeeb 			CAM_DEBUG(path, CAM_DEBUG_PERIPH,
112667ca7330SBjoern A. Zeeb 			     ("%s: Unable to attach to new device due to "
112767ca7330SBjoern A. Zeeb 			     "status %#02x\n", __func__, status));
112867ca7330SBjoern A. Zeeb 		break;
112967ca7330SBjoern A. Zeeb 	default:
113067ca7330SBjoern A. Zeeb 		CAM_DEBUG(path, CAM_DEBUG_PERIPH,
113167ca7330SBjoern A. Zeeb 		     ("%s: cannot handle async code %#02x\n", __func__, code));
113267ca7330SBjoern A. Zeeb 		cam_periph_async(periph, code, path, arg);
113367ca7330SBjoern A. Zeeb 		break;
113467ca7330SBjoern A. Zeeb 	}
113567ca7330SBjoern A. Zeeb }
113667ca7330SBjoern A. Zeeb 
113767ca7330SBjoern A. Zeeb static void
113867ca7330SBjoern A. Zeeb sdiobinit(void)
113967ca7330SBjoern A. Zeeb {
114067ca7330SBjoern A. Zeeb 	cam_status status;
114167ca7330SBjoern A. Zeeb 
114267ca7330SBjoern A. Zeeb 	/*
114367ca7330SBjoern A. Zeeb 	 * Register for new device notification.  We will be notified for all
114467ca7330SBjoern A. Zeeb 	 * already existing ones.
114567ca7330SBjoern A. Zeeb 	 */
114667ca7330SBjoern A. Zeeb 	status = xpt_register_async(AC_FOUND_DEVICE, sdiobasync, NULL, NULL);
114767ca7330SBjoern A. Zeeb 	if (status != CAM_REQ_CMP)
114867ca7330SBjoern A. Zeeb 		printf("%s: Failed to attach async callback, statux %#02x",
114967ca7330SBjoern A. Zeeb 		    __func__, status);
115067ca7330SBjoern A. Zeeb }
115167ca7330SBjoern A. Zeeb 
115267ca7330SBjoern A. Zeeb /* This function will allow unloading the KLD. */
115367ca7330SBjoern A. Zeeb static int
115467ca7330SBjoern A. Zeeb sdiobdeinit(void)
115567ca7330SBjoern A. Zeeb {
115667ca7330SBjoern A. Zeeb 
115767ca7330SBjoern A. Zeeb 	return (EOPNOTSUPP);
115867ca7330SBjoern A. Zeeb }
115967ca7330SBjoern A. Zeeb 
116067ca7330SBjoern A. Zeeb static struct periph_driver sdiobdriver =
116167ca7330SBjoern A. Zeeb {
116267ca7330SBjoern A. Zeeb 	.init =		sdiobinit,
116367ca7330SBjoern A. Zeeb 	.driver_name =	SDIOB_NAME_S,
116467ca7330SBjoern A. Zeeb 	.units =	TAILQ_HEAD_INITIALIZER(sdiobdriver.units),
116567ca7330SBjoern A. Zeeb 	.generation =	0,
116667ca7330SBjoern A. Zeeb 	.flags =	0,
116767ca7330SBjoern A. Zeeb 	.deinit =	sdiobdeinit,
116867ca7330SBjoern A. Zeeb };
116967ca7330SBjoern A. Zeeb 
117067ca7330SBjoern A. Zeeb PERIPHDRIVER_DECLARE(SDIOB_NAME, sdiobdriver);
117167ca7330SBjoern A. Zeeb MODULE_VERSION(SDIOB_NAME, 1);
1172