[PATCH v7 4/8] ARM: sunxi: Add driver for SD/MMC hosts found on Allwinner sunxi SoCs
Hans de Goede
hdegoede at redhat.com
Tue Feb 18 15:49:21 EST 2014
Hi,
On 02/18/2014 04:37 PM, Maxime Ripard wrote:
<snip>
>> +
>> + for (i = 0; i < data->sg_len; i++) {
>> + pdes[i].config = SDXC_IDMAC_DES0_CH | SDXC_IDMAC_DES0_OWN |
>> + SDXC_IDMAC_DES0_DIC;
>> +
>> + if (data->sg[i].length == max_len)
>> + pdes[i].buf_size = 0; /* 0 == max_len */
>> + else
>> + pdes[i].buf_size = data->sg[i].length;
>> +
>> + pdes[i].buf_addr_ptr1 = sg_dma_address(&data->sg[i]);
>> + pdes[i].buf_addr_ptr2 = (u32)&pdes_pa[i + 1];
>> + }
>> +
>> + pdes[0].config |= SDXC_IDMAC_DES0_FD;
>> + pdes[i - 1].config = SDXC_IDMAC_DES0_OWN | SDXC_IDMAC_DES0_LD;
>> +
>> + wmb(); /* Ensure idma_des hit main mem before we start the idmac */
>
> wmb ensure the proper ordering of the instructions, not flushing the
> caches like what your comment implies.
Since I put that comment there, allow me to explain. A modern ARM
cpu core has 2 or more units handling stores. One for regular
memory stores, and one for io-mem stores. Regular mem stores can
be re-ordered, io stores cannot. Normally there is no "syncing"
between the 2 store units. Cache flushing is not an issue here
since the memory holding the descriptors for the idma controller
is allocated cache coherent, which on arm means it is not cached.
What is an issue here is the io-store starting the idmac hitting
the io-mem before the descriptors hit the main-mem, the wmb()
ensures this does not happen.
>> +static int sunxi_mmc_prepare_dma(struct sunxi_mmc_host *smc_host,
>> + struct mmc_data *data)
>> +{
>> + u32 dma_len;
>> + u32 i;
>> + u32 temp;
>> + struct scatterlist *sg;
>> +
>> + dma_len = dma_map_sg(mmc_dev(smc_host->mmc), data->sg, data->sg_len,
>> + sunxi_mmc_get_dma_dir(data));
>> + if (dma_len == 0) {
>> + dev_err(mmc_dev(smc_host->mmc), "dma_map_sg failed\n");
>> + return -ENOMEM;
>> + }
>> +
>> + for_each_sg(data->sg, sg, data->sg_len, i) {
>> + if (sg->offset & 3 || sg->length & 3) {
>> + dev_err(mmc_dev(smc_host->mmc),
>> + "unaligned scatterlist: os %x length %d\n",
>> + sg->offset, sg->length);
>> + return -EINVAL;
>> + }
>> + }
>> +
>> + sunxi_mmc_init_idma_des(smc_host, data);
>> +
>> + temp = mci_readl(smc_host, REG_GCTRL);
>> + temp |= SDXC_DMA_ENABLE_BIT;
>> + mci_writel(smc_host, REG_GCTRL, temp);
>> + temp |= SDXC_DMA_RESET;
>> + mci_writel(smc_host, REG_GCTRL, temp);
>
> Does it really need to be done in two steps?
We don't know, so this is probably best left as is.
>
> (Newline)
>
>> + mci_writel(smc_host, REG_DMAC, SDXC_IDMAC_SOFT_RESET);
>> +
>> + if (!(data->flags & MMC_DATA_WRITE))
>> + mci_writel(smc_host, REG_IDIE, SDXC_IDMAC_RECEIVE_INTERRUPT);
>> +
>> + mci_writel(smc_host, REG_DMAC, SDXC_IDMAC_FIX_BURST | SDXC_IDMAC_IDMA_ON);
>> +
>> + return 0;
>> +}
>> +
>> +static void sunxi_mmc_send_manual_stop(struct sunxi_mmc_host *host,
>> + struct mmc_request *req)
>> +{
>> + u32 cmd_val = SDXC_START | SDXC_RESP_EXPIRE | SDXC_STOP_ABORT_CMD
>> + | SDXC_CHECK_RESPONSE_CRC | MMC_STOP_TRANSMISSION;
>> + u32 ri = 0;
>> + unsigned long expire = jiffies + msecs_to_jiffies(1000);
>> +
>> + mci_writel(host, REG_CARG, 0);
>> + mci_writel(host, REG_CMDR, cmd_val);
>> +
>> + do {
>> + ri = mci_readl(host, REG_RINTR);
>> + } while (!(ri & (SDXC_COMMAND_DONE | SDXC_INTERRUPT_ERROR_BIT)) &&
>> + time_before(jiffies, expire));
>> +
>> + if (ri & SDXC_INTERRUPT_ERROR_BIT) {
>> + dev_err(mmc_dev(host->mmc), "send stop command failed\n");
>> + if (req->stop)
>> + req->stop->resp[0] = -ETIMEDOUT;
>> + } else {
>> + if (req->stop)
>> + req->stop->resp[0] = mci_readl(host, REG_RESP0);
>> + }
>> +
>> + mci_writel(host, REG_RINTR, 0xffff);
>> +}
>> +
>> +static void sunxi_mmc_dump_errinfo(struct sunxi_mmc_host *smc_host)
>> +{
>> + struct mmc_command *cmd = smc_host->mrq->cmd;
>> + struct mmc_data *data = smc_host->mrq->data;
>> +
>> + /* For some cmds timeout is normal with sd/mmc cards */
>> + if ((smc_host->int_sum & SDXC_INTERRUPT_ERROR_BIT) == SDXC_RESP_TIMEOUT &&
>> + (cmd->opcode == SD_IO_SEND_OP_COND || cmd->opcode == SD_IO_RW_DIRECT))
>> + return;
>> +
>> + dev_err(mmc_dev(smc_host->mmc),
>
> I'd rather put it at a debug loglevel.
Erm, this only happens if something is seriously wrong.
>> + /* Make sure the controller is in a sane state before enabling irqs */
>> + ret = sunxi_mmc_init_host(host->mmc);
>> + if (ret)
>> + return ret;
>> +
>> + host->irq = platform_get_irq(pdev, 0);
>> + ret = devm_request_irq(&pdev->dev, host->irq, sunxi_mmc_irq, 0,
>> + "sunxi-mmc", host);
>> + if (ret == 0)
>> + disable_irq(host->irq);
>
> The disable_irq is useless here. Just exit.
No it is not note the ret == 0, this is not an error handling path!
This is done under an if because we want to do the sunxi_mmc_exit_host
regardless of the request_irq succeeding or not.
>
>> +
>> + /* And put it back in reset */
>> + sunxi_mmc_exit_host(host);
>
> Hu? If it's in reset, how can it generate some IRQs?
Yes, that is why we do the whole dance of init controller, get irq,
disable irq, drop it back in reset (until the mmc subsys does a power on
of the mmc card / sdio dev).
Sometime the controller asserts the irq in reset for some reason, so
without the dance as soon as we do the devm_request_irq we get an irq,
and worse, not only do we get an irq, we cannot clear it since writing to
the interrupt status register does not work when the controller is in reset,
so we get stuck re-entering the irq handler.
>
>> + return ret;
>> +}
>> +
>> +static int sunxi_mmc_probe(struct platform_device *pdev)
>> +{
>> + struct sunxi_mmc_host *host;
>> + struct mmc_host *mmc;
>> + int ret;
>> +
>> + mmc = mmc_alloc_host(sizeof(struct sunxi_mmc_host), &pdev->dev);
>> + if (!mmc) {
>> + dev_err(&pdev->dev, "mmc alloc host failed\n");
>> + return -ENOMEM;
>> + }
>> +
>> + ret = mmc_of_parse(mmc);
>> + if (ret)
>> + goto error_free_host;
>> +
>> + host = mmc_priv(mmc);
>> + host->mmc = mmc;
>> + spin_lock_init(&host->lock);
>> + tasklet_init(&host->tasklet, sunxi_mmc_tasklet, (unsigned long)host);
>> +
>> + ret = sunxi_mmc_resource_request(host, pdev);
>> + if (ret)
>> + goto error_free_host;
>> +
>> + host->sg_cpu = dma_alloc_coherent(&pdev->dev, PAGE_SIZE,
>> + &host->sg_dma, GFP_KERNEL);
>> + if (!host->sg_cpu) {
>> + dev_err(&pdev->dev, "Failed to allocate DMA descriptor mem\n");
>> + ret = -ENOMEM;
>> + goto error_free_host;
>> + }
>> +
>> + mmc->ops = &sunxi_mmc_ops;
>> + mmc->max_blk_count = 8192;
>> + mmc->max_blk_size = 4096;
>> + mmc->max_segs = PAGE_SIZE / sizeof(struct sunxi_idma_des);
>> + mmc->max_seg_size = (1 << host->idma_des_size_bits);
>> + mmc->max_req_size = mmc->max_seg_size * mmc->max_segs;
>> + /* 400kHz ~ 50MHz */
>> + mmc->f_min = 400000;
>> + mmc->f_max = 50000000;
>> + /* available voltages */
>> + if (!IS_ERR(host->vmmc))
>> + mmc->ocr_avail = mmc_regulator_get_ocrmask(host->vmmc);
>> + else
>> + mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34;
>> +
>> + mmc->caps = MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SD_HIGHSPEED |
>> + MMC_CAP_UHS_SDR12 | MMC_CAP_UHS_SDR25 | MMC_CAP_UHS_SDR50 |
>> + MMC_CAP_UHS_DDR50 | MMC_CAP_SDIO_IRQ | MMC_CAP_DRIVER_TYPE_A;
>> + mmc->caps2 = MMC_CAP2_NO_PRESCAN_POWERUP;
>> +
>> + ret = mmc_add_host(mmc);
>> +
>> + if (ret)
>> + goto error_free_dma;
>> +
>> + dev_info(&pdev->dev, "base:0x%p irq:%u\n", host->reg_base, host->irq);
>> + platform_set_drvdata(pdev, mmc);
>
> This should be before the registration. Otherwise, you're racy.
Nope, we only need this to get the data on sunxi_mmc_remove, everywhere
else the data is found through the mmc-host struct.
>
>> + return 0;
>> +
>> +error_free_dma:
>> + dma_free_coherent(&pdev->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma);
>> +error_free_host:
>> + mmc_free_host(mmc);
>> + return ret;
>> +}
>> +
>> +static int sunxi_mmc_remove(struct platform_device *pdev)
>> +{
>> + struct mmc_host *mmc = platform_get_drvdata(pdev);
>> + struct sunxi_mmc_host *host = mmc_priv(mmc);
>> +
>> + mmc_remove_host(mmc);
>> + sunxi_mmc_exit_host(host);
>> + tasklet_disable(&host->tasklet);
>> + dma_free_coherent(&pdev->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma);
>> + mmc_free_host(mmc);
>> +
>> + return 0;
>> +}
>> +
>> +static struct platform_driver sunxi_mmc_driver = {
>> + .driver = {
>> + .name = "sunxi-mmc",
>> + .owner = THIS_MODULE,
>> + .of_match_table = of_match_ptr(sunxi_mmc_of_match),
>> + },
>> + .probe = sunxi_mmc_probe,
>> + .remove = sunxi_mmc_remove,
>> +};
>> +module_platform_driver(sunxi_mmc_driver);
>> +
>> +MODULE_DESCRIPTION("Allwinner's SD/MMC Card Controller Driver");
>> +MODULE_LICENSE("GPL v2");
>> +MODULE_AUTHOR("David Lanzend?rfer <david.lanzendoerfer at o2s.ch>");
>> +MODULE_ALIAS("platform:sunxi-mmc");
>> diff --git a/drivers/mmc/host/sunxi-mmc.h b/drivers/mmc/host/sunxi-mmc.h
>> new file mode 100644
>> index 0000000..75eaa02
>> --- /dev/null
>> +++ b/drivers/mmc/host/sunxi-mmc.h
>> @@ -0,0 +1,239 @@
>> +/*
>> + * Driver for sunxi SD/MMC host controllers
>> + * (C) Copyright 2007-2011 Reuuimlla Technology Co., Ltd.
>> + * (C) Copyright 2007-2011 Aaron Maoye <leafy.myeh at reuuimllatech.com>
>> + * (C) Copyright 2013-2014 O2S GmbH <www.o2s.ch>
>> + * (C) Copyright 2013-2014 David Lanzend?rfer <david.lanzendoerfer at o2s.ch>
>> + * (C) Copyright 2013-2014 Hans de Goede <hdegoede at redhat.com>
>> + *
>> + * 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.
>> + */
>> +
>> +#ifndef __SUNXI_MMC_H__
>> +#define __SUNXI_MMC_H__
>> +
>> +/* register offset definitions */
>> +#define SDXC_REG_GCTRL (0x00) /* SMC Global Control Register */
>> +#define SDXC_REG_CLKCR (0x04) /* SMC Clock Control Register */
>> +#define SDXC_REG_TMOUT (0x08) /* SMC Time Out Register */
>> +#define SDXC_REG_WIDTH (0x0C) /* SMC Bus Width Register */
>> +#define SDXC_REG_BLKSZ (0x10) /* SMC Block Size Register */
>> +#define SDXC_REG_BCNTR (0x14) /* SMC Byte Count Register */
>> +#define SDXC_REG_CMDR (0x18) /* SMC Command Register */
>> +#define SDXC_REG_CARG (0x1C) /* SMC Argument Register */
>> +#define SDXC_REG_RESP0 (0x20) /* SMC Response Register 0 */
>> +#define SDXC_REG_RESP1 (0x24) /* SMC Response Register 1 */
>> +#define SDXC_REG_RESP2 (0x28) /* SMC Response Register 2 */
>> +#define SDXC_REG_RESP3 (0x2C) /* SMC Response Register 3 */
>> +#define SDXC_REG_IMASK (0x30) /* SMC Interrupt Mask Register */
>> +#define SDXC_REG_MISTA (0x34) /* SMC Masked Interrupt Status Register */
>> +#define SDXC_REG_RINTR (0x38) /* SMC Raw Interrupt Status Register */
>> +#define SDXC_REG_STAS (0x3C) /* SMC Status Register */
>> +#define SDXC_REG_FTRGL (0x40) /* SMC FIFO Threshold Watermark Registe */
>> +#define SDXC_REG_FUNS (0x44) /* SMC Function Select Register */
>> +#define SDXC_REG_CBCR (0x48) /* SMC CIU Byte Count Register */
>> +#define SDXC_REG_BBCR (0x4C) /* SMC BIU Byte Count Register */
>> +#define SDXC_REG_DBGC (0x50) /* SMC Debug Enable Register */
>> +#define SDXC_REG_HWRST (0x78) /* SMC Card Hardware Reset for Register */
>> +#define SDXC_REG_DMAC (0x80) /* SMC IDMAC Control Register */
>> +#define SDXC_REG_DLBA (0x84) /* SMC IDMAC Descriptor List Base Addre */
>> +#define SDXC_REG_IDST (0x88) /* SMC IDMAC Status Register */
>> +#define SDXC_REG_IDIE (0x8C) /* SMC IDMAC Interrupt Enable Register */
>> +#define SDXC_REG_CHDA (0x90)
>> +#define SDXC_REG_CBDA (0x94)
>> +
>> +#define mci_readl(host, reg) \
>> + readl((host)->reg_base + SDXC_##reg)
>> +#define mci_writel(host, reg, value) \
>> + writel((value), (host)->reg_base + SDXC_##reg)
>
> Please use some inline functions here.
>
>> +/* global control register bits */
>> +#define SDXC_SOFT_RESET BIT(0)
>> +#define SDXC_FIFO_RESET BIT(1)
>> +#define SDXC_DMA_RESET BIT(2)
>> +#define SDXC_HARDWARE_RESET (SDXC_SOFT_RESET|SDXC_FIFO_RESET|SDXC_DMA_RESET)
>> +#define SDXC_INTERRUPT_ENABLE_BIT BIT(4)
>> +#define SDXC_DMA_ENABLE_BIT BIT(5)
>> +#define SDXC_DEBOUNCE_ENABLE_BIT BIT(8)
>> +#define SDXC_POSEDGE_LATCH_DATA BIT(9)
>> +#define SDXC_DDR_MODE BIT(10)
>> +#define SDXC_MEMORY_ACCESS_DONE BIT(29)
>> +#define SDXC_ACCESS_DONE_DIRECT BIT(30)
>> +#define SDXC_ACCESS_BY_AHB BIT(31)
>> +#define SDXC_ACCESS_BY_DMA (0U << 31)
>
> Isn't it 0?
Yes, but is is the inverse of ACCESS_BY_DMA, which
this makes much clearer then just 0 does.
>
>> +/* clock control bits */
>> +#define SDXC_CARD_CLOCK_ON BIT(16)
>> +#define SDXC_LOW_POWER_ON BIT(17)
>> +/* bus width */
>> +#define SDXC_WIDTH1 (0)
>> +#define SDXC_WIDTH4 (1)
>> +#define SDXC_WIDTH8 (2)
>> +/* smc command bits */
>> +#define SDXC_RESP_EXPIRE BIT(6)
>> +#define SDXC_LONG_RESPONSE BIT(7)
>> +#define SDXC_CHECK_RESPONSE_CRC BIT(8)
>> +#define SDXC_DATA_EXPIRE BIT(9)
>> +#define SDXC_WRITE BIT(10)
>> +#define SDXC_SEQUENCE_MODE BIT(11)
>> +#define SDXC_SEND_AUTO_STOP BIT(12)
>> +#define SDXC_WAIT_PRE_OVER BIT(13)
>> +#define SDXC_STOP_ABORT_CMD BIT(14)
>> +#define SDXC_SEND_INIT_SEQUENCE BIT(15)
>> +#define SDXC_UPCLK_ONLY BIT(21)
>> +#define SDXC_READ_CEATA_DEV BIT(22)
>> +#define SDXC_CCS_EXPIRE BIT(23)
>> +#define SDXC_ENABLE_BIT_BOOT BIT(24)
>> +#define SDXC_ALT_BOOT_OPTIONS BIT(25)
>> +#define SDXC_BOOT_ACK_EXPIRE BIT(26)
>> +#define SDXC_BOOT_ABORT BIT(27)
>> +#define SDXC_VOLTAGE_SWITCH BIT(28)
>> +#define SDXC_USE_HOLD_REGISTER BIT(29)
>> +#define SDXC_START BIT(31)
>> +/* interrupt bits */
>> +#define SDXC_RESP_ERROR BIT(1)
>> +#define SDXC_COMMAND_DONE BIT(2)
>> +#define SDXC_DATA_OVER BIT(3)
>> +#define SDXC_TX_DATA_REQUEST BIT(4)
>> +#define SDXC_RX_DATA_REQUEST BIT(5)
>> +#define SDXC_RESP_CRC_ERROR BIT(6)
>> +#define SDXC_DATA_CRC_ERROR BIT(7)
>> +#define SDXC_RESP_TIMEOUT BIT(8)
>> +#define SDXC_DATA_TIMEOUT BIT(9)
>> +#define SDXC_VOLTAGE_CHANGE_DONE BIT(10)
>> +#define SDXC_FIFO_RUN_ERROR BIT(11)
>> +#define SDXC_HARD_WARE_LOCKED BIT(12)
>> +#define SDXC_START_BIT_ERROR BIT(13)
>> +#define SDXC_AUTO_COMMAND_DONE BIT(14)
>> +#define SDXC_END_BIT_ERROR BIT(15)
>> +#define SDXC_SDIO_INTERRUPT BIT(16)
>> +#define SDXC_CARD_INSERT BIT(30)
>> +#define SDXC_CARD_REMOVE BIT(31)
>> +#define SDXC_INTERRUPT_ERROR_BIT (SDXC_RESP_ERROR | SDXC_RESP_CRC_ERROR | \
>> + SDXC_DATA_CRC_ERROR | SDXC_RESP_TIMEOUT | \
>> + SDXC_DATA_TIMEOUT | SDXC_FIFO_RUN_ERROR | \
>> + SDXC_HARD_WARE_LOCKED | SDXC_START_BIT_ERROR | \
>> + SDXC_END_BIT_ERROR) /* 0xbbc2 */
>> +#define SDXC_INTERRUPT_DONE_BIT (SDXC_AUTO_COMMAND_DONE | SDXC_DATA_OVER | \
>> + SDXC_COMMAND_DONE | SDXC_VOLTAGE_CHANGE_DONE)
>> +/* status */
>> +#define SDXC_RXWL_FLAG BIT(0)
>> +#define SDXC_TXWL_FLAG BIT(1)
>> +#define SDXC_FIFO_EMPTY BIT(2)
>> +#define SDXC_FIFO_FULL BIT(3)
>> +#define SDXC_CARD_PRESENT BIT(8)
>> +#define SDXC_CARD_DATA_BUSY BIT(9)
>> +#define SDXC_DATA_FSM_BUSY BIT(10)
>> +#define SDXC_DMA_REQUEST BIT(31)
>> +#define SDXC_FIFO_SIZE (16)
>> +/* Function select */
>> +#define SDXC_CEATA_ON (0xceaaU << 16)
>> +#define SDXC_SEND_IRQ_RESPONSE BIT(0)
>> +#define SDXC_SDIO_READ_WAIT BIT(1)
>> +#define SDXC_ABORT_READ_DATA BIT(2)
>> +#define SDXC_SEND_CCSD BIT(8)
>> +#define SDXC_SEND_AUTO_STOPCCSD BIT(9)
>> +#define SDXC_CEATA_DEV_INTERRUPT_ENABLE_BIT BIT(10)
>> +/* IDMA controller bus mod bit field */
>> +#define SDXC_IDMAC_SOFT_RESET BIT(0)
>> +#define SDXC_IDMAC_FIX_BURST BIT(1)
>> +#define SDXC_IDMAC_IDMA_ON BIT(7)
>> +#define SDXC_IDMAC_REFETCH_DES BIT(31)
>> +/* IDMA status bit field */
>> +#define SDXC_IDMAC_TRANSMIT_INTERRUPT BIT(0)
>> +#define SDXC_IDMAC_RECEIVE_INTERRUPT BIT(1)
>> +#define SDXC_IDMAC_FATAL_BUS_ERROR BIT(2)
>> +#define SDXC_IDMAC_DESTINATION_INVALID BIT(4)
>> +#define SDXC_IDMAC_CARD_ERROR_SUM BIT(5)
>> +#define SDXC_IDMAC_NORMAL_INTERRUPT_SUM BIT(8)
>> +#define SDXC_IDMAC_ABNORMAL_INTERRUPT_SUM BIT(9)
>> +#define SDXC_IDMAC_HOST_ABORT_INTERRUPT_TX BIT(10)
>> +#define SDXC_IDMAC_HOST_ABORT_INTERRUPT_RX BIT(10)
>> +#define SDXC_IDMAC_IDLE (0U << 13)
>
> Ditto
>
>> +#define SDXC_IDMAC_SUSPEND (1U << 13)
>> +#define SDXC_IDMAC_DESC_READ (2U << 13)
>> +#define SDXC_IDMAC_DESC_CHECK (3U << 13)
>> +#define SDXC_IDMAC_READ_REQUEST_WAIT (4U << 13)
>> +#define SDXC_IDMAC_WRITE_REQUEST_WAIT (5U << 13)
>> +#define SDXC_IDMAC_READ (6U << 13)
>> +#define SDXC_IDMAC_WRITE (7U << 13)
>> +#define SDXC_IDMAC_DESC_CLOSE (8U << 13)
>
> Please use BIT as much as possible here.
Erm lets not do that, nor remove the 0 << 13, this are all
values to store in a multi-bit field which lives in bits 13-xx,
changing the values which happen to be power of 2 into BIT
macros is not helpful.
>
>
>> +
>> +/*
>> +* If the idma-des-size-bits of property is ie 13, bufsize bits are:
>> +* Bits 0-12: buf1 size
>> +* Bits 13-25: buf2 size
>> +* Bits 26-31: not used
>> +* Since we only ever set buf1 size, we can simply store it directly.
>> +*/
>> +#define SDXC_IDMAC_DES0_DIC BIT(1) /* disable interrupt on completion */
>> +#define SDXC_IDMAC_DES0_LD BIT(2) /* last descriptor */
>> +#define SDXC_IDMAC_DES0_FD BIT(3) /* first descriptor */
>> +#define SDXC_IDMAC_DES0_CH BIT(4) /* chain mode */
>> +#define SDXC_IDMAC_DES0_ER BIT(5) /* end of ring */
>> +#define SDXC_IDMAC_DES0_CES BIT(30) /* card error summary */
>> +#define SDXC_IDMAC_DES0_OWN BIT(31) /* 1-idma owns it, 0-host owns it */
>
> Overall, I prefer to have the registers right beneath the register
> declaration. It's easier to spot what bits belong to what registers
> that way.
>
>> +struct sunxi_idma_des {
>> + u32 config;
>> + u32 buf_size;
>> + u32 buf_addr_ptr1;
>> + u32 buf_addr_ptr2;
>> +};
>
> Some comments on top of this structure would be great.
>
>> +struct sunxi_mmc_host {
>> + struct mmc_host *mmc;
>> + struct regulator *vmmc;
>> +
>> + /* IO mapping base */
>> + void __iomem *reg_base;
>> +
>> + spinlock_t lock;
>> + struct tasklet_struct tasklet;
>> +
>> + /* clock management */
>> + struct clk *clk_ahb;
>> + struct clk *clk_mod;
>> +
>> + /* ios information */
>> + u32 clk_mod_rate;
>> + u32 bus_width;
>> + u32 idma_des_size_bits;
>> + u32 ddr;
>> + u32 voltage_switching;
>> +
>> + /* irq */
>> + int irq;
>> + u32 int_sum;
>> + u32 sdio_imask;
>> +
>> + /* flags */
>> + u32 power_on:1;
>> + u32 io_flag:1;
>> + u32 wait_dma:1;
>
> bool?
>
>> + dma_addr_t sg_dma;
>> + void *sg_cpu;
>> +
>> + struct mmc_request *mrq;
>> + u32 ferror;
>> +};
>
> Please be consistent in your spacing, either align the variable names,
> or don't, but make a decision :)
>
>> +#define MMC_CLK_400K 0
>> +#define MMC_CLK_25M 1
>> +#define MMC_CLK_50M 2
>> +#define MMC_CLK_50MDDR 3
>> +#define MMC_CLK_50MDDR_8BIT 4
>> +#define MMC_CLK_100M 5
>> +#define MMC_CLK_200M 6
>> +#define MMC_CLK_MOD_NUM 7
>
> Wouldn't an enum be better here?
>
>> +
>> +struct sunxi_mmc_clk_dly {
>> + u32 mode;
>> + u32 oclk_dly;
>> + u32 sclk_dly;
>> +};
>
> Comments here would be great too.
>
> Thanks for working on this!
> Maxime
>
Regards,
Hans
More information about the linux-arm-kernel
mailing list