[U-Boot] [PATCH 1/2] MMC: add sdhci generic framework
Gérald Kerma
geraker at gmail.com
Wed Jun 29 07:53:09 CEST 2011
Hello,
Is this driver adaptable for sheevaplug ?
Le 29/06/2011 05:46, Lei Wen a écrit :
> Hi All,
>
> Any comment to this patch set, could it be merged in this open window?
>
> Thanks,
> Lei
>
> On Thu, Jun 16, 2011 at 11:17 PM, Lei Wen <leiwen at marvell.com> wrote:
>> Nowdays, there are plenty of mmc driver in uboot adopt the sd standard
>> host design, aka as sdhci. It is better to centralize the common logic
>> together to better maintenance.
>>
>> Signed-off-by: Lei Wen <leiwen at marvell.com>
>> ---
>> drivers/mmc/Makefile | 1 +
>> drivers/mmc/sdhci.c | 433 ++++++++++++++++++++++++++++++++++++++++++++++++++
>> include/sdhci.h | 325 +++++++++++++++++++++++++++++++++++++
>> 3 files changed, 759 insertions(+), 0 deletions(-)
>> create mode 100644 drivers/mmc/sdhci.c
>> create mode 100644 include/sdhci.h
>>
>> diff --git a/drivers/mmc/Makefile b/drivers/mmc/Makefile
>> index a8fe17a..50b5117 100644
>> --- a/drivers/mmc/Makefile
>> +++ b/drivers/mmc/Makefile
>> @@ -38,6 +38,7 @@ COBJS-$(CONFIG_OMAP3_MMC) += omap3_mmc.o
>> COBJS-$(CONFIG_OMAP_HSMMC) += omap_hsmmc.o
>> COBJS-$(CONFIG_PXA_MMC) += pxa_mmc.o
>> COBJS-$(CONFIG_S5P_MMC) += s5p_mmc.o
>> +COBJS-$(CONFIG_SDHCI) += sdhci.o
>>
>> COBJS := $(COBJS-y)
>> SRCS := $(COBJS:.o=.c)
>> diff --git a/drivers/mmc/sdhci.c b/drivers/mmc/sdhci.c
>> new file mode 100644
>> index 0000000..9ebd33d
>> --- /dev/null
>> +++ b/drivers/mmc/sdhci.c
>> @@ -0,0 +1,433 @@
>> +/*
>> + * Copyright 2011, Marvell Semiconductor Inc.
>> + * Lei Wen <leiwen at marvell.com>
>> + *
>> + * See file CREDITS for list of people who contributed to this
>> + * project.
>> + *
>> + * This program is free software; you can redistribute it and/or
>> + * modify it under the terms of the GNU General Public License as
>> + * published by the Free Software Foundation; either version 2 of
>> + * the License, or (at your option) any later version.
>> + *
>> + * This program is distributed in the hope that it will be useful,
>> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
>> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
>> + * GNU General Public License for more details.
>> + *
>> + * You should have received a copy of the GNU General Public License
>> + * along with this program; if not, write to the Free Software
>> + * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
>> + * MA 02111-1307 USA
>> + *
>> + * Back ported to the 8xx platform (from the 8260 platform) by
>> + * Murray.Jensen at cmst.csiro.au, 27-Jan-01.
>> + */
>> +
>> +#include <common.h>
>> +#include <malloc.h>
>> +#include <mmc.h>
>> +#include <sdhci.h>
>> +
>> +void *aligned_buffer;
>> +
>> +static void sdhci_reset(struct sdhci_host *host, u8 mask)
>> +{
>> + unsigned long timeout;
>> +
>> + /* Wait max 100 ms */
>> + timeout = 100;
>> + sdhci_writeb(host, mask, SDHCI_SOFTWARE_RESET);
>> + while (sdhci_readb(host, SDHCI_SOFTWARE_RESET) & mask) {
>> + if (timeout == 0) {
>> + printf("Reset 0x%x never completed.\n", (int)mask);
>> + return;
>> + }
>> + timeout--;
>> + udelay(1000);
>> + }
>> +}
>> +
>> +static void sdhci_cmd_done(struct sdhci_host *host, struct mmc_cmd *cmd)
>> +{
>> + int i;
>> + if (cmd->resp_type & MMC_RSP_136) {
>> + /* CRC is stripped so we need to do some shifting. */
>> + for (i = 0; i < 4; i++) {
>> + cmd->response[i] = sdhci_readl(host,
>> + SDHCI_RESPONSE + (3-i)*4) << 8;
>> + if (i != 3)
>> + cmd->response[i] |= sdhci_readb(host,
>> + SDHCI_RESPONSE + (3-i)*4-1);
>> + }
>> + } else {
>> + cmd->response[0] = sdhci_readl(host, SDHCI_RESPONSE);
>> + }
>> +}
>> +
>> +static void sdhci_transfer_pio(struct sdhci_host *host, struct mmc_data *data)
>> +{
>> + int i;
>> + char *offs;
>> + for (i = 0; i < data->blocksize; i += 4) {
>> + offs = data->dest + i;
>> + if (data->flags == MMC_DATA_READ)
>> + *(u32 *)offs = sdhci_readl(host, SDHCI_BUFFER);
>> + else
>> + sdhci_writel(host, *(u32 *)offs, SDHCI_BUFFER);
>> + }
>> +}
>> +
>> +static int sdhci_transfer_data(struct sdhci_host *host, struct mmc_data *data,
>> + unsigned int start_addr)
>> +{
>> + unsigned int stat, rdy, mask, block = 0;
>> +
>> + rdy = SDHCI_INT_SPACE_AVAIL | SDHCI_INT_DATA_AVAIL;
>> + mask = SDHCI_DATA_AVAILABLE | SDHCI_SPACE_AVAILABLE;
>> + do {
>> + stat = sdhci_readl(host, SDHCI_INT_STATUS);
>> + if (stat & SDHCI_INT_ERROR) {
>> + printf("Error detected in status(0x%X)!\n", stat);
>> + return -1;
>> + }
>> + if (stat & rdy) {
>> + if (!(sdhci_readl(host, SDHCI_PRESENT_STATE) & mask))
>> + continue;
>> + sdhci_writel(host, rdy, SDHCI_INT_STATUS);
>> + sdhci_transfer_pio(host, data);
>> + data->dest += data->blocksize;
>> + if (++block >= data->blocks)
>> + break;
>> + }
>> +#ifdef CONFIG_MMC_SDMA
>> + if (stat & SDHCI_INT_DMA_END) {
>> + sdhci_writel(host, SDHCI_INT_DMA_END, SDHCI_INT_STATUS);
>> + start_addr &= SDHCI_DEFAULT_BOUNDARY_SIZE - 1;
>> + start_addr += SDHCI_DEFAULT_BOUNDARY_SIZE;
>> + sdhci_writel(host, start_addr, SDHCI_DMA_ADDRESS);
>> + }
>> +#endif
>> + } while (!(stat & SDHCI_INT_DATA_END));
>> + return 0;
>> +}
>> +
>> +int sdhci_send_command(struct mmc *mmc, struct mmc_cmd *cmd,
>> + struct mmc_data *data)
>> +{
>> + struct sdhci_host *host = (struct sdhci_host *)mmc->priv;
>> + unsigned int stat = 0;
>> + int ret = 0;
>> + int trans_bytes = 0, is_aligned = 1;
>> + u32 mask, flags, mode;
>> + unsigned int timeout, start_addr = 0;
>> +
>> + /* Wait max 10 ms */
>> + timeout = 10;
>> +
>> + sdhci_writel(host, SDHCI_INT_ALL_MASK, SDHCI_INT_STATUS);
>> + mask = SDHCI_CMD_INHIBIT | SDHCI_DATA_INHIBIT;
>> +
>> + /* We shouldn't wait for data inihibit for stop commands, even
>> + though they might use busy signaling */
>> + if (cmd->cmdidx == MMC_CMD_STOP_TRANSMISSION)
>> + mask &= ~SDHCI_DATA_INHIBIT;
>> +
>> + while (sdhci_readl(host, SDHCI_PRESENT_STATE) & mask) {
>> + if (timeout == 0) {
>> + printf("Controller never released inhibit bit(s).\n");
>> + return COMM_ERR;
>> + }
>> + timeout--;
>> + udelay(1000);
>> + }
>> +
>> + mask = SDHCI_INT_RESPONSE;
>> + if (!(cmd->resp_type & MMC_RSP_PRESENT))
>> + flags = SDHCI_CMD_RESP_NONE;
>> + else if (cmd->resp_type & MMC_RSP_136)
>> + flags = SDHCI_CMD_RESP_LONG;
>> + else if (cmd->resp_type & MMC_RSP_BUSY) {
>> + flags = SDHCI_CMD_RESP_SHORT_BUSY;
>> + mask |= SDHCI_INT_DATA_END;
>> + } else
>> + flags = SDHCI_CMD_RESP_SHORT;
>> +
>> + if (cmd->resp_type & MMC_RSP_CRC)
>> + flags |= SDHCI_CMD_CRC;
>> + if (cmd->resp_type & MMC_RSP_OPCODE)
>> + flags |= SDHCI_CMD_INDEX;
>> + if (data)
>> + flags |= SDHCI_CMD_DATA;
>> +
>> + /*Set Transfer mode regarding to data flag*/
>> + if (data != 0) {
>> + sdhci_writeb(host, 0xe, SDHCI_TIMEOUT_CONTROL);
>> + mode = SDHCI_TRNS_BLK_CNT_EN;
>> + trans_bytes = data->blocks * data->blocksize;
>> + if (data->blocks > 1)
>> + mode |= SDHCI_TRNS_MULTI;
>> +
>> + if (data->flags == MMC_DATA_READ)
>> + mode |= SDHCI_TRNS_READ;
>> +
>> +#ifdef CONFIG_MMC_SDMA
>> + if (data->flags == MMC_DATA_READ)
>> + start_addr = (unsigned int)data->dest;
>> + else
>> + start_addr = (unsigned int)data->src;
>> + if ((host->quirks & SDHCI_QUIRK_32BIT_DMA_ADDR) &&
>> + (start_addr & 0x7) != 0x0) {
>> + is_aligned = 0;
>> + start_addr = (unsigned int)aligned_buffer;
>> + if (data->flags != MMC_DATA_READ)
>> + memcpy(aligned_buffer, data->src, trans_bytes);
>> + }
>> +
>> + sdhci_writel(host, start_addr, SDHCI_DMA_ADDRESS);
>> + mode |= SDHCI_TRNS_DMA;
>> +#endif
>> + sdhci_writew(host, SDHCI_MAKE_BLKSZ(SDHCI_DEFAULT_BOUNDARY_ARG,
>> + data->blocksize),
>> + SDHCI_BLOCK_SIZE);
>> + sdhci_writew(host, data->blocks, SDHCI_BLOCK_COUNT);
>> + sdhci_writew(host, mode, SDHCI_TRANSFER_MODE);
>> + }
>> +
>> + sdhci_writel(host, cmd->cmdarg, SDHCI_ARGUMENT);
>> +#ifdef CONFIG_MMC_SDMA
>> + flush_cache(0, ~0);
>> +#endif
>> + sdhci_writew(host, SDHCI_MAKE_CMD(cmd->cmdidx, flags), SDHCI_COMMAND);
>> + do {
>> + stat = sdhci_readl(host, SDHCI_INT_STATUS);
>> + if (stat & SDHCI_INT_ERROR)
>> + break;
>> + } while ((stat & mask) != mask);
>> +
>> + if ((stat & (SDHCI_INT_ERROR | mask)) == mask) {
>> + sdhci_cmd_done(host, cmd);
>> + sdhci_writel(host, mask, SDHCI_INT_STATUS);
>> + } else
>> + ret = -1;
>> +
>> + if (!ret && data)
>> + ret = sdhci_transfer_data(host, data, start_addr);
>> +
>> + stat = sdhci_readl(host, SDHCI_INT_STATUS);
>> + sdhci_writel(host, SDHCI_INT_ALL_MASK, SDHCI_INT_STATUS);
>> + if (!ret) {
>> + if ((host->quirks & SDHCI_QUIRK_32BIT_DMA_ADDR) &&
>> + !is_aligned && (data->flags == MMC_DATA_READ))
>> + memcpy(data->dest, aligned_buffer, trans_bytes);
>> + return 0;
>> + }
>> +
>> + sdhci_reset(host, SDHCI_RESET_CMD);
>> + sdhci_reset(host, SDHCI_RESET_DATA);
>> + if (stat & SDHCI_INT_TIMEOUT)
>> + return TIMEOUT;
>> + else
>> + return COMM_ERR;
>> +}
>> +
>> +static int sdhci_set_clock(struct mmc *mmc, unsigned int clock)
>> +{
>> + struct sdhci_host *host = (struct sdhci_host *)mmc->priv;
>> + unsigned int div, clk, timeout;
>> +
>> + sdhci_writew(host, 0, SDHCI_CLOCK_CONTROL);
>> +
>> + if (clock == 0)
>> + return 0;
>> +
>> + if (host->version >= SDHCI_SPEC_300) {
>> + /* Version 3.00 divisors must be a multiple of 2. */
>> + if (mmc->f_max <= clock)
>> + div = 1;
>> + else {
>> + for (div = 2; div < SDHCI_MAX_DIV_SPEC_300; div += 2) {
>> + if ((mmc->f_max / div) <= clock)
>> + break;
>> + }
>> + }
>> + } else {
>> + /* Version 2.00 divisors must be a power of 2. */
>> + for (div = 1; div < SDHCI_MAX_DIV_SPEC_200; div *= 2) {
>> + if ((mmc->f_max / div) <= clock)
>> + break;
>> + }
>> + }
>> + div >>= 1;
>> +
>> + clk = (div & SDHCI_DIV_MASK) << SDHCI_DIVIDER_SHIFT;
>> + clk |= ((div & SDHCI_DIV_HI_MASK) >> SDHCI_DIV_MASK_LEN)
>> + << SDHCI_DIVIDER_HI_SHIFT;
>> + clk |= SDHCI_CLOCK_INT_EN;
>> + sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL);
>> +
>> + /* Wait max 20 ms */
>> + timeout = 20;
>> + while (!((clk = sdhci_readw(host, SDHCI_CLOCK_CONTROL))
>> + & SDHCI_CLOCK_INT_STABLE)) {
>> + if (timeout == 0) {
>> + printf("Internal clock never stabilised.\n");
>> + return -1;
>> + }
>> + timeout--;
>> + udelay(1000);
>> + }
>> +
>> + clk |= SDHCI_CLOCK_CARD_EN;
>> + sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL);
>> + return 0;
>> +}
>> +
>> +static void sdhci_set_power(struct sdhci_host *host, unsigned short power)
>> +{
>> + u8 pwr = 0;
>> +
>> + if (power != (unsigned short)-1) {
>> + switch (1 << power) {
>> + case MMC_VDD_165_195:
>> + pwr = SDHCI_POWER_180;
>> + break;
>> + case MMC_VDD_29_30:
>> + case MMC_VDD_30_31:
>> + pwr = SDHCI_POWER_300;
>> + break;
>> + case MMC_VDD_32_33:
>> + case MMC_VDD_33_34:
>> + pwr = SDHCI_POWER_330;
>> + break;
>> + }
>> + }
>> +
>> + if (pwr == 0) {
>> + sdhci_writeb(host, 0, SDHCI_POWER_CONTROL);
>> + return;
>> + }
>> +
>> + pwr |= SDHCI_POWER_ON;
>> +
>> + sdhci_writeb(host, pwr, SDHCI_POWER_CONTROL);
>> +}
>> +
>> +void sdhci_set_ios(struct mmc *mmc)
>> +{
>> + u32 ctrl;
>> + struct sdhci_host *host = (struct sdhci_host *)mmc->priv;
>> +
>> + if (mmc->clock != host->clock)
>> + sdhci_set_clock(mmc, mmc->clock);
>> +
>> + /* Set bus width */
>> + ctrl = sdhci_readb(host, SDHCI_HOST_CONTROL);
>> + if (mmc->bus_width == 8) {
>> + ctrl &= ~SDHCI_CTRL_4BITBUS;
>> + if (host->version >= SDHCI_SPEC_300)
>> + ctrl |= SDHCI_CTRL_8BITBUS;
>> + } else {
>> + if (host->version >= SDHCI_SPEC_300)
>> + ctrl &= ~SDHCI_CTRL_8BITBUS;
>> + if (mmc->bus_width == 4)
>> + ctrl |= SDHCI_CTRL_4BITBUS;
>> + else
>> + ctrl &= ~SDHCI_CTRL_4BITBUS;
>> + }
>> +
>> + if (mmc->clock > 26000000)
>> + ctrl |= SDHCI_CTRL_HISPD;
>> + else
>> + ctrl &= ~SDHCI_CTRL_HISPD;
>> +
>> + sdhci_writeb(host, ctrl, SDHCI_HOST_CONTROL);
>> +}
>> +
>> +int sdhci_init(struct mmc *mmc)
>> +{
>> + struct sdhci_host *host = (struct sdhci_host *)mmc->priv;
>> +
>> + if ((host->quirks & SDHCI_QUIRK_32BIT_DMA_ADDR) && !aligned_buffer) {
>> + aligned_buffer = memalign(8, 512*1024);
>> + if (!aligned_buffer) {
>> + printf("Aligned buffer alloc failed!!!");
>> + return -1;
>> + }
>> + }
>> +
>> + /* Eable all state */
>> + sdhci_writel(host, SDHCI_INT_ALL_MASK, SDHCI_INT_ENABLE);
>> + sdhci_writel(host, SDHCI_INT_ALL_MASK, SDHCI_SIGNAL_ENABLE);
>> +
>> + sdhci_set_power(host, fls(mmc->voltages) - 1);
>> +
>> + return 0;
>> +}
>> +
>> +int add_sdhci(struct sdhci_host *host, u32 max_clk, u32 min_clk)
>> +{
>> + struct mmc *mmc;
>> + unsigned int caps;
>> +
>> + mmc = malloc(sizeof(struct mmc));
>> + if (!mmc) {
>> + printf("mmc malloc fail!\n");
>> + return -1;
>> + }
>> +
>> + mmc->priv = host;
>> +
>> + sprintf(mmc->name, "%s", host->name);
>> + mmc->send_cmd = sdhci_send_command;
>> + mmc->set_ios = sdhci_set_ios;
>> + mmc->init = sdhci_init;
>> +
>> + caps = sdhci_readl(host, SDHCI_CAPABILITIES);
>> +#ifdef CONFIG_MMC_SDMA
>> + if (!(caps & SDHCI_CAN_DO_SDMA)) {
>> + printf("Your controller don't support sdma!!\n");
>> + return -1;
>> + }
>> +#endif
>> +
>> + if (max_clk)
>> + mmc->f_max = max_clk;
>> + else {
>> + if (host->version >= SDHCI_SPEC_300)
>> + mmc->f_max = (caps & SDHCI_CLOCK_V3_BASE_MASK)
>> + >> SDHCI_CLOCK_BASE_SHIFT;
>> + else
>> + mmc->f_max = (caps & SDHCI_CLOCK_BASE_MASK)
>> + >> SDHCI_CLOCK_BASE_SHIFT;
>> + mmc->f_max *= 1000000;
>> + }
>> + if (mmc->f_max == 0) {
>> + printf("Hardware doesn't specify base clock frequency\n");
>> + return -1;
>> + }
>> + if (min_clk)
>> + mmc->f_min = min_clk;
>> + else {
>> + if (host->version >= SDHCI_SPEC_300)
>> + mmc->f_min = mmc->f_max / SDHCI_MAX_DIV_SPEC_300;
>> + else
>> + mmc->f_min = mmc->f_max / SDHCI_MAX_DIV_SPEC_200;
>> + }
>> +
>> + mmc->voltages = 0;
>> + if (caps & SDHCI_CAN_VDD_330)
>> + mmc->voltages |= MMC_VDD_32_33 | MMC_VDD_33_34;
>> + if (caps & SDHCI_CAN_VDD_300)
>> + mmc->voltages |= MMC_VDD_29_30 | MMC_VDD_30_31;
>> + if (caps & SDHCI_CAN_VDD_180)
>> + mmc->voltages |= MMC_VDD_165_195;
>> + mmc->host_caps = MMC_MODE_HS | MMC_MODE_HS_52MHz | MMC_MODE_4BIT;
>> + if (caps & SDHCI_CAN_DO_8BIT)
>> + mmc->host_caps |= MMC_MODE_8BIT;
>> +
>> + sdhci_reset(host, SDHCI_RESET_ALL);
>> + mmc_register(mmc);
>> +
>> + return 0;
>> +}
>> diff --git a/include/sdhci.h b/include/sdhci.h
>> new file mode 100644
>> index 0000000..6d52ce9
>> --- /dev/null
>> +++ b/include/sdhci.h
>> @@ -0,0 +1,325 @@
>> +/*
>> + * Copyright 2011, Marvell Semiconductor Inc.
>> + * Lei Wen <leiwen at marvell.com>
>> + *
>> + * See file CREDITS for list of people who contributed to this
>> + * project.
>> + *
>> + * This program is free software; you can redistribute it and/or
>> + * modify it under the terms of the GNU General Public License as
>> + * published by the Free Software Foundation; either version 2 of
>> + * the License, or (at your option) any later version.
>> + *
>> + * This program is distributed in the hope that it will be useful,
>> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
>> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
>> + * GNU General Public License for more details.
>> + *
>> + * You should have received a copy of the GNU General Public License
>> + * along with this program; if not, write to the Free Software
>> + * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
>> + * MA 02111-1307 USA
>> + *
>> + * Back ported to the 8xx platform (from the 8260 platform) by
>> + * Murray.Jensen at cmst.csiro.au, 27-Jan-01.
>> + */
>> +#ifndef __SDHCI_HW_H
>> +#define __SDHCI_HW_H
>> +
>> +#include <asm/io.h>
>> +/*
>> + * Controller registers
>> + */
>> +
>> +#define SDHCI_DMA_ADDRESS 0x00
>> +
>> +#define SDHCI_BLOCK_SIZE 0x04
>> +#define SDHCI_MAKE_BLKSZ(dma, blksz) (((dma & 0x7) << 12) | (blksz & 0xFFF))
>> +
>> +#define SDHCI_BLOCK_COUNT 0x06
>> +
>> +#define SDHCI_ARGUMENT 0x08
>> +
>> +#define SDHCI_TRANSFER_MODE 0x0C
>> +#define SDHCI_TRNS_DMA 0x01
>> +#define SDHCI_TRNS_BLK_CNT_EN 0x02
>> +#define SDHCI_TRNS_ACMD12 0x04
>> +#define SDHCI_TRNS_READ 0x10
>> +#define SDHCI_TRNS_MULTI 0x20
>> +
>> +#define SDHCI_COMMAND 0x0E
>> +#define SDHCI_CMD_RESP_MASK 0x03
>> +#define SDHCI_CMD_CRC 0x08
>> +#define SDHCI_CMD_INDEX 0x10
>> +#define SDHCI_CMD_DATA 0x20
>> +#define SDHCI_CMD_ABORTCMD 0xC0
>> +
>> +#define SDHCI_CMD_RESP_NONE 0x00
>> +#define SDHCI_CMD_RESP_LONG 0x01
>> +#define SDHCI_CMD_RESP_SHORT 0x02
>> +#define SDHCI_CMD_RESP_SHORT_BUSY 0x03
>> +
>> +#define SDHCI_MAKE_CMD(c, f) (((c & 0xff) << 8) | (f & 0xff))
>> +#define SDHCI_GET_CMD(c) ((c>>8) & 0x3f)
>> +
>> +#define SDHCI_RESPONSE 0x10
>> +
>> +#define SDHCI_BUFFER 0x20
>> +
>> +#define SDHCI_PRESENT_STATE 0x24
>> +#define SDHCI_CMD_INHIBIT 0x00000001
>> +#define SDHCI_DATA_INHIBIT 0x00000002
>> +#define SDHCI_DOING_WRITE 0x00000100
>> +#define SDHCI_DOING_READ 0x00000200
>> +#define SDHCI_SPACE_AVAILABLE 0x00000400
>> +#define SDHCI_DATA_AVAILABLE 0x00000800
>> +#define SDHCI_CARD_PRESENT 0x00010000
>> +#define SDHCI_WRITE_PROTECT 0x00080000
>> +
>> +#define SDHCI_HOST_CONTROL 0x28
>> +#define SDHCI_CTRL_LED 0x01
>> +#define SDHCI_CTRL_4BITBUS 0x02
>> +#define SDHCI_CTRL_HISPD 0x04
>> +#define SDHCI_CTRL_DMA_MASK 0x18
>> +#define SDHCI_CTRL_SDMA 0x00
>> +#define SDHCI_CTRL_ADMA1 0x08
>> +#define SDHCI_CTRL_ADMA32 0x10
>> +#define SDHCI_CTRL_ADMA64 0x18
>> +#define SDHCI_CTRL_8BITBUS 0x20
>> +
>> +#define SDHCI_POWER_CONTROL 0x29
>> +#define SDHCI_POWER_ON 0x01
>> +#define SDHCI_POWER_180 0x0A
>> +#define SDHCI_POWER_300 0x0C
>> +#define SDHCI_POWER_330 0x0E
>> +
>> +#define SDHCI_BLOCK_GAP_CONTROL 0x2A
>> +
>> +#define SDHCI_WAKE_UP_CONTROL 0x2B
>> +#define SDHCI_WAKE_ON_INT 0x01
>> +#define SDHCI_WAKE_ON_INSERT 0x02
>> +#define SDHCI_WAKE_ON_REMOVE 0x04
>> +
>> +#define SDHCI_CLOCK_CONTROL 0x2C
>> +#define SDHCI_DIVIDER_SHIFT 8
>> +#define SDHCI_DIVIDER_HI_SHIFT 6
>> +#define SDHCI_DIV_MASK 0xFF
>> +#define SDHCI_DIV_MASK_LEN 8
>> +#define SDHCI_DIV_HI_MASK 0x300
>> +#define SDHCI_CLOCK_CARD_EN 0x0004
>> +#define SDHCI_CLOCK_INT_STABLE 0x0002
>> +#define SDHCI_CLOCK_INT_EN 0x0001
>> +
>> +#define SDHCI_TIMEOUT_CONTROL 0x2E
>> +
>> +#define SDHCI_SOFTWARE_RESET 0x2F
>> +#define SDHCI_RESET_ALL 0x01
>> +#define SDHCI_RESET_CMD 0x02
>> +#define SDHCI_RESET_DATA 0x04
>> +
>> +#define SDHCI_INT_STATUS 0x30
>> +#define SDHCI_INT_ENABLE 0x34
>> +#define SDHCI_SIGNAL_ENABLE 0x38
>> +#define SDHCI_INT_RESPONSE 0x00000001
>> +#define SDHCI_INT_DATA_END 0x00000002
>> +#define SDHCI_INT_DMA_END 0x00000008
>> +#define SDHCI_INT_SPACE_AVAIL 0x00000010
>> +#define SDHCI_INT_DATA_AVAIL 0x00000020
>> +#define SDHCI_INT_CARD_INSERT 0x00000040
>> +#define SDHCI_INT_CARD_REMOVE 0x00000080
>> +#define SDHCI_INT_CARD_INT 0x00000100
>> +#define SDHCI_INT_ERROR 0x00008000
>> +#define SDHCI_INT_TIMEOUT 0x00010000
>> +#define SDHCI_INT_CRC 0x00020000
>> +#define SDHCI_INT_END_BIT 0x00040000
>> +#define SDHCI_INT_INDEX 0x00080000
>> +#define SDHCI_INT_DATA_TIMEOUT 0x00100000
>> +#define SDHCI_INT_DATA_CRC 0x00200000
>> +#define SDHCI_INT_DATA_END_BIT 0x00400000
>> +#define SDHCI_INT_BUS_POWER 0x00800000
>> +#define SDHCI_INT_ACMD12ERR 0x01000000
>> +#define SDHCI_INT_ADMA_ERROR 0x02000000
>> +
>> +#define SDHCI_INT_NORMAL_MASK 0x00007FFF
>> +#define SDHCI_INT_ERROR_MASK 0xFFFF8000
>> +
>> +#define SDHCI_INT_CMD_MASK (SDHCI_INT_RESPONSE | SDHCI_INT_TIMEOUT | \
>> + SDHCI_INT_CRC | SDHCI_INT_END_BIT | SDHCI_INT_INDEX)
>> +#define SDHCI_INT_DATA_MASK (SDHCI_INT_DATA_END | SDHCI_INT_DMA_END | \
>> + SDHCI_INT_DATA_AVAIL | SDHCI_INT_SPACE_AVAIL | \
>> + SDHCI_INT_DATA_TIMEOUT | SDHCI_INT_DATA_CRC | \
>> + SDHCI_INT_DATA_END_BIT | SDHCI_INT_ADMA_ERROR)
>> +#define SDHCI_INT_ALL_MASK ((unsigned int)-1)
>> +
>> +#define SDHCI_ACMD12_ERR 0x3C
>> +
>> +/* 3E-3F reserved */
>> +
>> +#define SDHCI_CAPABILITIES 0x40
>> +#define SDHCI_TIMEOUT_CLK_MASK 0x0000003F
>> +#define SDHCI_TIMEOUT_CLK_SHIFT 0
>> +#define SDHCI_TIMEOUT_CLK_UNIT 0x00000080
>> +#define SDHCI_CLOCK_BASE_MASK 0x00003F00
>> +#define SDHCI_CLOCK_V3_BASE_MASK 0x0000FF00
>> +#define SDHCI_CLOCK_BASE_SHIFT 8
>> +#define SDHCI_MAX_BLOCK_MASK 0x00030000
>> +#define SDHCI_MAX_BLOCK_SHIFT 16
>> +#define SDHCI_CAN_DO_8BIT 0x00040000
>> +#define SDHCI_CAN_DO_ADMA2 0x00080000
>> +#define SDHCI_CAN_DO_ADMA1 0x00100000
>> +#define SDHCI_CAN_DO_HISPD 0x00200000
>> +#define SDHCI_CAN_DO_SDMA 0x00400000
>> +#define SDHCI_CAN_VDD_330 0x01000000
>> +#define SDHCI_CAN_VDD_300 0x02000000
>> +#define SDHCI_CAN_VDD_180 0x04000000
>> +#define SDHCI_CAN_64BIT 0x10000000
>> +
>> +#define SDHCI_CAPABILITIES_1 0x44
>> +
>> +#define SDHCI_MAX_CURRENT 0x48
>> +
>> +/* 4C-4F reserved for more max current */
>> +
>> +#define SDHCI_SET_ACMD12_ERROR 0x50
>> +#define SDHCI_SET_INT_ERROR 0x52
>> +
>> +#define SDHCI_ADMA_ERROR 0x54
>> +
>> +/* 55-57 reserved */
>> +
>> +#define SDHCI_ADMA_ADDRESS 0x58
>> +
>> +/* 60-FB reserved */
>> +
>> +#define SDHCI_SLOT_INT_STATUS 0xFC
>> +
>> +#define SDHCI_HOST_VERSION 0xFE
>> +#define SDHCI_VENDOR_VER_MASK 0xFF00
>> +#define SDHCI_VENDOR_VER_SHIFT 8
>> +#define SDHCI_SPEC_VER_MASK 0x00FF
>> +#define SDHCI_SPEC_VER_SHIFT 0
>> +#define SDHCI_SPEC_100 0
>> +#define SDHCI_SPEC_200 1
>> +#define SDHCI_SPEC_300 2
>> +
>> +/*
>> + * End of controller registers.
>> + */
>> +
>> +#define SDHCI_MAX_DIV_SPEC_200 256
>> +#define SDHCI_MAX_DIV_SPEC_300 2046
>> +
>> +/*
>> + * quirks
>> + */
>> +#define SDHCI_QUIRK_32BIT_DMA_ADDR (1 << 0)
>> +
>> +/*
>> + * Host SDMA buffer boundary. Valid values from 4K to 512K in powers of 2.
>> + */
>> +#define SDHCI_DEFAULT_BOUNDARY_SIZE (512 * 1024)
>> +#define SDHCI_DEFAULT_BOUNDARY_ARG (7)
>> +struct sdhci_ops {
>> +#ifdef CONFIG_MMC_SDHCI_IO_ACCESSORS
>> + u32 (*read_l)(struct sdhci_host *host, int reg);
>> + u16 (*read_w)(struct sdhci_host *host, int reg);
>> + u8 (*read_b)(struct sdhci_host *host, int reg);
>> + void (*write_l)(struct sdhci_host *host, u32 val, int reg);
>> + void (*write_w)(struct sdhci_host *host, u16 val, int reg);
>> + void (*write_b)(struct sdhci_host *host, u8 val, int reg);
>> +#endif
>> +};
>> +
>> +struct sdhci_host {
>> + char *name;
>> + void *ioaddr;
>> + unsigned int quirks;
>> + unsigned int version;
>> + unsigned int clock;
>> + const struct sdhci_ops *ops;
>> +};
>> +
>> +#ifdef CONFIG_MMC_SDHCI_IO_ACCESSORS
>> +
>> +static inline void sdhci_writel(struct sdhci_host *host, u32 val, int reg)
>> +{
>> + if (unlikely(host->ops->write_l))
>> + host->ops->write_l(host, val, reg);
>> + else
>> + writel(val, host->ioaddr + reg);
>> +}
>> +
>> +static inline void sdhci_writew(struct sdhci_host *host, u16 val, int reg)
>> +{
>> + if (unlikely(host->ops->write_w))
>> + host->ops->write_w(host, val, reg);
>> + else
>> + writew(val, host->ioaddr + reg);
>> +}
>> +
>> +static inline void sdhci_writeb(struct sdhci_host *host, u8 val, int reg)
>> +{
>> + if (unlikely(host->ops->write_b))
>> + host->ops->write_b(host, val, reg);
>> + else
>> + writeb(val, host->ioaddr + reg);
>> +}
>> +
>> +static inline u32 sdhci_readl(struct sdhci_host *host, int reg)
>> +{
>> + if (unlikely(host->ops->read_l))
>> + return host->ops->read_l(host, reg);
>> + else
>> + return readl(host->ioaddr + reg);
>> +}
>> +
>> +static inline u16 sdhci_readw(struct sdhci_host *host, int reg)
>> +{
>> + if (unlikely(host->ops->read_w))
>> + return host->ops->read_w(host, reg);
>> + else
>> + return readw(host->ioaddr + reg);
>> +}
>> +
>> +static inline u8 sdhci_readb(struct sdhci_host *host, int reg)
>> +{
>> + if (unlikely(host->ops->read_b))
>> + return host->ops->read_b(host, reg);
>> + else
>> + return readb(host->ioaddr + reg);
>> +}
>> +
>> +#else
>> +
>> +static inline void sdhci_writel(struct sdhci_host *host, u32 val, int reg)
>> +{
>> + writel(val, host->ioaddr + reg);
>> +}
>> +
>> +static inline void sdhci_writew(struct sdhci_host *host, u16 val, int reg)
>> +{
>> + writew(val, host->ioaddr + reg);
>> +}
>> +
>> +static inline void sdhci_writeb(struct sdhci_host *host, u8 val, int reg)
>> +{
>> + writeb(val, host->ioaddr + reg);
>> +}
>> +static inline u32 sdhci_readl(struct sdhci_host *host, int reg)
>> +{
>> + return readl(host->ioaddr + reg);
>> +}
>> +
>> +static inline u16 sdhci_readw(struct sdhci_host *host, int reg)
>> +{
>> + return readw(host->ioaddr + reg);
>> +}
>> +
>> +static inline u8 sdhci_readb(struct sdhci_host *host, int reg)
>> +{
>> + return readb(host->ioaddr + reg);
>> +}
>> +#endif
>> +
>> +int add_sdhci(struct sdhci_host *host, u32 max_clk, u32 min_clk);
>> +#endif /* __SDHCI_HW_H */
>> --
>> 1.7.0.4
>>
>> _______________________________________________
>> U-Boot mailing list
>> U-Boot at lists.denx.de
>> http://lists.denx.de/mailman/listinfo/u-boot
>>
> _______________________________________________
> U-Boot mailing list
> U-Boot at lists.denx.de
> http://lists.denx.de/mailman/listinfo/u-boot
--
A+ - Amicalement - Gk2 [:-]
---
Un arc-en-ciel ça n'existe pas ! Et pourtant tout le monde en a déjà
vu... (Albert Jacquart)
More information about the U-Boot
mailing list