[PATCH v7 4/8] ARM: sunxi: Add driver for SD/MMC hosts found on Allwinner sunxi SoCs
Maxime Ripard
maxime.ripard at free-electrons.com
Wed Feb 19 04:46:15 EST 2014
Hi Hans,
On Tue, Feb 18, 2014 at 09:49:21PM +0100, Hans de Goede wrote:
> 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.
To expand a bit, my point was not that it was functionnally
wrong. Since you put a barrier in there, and that it resides in a
cache coherent section, we're fine.
My point was that the comment itself was misleading.
> >>+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.
Ok.
> >
> >(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.
Still. Something would be seriously wrong in the MMC
driver/controller. You don't want to bloat the whole kernel logs with
the dump of your registers just because the MMC is failing. This is of
no interest to anyone but someone that would actually try to debug
what's 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!
Doh... Right.
My bad.
> 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.
Hmmm, I see. It probably deserves some commenting here too then.
> >>+ 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.
Still, if anyone makes a following patch using the platform_device for
some reason, we will have a race condition, without any way to notice
it.
Plus, you're doing all the other bits of initialization of your
structures much earlier, why not be consistent and having all of them
at the same place?
> >
> >>+ 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.
Ok.
Thanks!
Maxime
--
Maxime Ripard, Free Electrons
Embedded Linux, Kernel and Android engineering
http://free-electrons.com
-------------- next part --------------
A non-text attachment was scrubbed...
Name: signature.asc
Type: application/pgp-signature
Size: 836 bytes
Desc: Digital signature
URL: <http://lists.infradead.org/pipermail/linux-arm-kernel/attachments/20140219/d7dad90d/attachment.sig>
More information about the linux-arm-kernel
mailing list