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