diff options
Diffstat (limited to 'drivers/mmc')
45 files changed, 1322 insertions, 374 deletions
| diff --git a/drivers/mmc/core/mmc_ops.c b/drivers/mmc/core/mmc_ops.c index 74e4364bc9fb..09113b9ad679 100644 --- a/drivers/mmc/core/mmc_ops.c +++ b/drivers/mmc/core/mmc_ops.c @@ -564,7 +564,7 @@ int __mmc_switch(struct mmc_card *card, u8 set, u8 index, u8 value,  	if (index == EXT_CSD_SANITIZE_START)  		cmd.sanitize_busy = true; -	err = mmc_wait_for_cmd(host, &cmd, 0); +	err = mmc_wait_for_cmd(host, &cmd, MMC_CMD_RETRIES);  	if (err)  		goto out; diff --git a/drivers/mmc/core/queue.c b/drivers/mmc/core/queue.c index 7102e2ebc614..9edc08685e86 100644 --- a/drivers/mmc/core/queue.c +++ b/drivers/mmc/core/queue.c @@ -21,6 +21,8 @@  #include "card.h"  #include "host.h" +#define MMC_DMA_MAP_MERGE_SEGMENTS	512 +  static inline bool mmc_cqe_dcmd_busy(struct mmc_queue *mq)  {  	/* Allow only 1 DCMD at a time */ @@ -193,6 +195,12 @@ static void mmc_queue_setup_discard(struct request_queue *q,  		blk_queue_flag_set(QUEUE_FLAG_SECERASE, q);  } +static unsigned int mmc_get_max_segments(struct mmc_host *host) +{ +	return host->can_dma_map_merge ? MMC_DMA_MAP_MERGE_SEGMENTS : +					 host->max_segs; +} +  /**   * mmc_init_request() - initialize the MMC-specific per-request data   * @q: the request queue @@ -206,7 +214,7 @@ static int __mmc_init_request(struct mmc_queue *mq, struct request *req,  	struct mmc_card *card = mq->card;  	struct mmc_host *host = card->host; -	mq_rq->sg = mmc_alloc_sg(host->max_segs, gfp); +	mq_rq->sg = mmc_alloc_sg(mmc_get_max_segments(host), gfp);  	if (!mq_rq->sg)  		return -ENOMEM; @@ -362,13 +370,23 @@ static void mmc_setup_queue(struct mmc_queue *mq, struct mmc_card *card)  		blk_queue_bounce_limit(mq->queue, BLK_BOUNCE_HIGH);  	blk_queue_max_hw_sectors(mq->queue,  		min(host->max_blk_count, host->max_req_size / 512)); -	blk_queue_max_segments(mq->queue, host->max_segs); +	if (host->can_dma_map_merge) +		WARN(!blk_queue_can_use_dma_map_merging(mq->queue, +							mmc_dev(host)), +		     "merging was advertised but not possible"); +	blk_queue_max_segments(mq->queue, mmc_get_max_segments(host));  	if (mmc_card_mmc(card))  		block_size = card->ext_csd.data_sector_size;  	blk_queue_logical_block_size(mq->queue, block_size); -	blk_queue_max_segment_size(mq->queue, +	/* +	 * After blk_queue_can_use_dma_map_merging() was called with succeed, +	 * since it calls blk_queue_virt_boundary(), the mmc should not call +	 * both blk_queue_max_segment_size(). +	 */ +	if (!host->can_dma_map_merge) +		blk_queue_max_segment_size(mq->queue,  			round_down(host->max_seg_size, block_size));  	dma_set_max_seg_size(mmc_dev(host), queue_max_segment_size(mq->queue)); @@ -381,6 +399,11 @@ static void mmc_setup_queue(struct mmc_queue *mq, struct mmc_card *card)  	init_waitqueue_head(&mq->wait);  } +static inline bool mmc_merge_capable(struct mmc_host *host) +{ +	return host->caps2 & MMC_CAP2_MERGE_CAPABLE; +} +  /* Set queue depth to get a reasonable value for q->nr_requests */  #define MMC_QUEUE_DEPTH 64 @@ -418,6 +441,18 @@ int mmc_init_queue(struct mmc_queue *mq, struct mmc_card *card)  	mq->tag_set.cmd_size = sizeof(struct mmc_queue_req);  	mq->tag_set.driver_data = mq; +	/* +	 * Since blk_mq_alloc_tag_set() calls .init_request() of mmc_mq_ops, +	 * the host->can_dma_map_merge should be set before to get max_segs +	 * from mmc_get_max_segments(). +	 */ +	if (mmc_merge_capable(host) && +	    host->max_segs < MMC_DMA_MAP_MERGE_SEGMENTS && +	    dma_get_merge_boundary(mmc_dev(host))) +		host->can_dma_map_merge = 1; +	else +		host->can_dma_map_merge = 0; +  	ret = blk_mq_alloc_tag_set(&mq->tag_set);  	if (ret)  		return ret; diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c index d681e8aaca83..fe914ff5f5d6 100644 --- a/drivers/mmc/core/sd.c +++ b/drivers/mmc/core/sd.c @@ -1292,6 +1292,12 @@ int mmc_attach_sd(struct mmc_host *host)  			goto err;  	} +	/* +	 * Some SD cards claims an out of spec VDD voltage range. Let's treat +	 * these bits as being in-valid and especially also bit7. +	 */ +	ocr &= ~0x7FFF; +  	rocr = mmc_select_voltage(host, ocr);  	/* diff --git a/drivers/mmc/core/sdio.c b/drivers/mmc/core/sdio.c index 8dd8fc32ecca..26cabd53ddc5 100644 --- a/drivers/mmc/core/sdio.c +++ b/drivers/mmc/core/sdio.c @@ -951,6 +951,8 @@ static int mmc_sdio_pre_suspend(struct mmc_host *host)   */  static int mmc_sdio_suspend(struct mmc_host *host)  { +	WARN_ON(host->sdio_irqs && !mmc_card_keep_power(host)); +  	/* Prevent processing of SDIO IRQs in suspended state. */  	mmc_card_set_suspended(host->card);  	cancel_delayed_work_sync(&host->sdio_irq_work); @@ -1013,7 +1015,7 @@ static int mmc_sdio_resume(struct mmc_host *host)  		if (!(host->caps2 & MMC_CAP2_SDIO_IRQ_NOTHREAD))  			wake_up_process(host->sdio_irq_thread);  		else if (host->caps & MMC_CAP_SDIO_IRQ) -			host->ops->enable_sdio_irq(host, 1); +			queue_delayed_work(system_wq, &host->sdio_irq_work, 0);  	}  out: diff --git a/drivers/mmc/core/sdio_irq.c b/drivers/mmc/core/sdio_irq.c index 0bcc5e83bd1a..900871073bd7 100644 --- a/drivers/mmc/core/sdio_irq.c +++ b/drivers/mmc/core/sdio_irq.c @@ -27,10 +27,39 @@  #include "core.h"  #include "card.h" +static int sdio_get_pending_irqs(struct mmc_host *host, u8 *pending) +{ +	struct mmc_card *card = host->card; +	int ret; + +	WARN_ON(!host->claimed); + +	ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_INTx, 0, pending); +	if (ret) { +		pr_debug("%s: error %d reading SDIO_CCCR_INTx\n", +		       mmc_card_id(card), ret); +		return ret; +	} + +	if (*pending && mmc_card_broken_irq_polling(card) && +	    !(host->caps & MMC_CAP_SDIO_IRQ)) { +		unsigned char dummy; + +		/* A fake interrupt could be created when we poll SDIO_CCCR_INTx +		 * register with a Marvell SD8797 card. A dummy CMD52 read to +		 * function 0 register 0xff can avoid this. +		 */ +		mmc_io_rw_direct(card, 0, 0, 0xff, 0, &dummy); +	} + +	return 0; +} +  static int process_sdio_pending_irqs(struct mmc_host *host)  {  	struct mmc_card *card = host->card;  	int i, ret, count; +	bool sdio_irq_pending = host->sdio_irq_pending;  	unsigned char pending;  	struct sdio_func *func; @@ -38,34 +67,23 @@ static int process_sdio_pending_irqs(struct mmc_host *host)  	if (mmc_card_suspended(card))  		return 0; +	/* Clear the flag to indicate that we have processed the IRQ. */ +	host->sdio_irq_pending = false; +  	/*  	 * Optimization, if there is only 1 function interrupt registered  	 * and we know an IRQ was signaled then call irq handler directly.  	 * Otherwise do the full probe.  	 */  	func = card->sdio_single_irq; -	if (func && host->sdio_irq_pending) { +	if (func && sdio_irq_pending) {  		func->irq_handler(func);  		return 1;  	} -	ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_INTx, 0, &pending); -	if (ret) { -		pr_debug("%s: error %d reading SDIO_CCCR_INTx\n", -		       mmc_card_id(card), ret); +	ret = sdio_get_pending_irqs(host, &pending); +	if (ret)  		return ret; -	} - -	if (pending && mmc_card_broken_irq_polling(card) && -	    !(host->caps & MMC_CAP_SDIO_IRQ)) { -		unsigned char dummy; - -		/* A fake interrupt could be created when we poll SDIO_CCCR_INTx -		 * register with a Marvell SD8797 card. A dummy CMD52 read to -		 * function 0 register 0xff can avoid this. -		 */ -		mmc_io_rw_direct(card, 0, 0, 0xff, 0, &dummy); -	}  	count = 0;  	for (i = 1; i <= 7; i++) { @@ -96,9 +114,8 @@ static void sdio_run_irqs(struct mmc_host *host)  {  	mmc_claim_host(host);  	if (host->sdio_irqs) { -		host->sdio_irq_pending = true;  		process_sdio_pending_irqs(host); -		if (host->ops->ack_sdio_irq) +		if (!host->sdio_irq_pending)  			host->ops->ack_sdio_irq(host);  	}  	mmc_release_host(host); @@ -114,6 +131,7 @@ void sdio_irq_work(struct work_struct *work)  void sdio_signal_irq(struct mmc_host *host)  { +	host->sdio_irq_pending = true;  	queue_delayed_work(system_wq, &host->sdio_irq_work, 0);  }  EXPORT_SYMBOL_GPL(sdio_signal_irq); @@ -159,7 +177,6 @@ static int sdio_irq_thread(void *_host)  		if (ret)  			break;  		ret = process_sdio_pending_irqs(host); -		host->sdio_irq_pending = false;  		mmc_release_host(host);  		/* diff --git a/drivers/mmc/host/Kconfig b/drivers/mmc/host/Kconfig index 14d89a108edd..49ea02c467bf 100644 --- a/drivers/mmc/host/Kconfig +++ b/drivers/mmc/host/Kconfig @@ -94,6 +94,7 @@ config MMC_SDHCI_PCI  	depends on MMC_SDHCI && PCI  	select MMC_CQHCI  	select IOSF_MBI if X86 +	select MMC_SDHCI_IO_ACCESSORS  	help  	  This selects the PCI Secure Digital Host Controller Interface.  	  Most controllers found today are PCI devices. @@ -154,6 +155,18 @@ config MMC_SDHCI_OF_ARASAN  	  If unsure, say N. +config MMC_SDHCI_OF_ASPEED +	tristate "SDHCI OF support for the ASPEED SDHCI controller" +	depends on MMC_SDHCI_PLTFM +	depends on OF && OF_ADDRESS +	help +	  This selects the ASPEED Secure Digital Host Controller Interface. + +	  If you have a controller with this interface, say Y or M here. You +	  also need to enable an appropriate bus interface. + +	  If unsure, say N. +  config MMC_SDHCI_OF_AT91  	tristate "SDHCI OF support for the Atmel SDMMC controller"  	depends on MMC_SDHCI_PLTFM diff --git a/drivers/mmc/host/Makefile b/drivers/mmc/host/Makefile index 73578718f119..11c4598e91d9 100644 --- a/drivers/mmc/host/Makefile +++ b/drivers/mmc/host/Makefile @@ -13,7 +13,7 @@ obj-$(CONFIG_MMC_MXS)		+= mxs-mmc.o  obj-$(CONFIG_MMC_SDHCI)		+= sdhci.o  obj-$(CONFIG_MMC_SDHCI_PCI)	+= sdhci-pci.o  sdhci-pci-y			+= sdhci-pci-core.o sdhci-pci-o2micro.o sdhci-pci-arasan.o \ -				   sdhci-pci-dwc-mshc.o +				   sdhci-pci-dwc-mshc.o sdhci-pci-gli.o  obj-$(subst m,y,$(CONFIG_MMC_SDHCI_PCI))	+= sdhci-pci-data.o  obj-$(CONFIG_MMC_SDHCI_ACPI)	+= sdhci-acpi.o  obj-$(CONFIG_MMC_SDHCI_PXAV3)	+= sdhci-pxav3.o @@ -84,6 +84,7 @@ obj-$(CONFIG_MMC_SDHCI_ESDHC_IMX)	+= sdhci-esdhc-imx.o  obj-$(CONFIG_MMC_SDHCI_DOVE)		+= sdhci-dove.o  obj-$(CONFIG_MMC_SDHCI_TEGRA)		+= sdhci-tegra.o  obj-$(CONFIG_MMC_SDHCI_OF_ARASAN)	+= sdhci-of-arasan.o +obj-$(CONFIG_MMC_SDHCI_OF_ASPEED)	+= sdhci-of-aspeed.o  obj-$(CONFIG_MMC_SDHCI_OF_AT91)		+= sdhci-of-at91.o  obj-$(CONFIG_MMC_SDHCI_OF_ESDHC)	+= sdhci-of-esdhc.o  obj-$(CONFIG_MMC_SDHCI_OF_HLWD)		+= sdhci-of-hlwd.o diff --git a/drivers/mmc/host/atmel-mci.c b/drivers/mmc/host/atmel-mci.c index 9ee0bc0ce6d0..c26fbe5f2222 100644 --- a/drivers/mmc/host/atmel-mci.c +++ b/drivers/mmc/host/atmel-mci.c @@ -2413,6 +2413,7 @@ static void atmci_get_cap(struct atmel_mci *host)  	case 0x600:  	case 0x500:  		host->caps.has_odd_clk_div = 1; +		/* Fall through */  	case 0x400:  	case 0x300:  		host->caps.has_dma_conf_reg = 1; @@ -2420,13 +2421,16 @@ static void atmci_get_cap(struct atmel_mci *host)  		host->caps.has_cfg_reg = 1;  		host->caps.has_cstor_reg = 1;  		host->caps.has_highspeed = 1; +		/* Fall through */  	case 0x200:  		host->caps.has_rwproof = 1;  		host->caps.need_blksz_mul_4 = 0;  		host->caps.need_notbusy_for_read_ops = 1; +		/* Fall through */  	case 0x100:  		host->caps.has_bad_data_ordering = 0;  		host->caps.need_reset_after_xfer = 0; +		/* Fall through */  	case 0x0:  		break;  	default: diff --git a/drivers/mmc/host/bcm2835.c b/drivers/mmc/host/bcm2835.c index 7e0d3a49c06d..148414d7f0c9 100644 --- a/drivers/mmc/host/bcm2835.c +++ b/drivers/mmc/host/bcm2835.c @@ -597,7 +597,7 @@ static void bcm2835_finish_request(struct bcm2835_host *host)  	struct dma_chan *terminate_chan = NULL;  	struct mmc_request *mrq; -	cancel_delayed_work_sync(&host->timeout_work); +	cancel_delayed_work(&host->timeout_work);  	mrq = host->mrq; @@ -1314,7 +1314,7 @@ static int bcm2835_add_host(struct bcm2835_host *host)  	}  	mmc->max_segs = 128; -	mmc->max_req_size = 524288; +	mmc->max_req_size = min_t(size_t, 524288, dma_max_mapping_size(dev));  	mmc->max_seg_size = mmc->max_req_size;  	mmc->max_blk_size = 1024;  	mmc->max_blk_count =  65535; @@ -1409,7 +1409,6 @@ static int bcm2835_probe(struct platform_device *pdev)  	host->irq = platform_get_irq(pdev, 0);  	if (host->irq <= 0) { -		dev_err(dev, "get IRQ failed\n");  		ret = -EINVAL;  		goto err;  	} diff --git a/drivers/mmc/host/dw_mmc-hi3798cv200.c b/drivers/mmc/host/dw_mmc-hi3798cv200.c index bc51cef47c47..83e1bad0a008 100644 --- a/drivers/mmc/host/dw_mmc-hi3798cv200.c +++ b/drivers/mmc/host/dw_mmc-hi3798cv200.c @@ -66,7 +66,7 @@ static void dw_mci_hi3798cv200_set_ios(struct dw_mci *host, struct mmc_ios *ios)  static int dw_mci_hi3798cv200_execute_tuning(struct dw_mci_slot *slot,  					     u32 opcode)  { -	int degrees[] = { 0, 45, 90, 135, 180, 225, 270, 315 }; +	static const int degrees[] = { 0, 45, 90, 135, 180, 225, 270, 315 };  	struct dw_mci *host = slot->host;  	struct hi3798cv200_priv *priv = host->priv;  	int raise_point = -1, fall_point = -1; diff --git a/drivers/mmc/host/dw_mmc.c b/drivers/mmc/host/dw_mmc.c index eea52e2c5a0c..79c55c7b4afd 100644 --- a/drivers/mmc/host/dw_mmc.c +++ b/drivers/mmc/host/dw_mmc.c @@ -3460,6 +3460,10 @@ int dw_mci_runtime_resume(struct device *dev)  	/* Force setup bus to guarantee available clock output */  	dw_mci_setup_bus(host->slot, true); +	/* Re-enable SDIO interrupts. */ +	if (sdio_irq_claimed(host->slot->mmc)) +		__dw_mci_enable_sdio_irq(host->slot, 1); +  	/* Now that slots are all setup, we can enable card detect */  	dw_mci_enable_cd(host); diff --git a/drivers/mmc/host/jz4740_mmc.c b/drivers/mmc/host/jz4740_mmc.c index ffdbfaadd3f2..f816c06ef916 100644 --- a/drivers/mmc/host/jz4740_mmc.c +++ b/drivers/mmc/host/jz4740_mmc.c @@ -25,8 +25,6 @@  #include <asm/cacheflush.h> -#include <asm/mach-jz4740/dma.h> -  #define JZ_REG_MMC_STRPCL	0x00  #define JZ_REG_MMC_STATUS	0x04  #define JZ_REG_MMC_CLKRT	0x08 @@ -186,9 +184,9 @@ static void jz4740_mmc_write_irq_reg(struct jz4740_mmc_host *host,  				     uint32_t val)  {  	if (host->version >= JZ_MMC_JZ4780) -		return writel(val, host->base + JZ_REG_MMC_IREG); +		writel(val, host->base + JZ_REG_MMC_IREG);  	else -		return writew(val, host->base + JZ_REG_MMC_IREG); +		writew(val, host->base + JZ_REG_MMC_IREG);  }  static uint32_t jz4740_mmc_read_irq_reg(struct jz4740_mmc_host *host) @@ -292,11 +290,9 @@ static int jz4740_mmc_start_dma_transfer(struct jz4740_mmc_host *host,  	if (data->flags & MMC_DATA_WRITE) {  		conf.direction = DMA_MEM_TO_DEV;  		conf.dst_addr = host->mem_res->start + JZ_REG_MMC_TXFIFO; -		conf.slave_id = JZ4740_DMA_TYPE_MMC_TRANSMIT;  	} else {  		conf.direction = DMA_DEV_TO_MEM;  		conf.src_addr = host->mem_res->start + JZ_REG_MMC_RXFIFO; -		conf.slave_id = JZ4740_DMA_TYPE_MMC_RECEIVE;  	}  	sg_count = jz4740_mmc_prepare_dma_data(host, data, COOKIE_MAPPED); @@ -820,14 +816,14 @@ static irqreturn_t jz_mmc_irq(int irq, void *devid)  			del_timer(&host->timeout_timer);  			if (status & JZ_MMC_STATUS_TIMEOUT_RES) { -					cmd->error = -ETIMEDOUT; +				cmd->error = -ETIMEDOUT;  			} else if (status & JZ_MMC_STATUS_CRC_RES_ERR) { -					cmd->error = -EIO; +				cmd->error = -EIO;  			} else if (status & (JZ_MMC_STATUS_CRC_READ_ERROR |  				    JZ_MMC_STATUS_CRC_WRITE_ERROR)) { -					if (cmd->data) -							cmd->data->error = -EIO; -					cmd->error = -EIO; +				if (cmd->data) +					cmd->data->error = -EIO; +				cmd->error = -EIO;  			}  			jz4740_mmc_set_irq_enabled(host, irq_reg, false); @@ -969,7 +965,6 @@ static int jz4740_mmc_probe(struct platform_device* pdev)  	host->irq = platform_get_irq(pdev, 0);  	if (host->irq < 0) {  		ret = host->irq; -		dev_err(&pdev->dev, "Failed to get platform irq: %d\n", ret);  		goto err_free_host;  	} diff --git a/drivers/mmc/host/meson-gx-mmc.c b/drivers/mmc/host/meson-gx-mmc.c index 037311db3551..e712315c7e8d 100644 --- a/drivers/mmc/host/meson-gx-mmc.c +++ b/drivers/mmc/host/meson-gx-mmc.c @@ -1091,7 +1091,6 @@ static int meson_mmc_probe(struct platform_device *pdev)  	host->irq = platform_get_irq(pdev, 0);  	if (host->irq <= 0) { -		dev_err(&pdev->dev, "failed to get interrupt resource.\n");  		ret = -EINVAL;  		goto free_host;  	} diff --git a/drivers/mmc/host/mmc_spi.c b/drivers/mmc/host/mmc_spi.c index 19544b121276..66e354d51ee9 100644 --- a/drivers/mmc/host/mmc_spi.c +++ b/drivers/mmc/host/mmc_spi.c @@ -891,7 +891,7 @@ mmc_spi_data_do(struct mmc_spi_host *host, struct mmc_command *cmd,  	/* Handle scatterlist segments one at a time, with synch for  	 * each 512-byte block  	 */ -	for (sg = data->sg, n_sg = data->sg_len; n_sg; n_sg--, sg++) { +	for_each_sg(data->sg, sg, data->sg_len, n_sg) {  		int			status = 0;  		dma_addr_t		dma_addr = 0;  		void			*kmap_addr; diff --git a/drivers/mmc/host/mmci.c b/drivers/mmc/host/mmci.c index b8554bf38f72..c37e70dbe250 100644 --- a/drivers/mmc/host/mmci.c +++ b/drivers/mmc/host/mmci.c @@ -1219,47 +1219,58 @@ mmci_cmd_irq(struct mmci_host *host, struct mmc_command *cmd,  	      (MCI_CMDCRCFAIL|MCI_CMDTIMEOUT|MCI_CMDSENT|MCI_CMDRESPEND)))  		return; -	/* -	 * ST Micro variant: handle busy detection. -	 */ +	/* Handle busy detection on DAT0 if the variant supports it. */  	if (busy_resp && host->variant->busy_detect) { -		/* We are busy with a command, return */ -		if (host->busy_status && -		    (status & host->variant->busy_detect_flag)) -			return; -  		/* -		 * We were not busy, but we now got a busy response on -		 * something that was not an error, and we double-check -		 * that the special busy status bit is still set before -		 * proceeding. +		 * Before unmasking for the busy end IRQ, confirm that the +		 * command was sent successfully. To keep track of having a +		 * command in-progress, waiting for busy signaling to end, +		 * store the status in host->busy_status. +		 * +		 * Note that, the card may need a couple of clock cycles before +		 * it starts signaling busy on DAT0, hence re-read the +		 * MMCISTATUS register here, to allow the busy bit to be set. +		 * Potentially we may even need to poll the register for a +		 * while, to allow it to be set, but tests indicates that it +		 * isn't needed.  		 */  		if (!host->busy_status &&  		    !(status & (MCI_CMDCRCFAIL|MCI_CMDTIMEOUT)) &&  		    (readl(base + MMCISTATUS) & host->variant->busy_detect_flag)) { -			/* Clear the busy start IRQ */ -			writel(host->variant->busy_detect_mask, -			       host->base + MMCICLEAR); - -			/* Unmask the busy end IRQ */  			writel(readl(base + MMCIMASK0) |  			       host->variant->busy_detect_mask,  			       base + MMCIMASK0); -			/* -			 * Now cache the last response status code (until -			 * the busy bit goes low), and return. -			 */ +  			host->busy_status =  				status & (MCI_CMDSENT|MCI_CMDRESPEND);  			return;  		}  		/* -		 * At this point we are not busy with a command, we have -		 * not received a new busy request, clear and mask the busy -		 * end IRQ and fall through to process the IRQ. +		 * If there is a command in-progress that has been successfully +		 * sent, then bail out if busy status is set and wait for the +		 * busy end IRQ. +		 * +		 * Note that, the HW triggers an IRQ on both edges while +		 * monitoring DAT0 for busy completion, but there is only one +		 * status bit in MMCISTATUS for the busy state. Therefore +		 * both the start and the end interrupts needs to be cleared, +		 * one after the other. So, clear the busy start IRQ here. +		 */ +		if (host->busy_status && +		    (status & host->variant->busy_detect_flag)) { +			writel(host->variant->busy_detect_mask, +			       host->base + MMCICLEAR); +			return; +		} + +		/* +		 * If there is a command in-progress that has been successfully +		 * sent and the busy bit isn't set, it means we have received +		 * the busy end IRQ. Clear and mask the IRQ, then continue to +		 * process the command.  		 */  		if (host->busy_status) { @@ -1505,14 +1516,8 @@ static irqreturn_t mmci_irq(int irq, void *dev_id)  		}  		/* -		 * We intentionally clear the MCI_ST_CARDBUSY IRQ (if it's -		 * enabled) in mmci_cmd_irq() function where ST Micro busy -		 * detection variant is handled. Considering the HW seems to be -		 * triggering the IRQ on both edges while monitoring DAT0 for -		 * busy completion and that same status bit is used to monitor -		 * start and end of busy detection, special care must be taken -		 * to make sure that both start and end interrupts are always -		 * cleared one after the other. +		 * Busy detection is managed by mmci_cmd_irq(), including to +		 * clear the corresponding IRQ.  		 */  		status &= readl(host->base + MMCIMASK0);  		if (host->variant->busy_detect) diff --git a/drivers/mmc/host/mtk-sd.c b/drivers/mmc/host/mtk-sd.c index 33f4b6387ef7..189e42674d85 100644 --- a/drivers/mmc/host/mtk-sd.c +++ b/drivers/mmc/host/mtk-sd.c @@ -192,6 +192,7 @@  #define SDC_STS_CMDBUSY         (0x1 << 1)	/* RW */  #define SDC_STS_SWR_COMPL       (0x1 << 31)	/* RW */ +#define SDC_DAT1_IRQ_TRIGGER	(0x1 << 19)	/* RW */  /* SDC_ADV_CFG0 mask */  #define SDC_RX_ENHANCE_EN	(0x1 << 20)	/* RW */ @@ -328,6 +329,7 @@ struct mt_bdma_desc {  	u32 ptr;  	u32 bd_data_len;  #define BDMA_DESC_BUFLEN	(0xffff) /* bit0 ~ bit15 */ +#define BDMA_DESC_BUFLEN_EXT	(0xffffff) /* bit0 ~ bit23 */  };  struct msdc_dma { @@ -641,8 +643,14 @@ static inline void msdc_dma_setup(struct msdc_host *host, struct msdc_dma *dma,  			bd[j].bd_info |= (upper_32_bits(dma_address) & 0xf)  					 << 28;  		} -		bd[j].bd_data_len &= ~BDMA_DESC_BUFLEN; -		bd[j].bd_data_len |= (dma_len & BDMA_DESC_BUFLEN); + +		if (host->dev_comp->support_64g) { +			bd[j].bd_data_len &= ~BDMA_DESC_BUFLEN_EXT; +			bd[j].bd_data_len |= (dma_len & BDMA_DESC_BUFLEN_EXT); +		} else { +			bd[j].bd_data_len &= ~BDMA_DESC_BUFLEN; +			bd[j].bd_data_len |= (dma_len & BDMA_DESC_BUFLEN); +		}  		if (j == data->sg_count - 1) /* the last bd */  			bd[j].bd_info |= BDMA_DESC_EOL; @@ -1071,11 +1079,13 @@ static bool msdc_cmd_done(struct msdc_host *host, int events,  	}  	if (!sbc_error && !(events & MSDC_INT_CMDRDY)) { -		if (cmd->opcode != MMC_SEND_TUNING_BLOCK && -		    cmd->opcode != MMC_SEND_TUNING_BLOCK_HS200) +		if (events & MSDC_INT_CMDTMO || +		    (cmd->opcode != MMC_SEND_TUNING_BLOCK && +		     cmd->opcode != MMC_SEND_TUNING_BLOCK_HS200))  			/*  			 * should not clear fifo/interrupt as the tune data -			 * may have alreay come. +			 * may have alreay come when cmd19/cmd21 gets response +			 * CRC error.  			 */  			msdc_reset_hw(host);  		if (events & MSDC_INT_RSPCRCERR) { @@ -1568,6 +1578,7 @@ static void msdc_init_hw(struct msdc_host *host)  	/* Config SDIO device detect interrupt function */  	sdr_clr_bits(host->base + SDC_CFG, SDC_CFG_SDIOIDE); +	sdr_set_bits(host->base + SDC_ADV_CFG0, SDC_DAT1_IRQ_TRIGGER);  	/* Configure to default data timeout */  	sdr_set_field(host->base + SDC_CFG, SDC_CFG_DTOC, 3); @@ -2275,7 +2286,10 @@ static int msdc_drv_probe(struct platform_device *pdev)  	mmc->caps |= MMC_CAP_ERASE | MMC_CAP_CMD23;  	/* MMC core transfer sizes tunable parameters */  	mmc->max_segs = MAX_BD_NUM; -	mmc->max_seg_size = BDMA_DESC_BUFLEN; +	if (host->dev_comp->support_64g) +		mmc->max_seg_size = BDMA_DESC_BUFLEN_EXT; +	else +		mmc->max_seg_size = BDMA_DESC_BUFLEN;  	mmc->max_blk_size = 2048;  	mmc->max_req_size = 512 * 1024;  	mmc->max_blk_count = mmc->max_req_size / 512; @@ -2421,6 +2435,9 @@ static void msdc_restore_reg(struct msdc_host *host)  	} else {  		writel(host->save_para.pad_tune, host->base + tune_reg);  	} + +	if (sdio_irq_claimed(host->mmc)) +		__msdc_enable_sdio_irq(host, 1);  }  static int msdc_runtime_suspend(struct device *dev) diff --git a/drivers/mmc/host/mxcmmc.c b/drivers/mmc/host/mxcmmc.c index 750604f7fac9..011b59a3602e 100644 --- a/drivers/mmc/host/mxcmmc.c +++ b/drivers/mmc/host/mxcmmc.c @@ -1010,10 +1010,8 @@ static int mxcmci_probe(struct platform_device *pdev)  	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);  	irq = platform_get_irq(pdev, 0); -	if (irq < 0) { -		dev_err(&pdev->dev, "failed to get IRQ: %d\n", irq); +	if (irq < 0)  		return irq; -	}  	mmc = mmc_alloc_host(sizeof(*host), &pdev->dev);  	if (!mmc) diff --git a/drivers/mmc/host/mxs-mmc.c b/drivers/mmc/host/mxs-mmc.c index b334e81c5cab..78e7e350655c 100644 --- a/drivers/mmc/host/mxs-mmc.c +++ b/drivers/mmc/host/mxs-mmc.c @@ -571,7 +571,6 @@ static int mxs_mmc_probe(struct platform_device *pdev)  	struct device_node *np = pdev->dev.of_node;  	struct mxs_mmc_host *host;  	struct mmc_host *mmc; -	struct resource *iores;  	int ret = 0, irq_err;  	struct regulator *reg_vmmc;  	struct mxs_ssp *ssp; @@ -587,8 +586,7 @@ static int mxs_mmc_probe(struct platform_device *pdev)  	host = mmc_priv(mmc);  	ssp = &host->ssp;  	ssp->dev = &pdev->dev; -	iores = platform_get_resource(pdev, IORESOURCE_MEM, 0); -	ssp->base = devm_ioremap_resource(&pdev->dev, iores); +	ssp->base = devm_platform_ioremap_resource(pdev, 0);  	if (IS_ERR(ssp->base)) {  		ret = PTR_ERR(ssp->base);  		goto out_mmc_free; diff --git a/drivers/mmc/host/renesas_sdhi_core.c b/drivers/mmc/host/renesas_sdhi_core.c index 64d3b5fb7fe5..d4ada5cca2d1 100644 --- a/drivers/mmc/host/renesas_sdhi_core.c +++ b/drivers/mmc/host/renesas_sdhi_core.c @@ -124,7 +124,7 @@ static unsigned int renesas_sdhi_clk_update(struct tmio_mmc_host *host,  {  	struct renesas_sdhi *priv = host_to_priv(host);  	unsigned int freq, diff, best_freq = 0, diff_min = ~0; -	int i, ret; +	int i;  	/* tested only on R-Car Gen2+ currently; may work for others */  	if (!(host->pdata->flags & TMIO_MMC_MIN_RCAR2)) @@ -153,9 +153,9 @@ static unsigned int renesas_sdhi_clk_update(struct tmio_mmc_host *host,  		}  	} -	ret = clk_set_rate(priv->clk, best_freq); +	clk_set_rate(priv->clk, best_freq); -	return ret == 0 ? best_freq : clk_get_rate(priv->clk); +	return clk_get_rate(priv->clk);  }  static void renesas_sdhi_set_clock(struct tmio_mmc_host *host, @@ -166,10 +166,13 @@ static void renesas_sdhi_set_clock(struct tmio_mmc_host *host,  	sd_ctrl_write16(host, CTL_SD_CARD_CLK_CTL, ~CLK_CTL_SCLKEN &  		sd_ctrl_read16(host, CTL_SD_CARD_CLK_CTL)); -	if (new_clock == 0) +	if (new_clock == 0) { +		host->mmc->actual_clock = 0;  		goto out; +	} -	clock = renesas_sdhi_clk_update(host, new_clock) / 512; +	host->mmc->actual_clock = renesas_sdhi_clk_update(host, new_clock); +	clock = host->mmc->actual_clock / 512;  	for (clk = 0x80000080; new_clock >= (clock << 1); clk >>= 1)  		clock <<= 1; @@ -774,8 +777,6 @@ int renesas_sdhi_probe(struct platform_device *pdev,  	/* All SDHI have SDIO status bits which must be 1 */  	mmc_data->flags |= TMIO_MMC_SDIO_STATUS_SETBITS; -	pm_runtime_enable(&pdev->dev); -  	ret = renesas_sdhi_clk_enable(host);  	if (ret)  		goto efree; @@ -856,8 +857,6 @@ edisclk:  efree:  	tmio_mmc_host_free(host); -	pm_runtime_disable(&pdev->dev); -  	return ret;  }  EXPORT_SYMBOL_GPL(renesas_sdhi_probe); @@ -869,8 +868,6 @@ int renesas_sdhi_remove(struct platform_device *pdev)  	tmio_mmc_host_remove(host);  	renesas_sdhi_clk_disable(host); -	pm_runtime_disable(&pdev->dev); -  	return 0;  }  EXPORT_SYMBOL_GPL(renesas_sdhi_remove); diff --git a/drivers/mmc/host/renesas_sdhi_internal_dmac.c b/drivers/mmc/host/renesas_sdhi_internal_dmac.c index 751fe91c7571..a66f8d6d61d1 100644 --- a/drivers/mmc/host/renesas_sdhi_internal_dmac.c +++ b/drivers/mmc/host/renesas_sdhi_internal_dmac.c @@ -106,7 +106,7 @@ static const struct renesas_sdhi_of_data of_rcar_gen3_compatible = {  			  TMIO_MMC_HAVE_CBSY | TMIO_MMC_MIN_RCAR2,  	.capabilities	= MMC_CAP_SD_HIGHSPEED | MMC_CAP_SDIO_IRQ |  			  MMC_CAP_CMD23, -	.capabilities2	= MMC_CAP2_NO_WRITE_PROTECT, +	.capabilities2	= MMC_CAP2_NO_WRITE_PROTECT | MMC_CAP2_MERGE_CAPABLE,  	.bus_shift	= 2,  	.scc_offset	= 0x1000,  	.taps		= rcar_gen3_scc_taps, diff --git a/drivers/mmc/host/renesas_sdhi_sys_dmac.c b/drivers/mmc/host/renesas_sdhi_sys_dmac.c index 1d29b822efb8..13ff023fbee9 100644 --- a/drivers/mmc/host/renesas_sdhi_sys_dmac.c +++ b/drivers/mmc/host/renesas_sdhi_sys_dmac.c @@ -68,26 +68,6 @@ static const struct renesas_sdhi_of_data of_rcar_gen2_compatible = {  	.max_blk_count	= UINT_MAX / TMIO_MAX_BLK_SIZE,  }; -/* Definitions for sampling clocks */ -static struct renesas_sdhi_scc rcar_gen3_scc_taps[] = { -	{ -		.clk_rate = 0, -		.tap = 0x00000300, -	}, -}; - -static const struct renesas_sdhi_of_data of_rcar_gen3_compatible = { -	.tmio_flags	= TMIO_MMC_HAS_IDLE_WAIT | TMIO_MMC_CLK_ACTUAL | -			  TMIO_MMC_HAVE_CBSY | TMIO_MMC_MIN_RCAR2, -	.capabilities	= MMC_CAP_SD_HIGHSPEED | MMC_CAP_SDIO_IRQ | -			  MMC_CAP_CMD23, -	.capabilities2	= MMC_CAP2_NO_WRITE_PROTECT, -	.bus_shift	= 2, -	.scc_offset	= 0x1000, -	.taps		= rcar_gen3_scc_taps, -	.taps_num	= ARRAY_SIZE(rcar_gen3_scc_taps), -}; -  static const struct of_device_id renesas_sdhi_sys_dmac_of_match[] = {  	{ .compatible = "renesas,sdhi-sh73a0", .data = &of_default_cfg, },  	{ .compatible = "renesas,sdhi-r8a73a4", .data = &of_default_cfg, }, @@ -102,11 +82,8 @@ static const struct of_device_id renesas_sdhi_sys_dmac_of_match[] = {  	{ .compatible = "renesas,sdhi-r8a7792", .data = &of_rcar_gen2_compatible, },  	{ .compatible = "renesas,sdhi-r8a7793", .data = &of_rcar_gen2_compatible, },  	{ .compatible = "renesas,sdhi-r8a7794", .data = &of_rcar_gen2_compatible, }, -	{ .compatible = "renesas,sdhi-r8a7795", .data = &of_rcar_gen3_compatible, }, -	{ .compatible = "renesas,sdhi-r8a7796", .data = &of_rcar_gen3_compatible, },  	{ .compatible = "renesas,rcar-gen1-sdhi", .data = &of_rcar_gen1_compatible, },  	{ .compatible = "renesas,rcar-gen2-sdhi", .data = &of_rcar_gen2_compatible, }, -	{ .compatible = "renesas,rcar-gen3-sdhi", .data = &of_rcar_gen3_compatible, },  	{ .compatible = "renesas,sdhi-shmobile" },  	{},  }; @@ -470,21 +447,8 @@ static const struct tmio_mmc_dma_ops renesas_sdhi_sys_dmac_dma_ops = {  	.dataend = renesas_sdhi_sys_dmac_dataend_dma,  }; -/* - * Whitelist of specific R-Car Gen3 SoC ES versions to use this DMAC - * implementation. Currently empty as all supported ES versions use - * the internal DMAC. - */ -static const struct soc_device_attribute gen3_soc_whitelist[] = { -        { /* sentinel */ } -}; -  static int renesas_sdhi_sys_dmac_probe(struct platform_device *pdev)  { -	if (of_device_get_match_data(&pdev->dev) == &of_rcar_gen3_compatible && -	    !soc_device_match(gen3_soc_whitelist)) -		return -ENODEV; -  	return renesas_sdhi_probe(pdev, &renesas_sdhi_sys_dmac_dma_ops);  } diff --git a/drivers/mmc/host/s3cmci.c b/drivers/mmc/host/s3cmci.c index ccc5f095775f..bce9c33bc4b5 100644 --- a/drivers/mmc/host/s3cmci.c +++ b/drivers/mmc/host/s3cmci.c @@ -1614,7 +1614,6 @@ static int s3cmci_probe(struct platform_device *pdev)  	host->irq = platform_get_irq(pdev, 0);  	if (host->irq <= 0) { -		dev_err(&pdev->dev, "failed to get interrupt resource.\n");  		ret = -EINVAL;  		goto probe_iounmap;  	} diff --git a/drivers/mmc/host/sdhci-cadence.c b/drivers/mmc/host/sdhci-cadence.c index 163d1cf4367e..ae0ec27dd7cc 100644 --- a/drivers/mmc/host/sdhci-cadence.c +++ b/drivers/mmc/host/sdhci-cadence.c @@ -337,10 +337,10 @@ static int sdhci_cdns_probe(struct platform_device *pdev)  	struct sdhci_pltfm_host *pltfm_host;  	struct sdhci_cdns_priv *priv;  	struct clk *clk; -	size_t priv_size;  	unsigned int nr_phy_params;  	int ret;  	struct device *dev = &pdev->dev; +	static const u16 version = SDHCI_SPEC_400 << SDHCI_SPEC_VER_SHIFT;  	clk = devm_clk_get(dev, NULL);  	if (IS_ERR(clk)) @@ -351,8 +351,8 @@ static int sdhci_cdns_probe(struct platform_device *pdev)  		return ret;  	nr_phy_params = sdhci_cdns_phy_param_count(dev->of_node); -	priv_size = sizeof(*priv) + sizeof(priv->phy_params[0]) * nr_phy_params; -	host = sdhci_pltfm_init(pdev, &sdhci_cdns_pltfm_data, priv_size); +	host = sdhci_pltfm_init(pdev, &sdhci_cdns_pltfm_data, +				struct_size(priv, phy_params, nr_phy_params));  	if (IS_ERR(host)) {  		ret = PTR_ERR(host);  		goto disable_clk; @@ -369,6 +369,8 @@ static int sdhci_cdns_probe(struct platform_device *pdev)  	host->mmc_host_ops.execute_tuning = sdhci_cdns_execute_tuning;  	host->mmc_host_ops.hs400_enhanced_strobe =  				sdhci_cdns_hs400_enhanced_strobe; +	sdhci_enable_v4_mode(host); +	__sdhci_read_caps(host, &version, NULL, NULL);  	sdhci_get_of_property(pdev); diff --git a/drivers/mmc/host/sdhci-esdhc-imx.c b/drivers/mmc/host/sdhci-esdhc-imx.c index 776a94216248..1c988d6a2433 100644 --- a/drivers/mmc/host/sdhci-esdhc-imx.c +++ b/drivers/mmc/host/sdhci-esdhc-imx.c @@ -1666,12 +1666,10 @@ static int sdhci_esdhc_runtime_suspend(struct device *dev)  	if (host->tuning_mode != SDHCI_TUNING_MODE_3)  		mmc_retune_needed(host->mmc); -	if (!sdhci_sdio_irq_enabled(host)) { -		imx_data->actual_clock = host->mmc->actual_clock; -		esdhc_pltfm_set_clock(host, 0); -		clk_disable_unprepare(imx_data->clk_per); -		clk_disable_unprepare(imx_data->clk_ipg); -	} +	imx_data->actual_clock = host->mmc->actual_clock; +	esdhc_pltfm_set_clock(host, 0); +	clk_disable_unprepare(imx_data->clk_per); +	clk_disable_unprepare(imx_data->clk_ipg);  	clk_disable_unprepare(imx_data->clk_ahb);  	if (imx_data->socdata->flags & ESDHC_FLAG_PMQOS) @@ -1695,15 +1693,15 @@ static int sdhci_esdhc_runtime_resume(struct device *dev)  	if (err)  		goto remove_pm_qos_request; -	if (!sdhci_sdio_irq_enabled(host)) { -		err = clk_prepare_enable(imx_data->clk_per); -		if (err) -			goto disable_ahb_clk; -		err = clk_prepare_enable(imx_data->clk_ipg); -		if (err) -			goto disable_per_clk; -		esdhc_pltfm_set_clock(host, imx_data->actual_clock); -	} +	err = clk_prepare_enable(imx_data->clk_per); +	if (err) +		goto disable_ahb_clk; + +	err = clk_prepare_enable(imx_data->clk_ipg); +	if (err) +		goto disable_per_clk; + +	esdhc_pltfm_set_clock(host, imx_data->actual_clock);  	err = sdhci_runtime_resume_host(host, 0);  	if (err) @@ -1715,11 +1713,9 @@ static int sdhci_esdhc_runtime_resume(struct device *dev)  	return err;  disable_ipg_clk: -	if (!sdhci_sdio_irq_enabled(host)) -		clk_disable_unprepare(imx_data->clk_ipg); +	clk_disable_unprepare(imx_data->clk_ipg);  disable_per_clk: -	if (!sdhci_sdio_irq_enabled(host)) -		clk_disable_unprepare(imx_data->clk_per); +	clk_disable_unprepare(imx_data->clk_per);  disable_ahb_clk:  	clk_disable_unprepare(imx_data->clk_ahb);  remove_pm_qos_request: diff --git a/drivers/mmc/host/sdhci-iproc.c b/drivers/mmc/host/sdhci-iproc.c index 2feb4ef32035..2b9cdcd1dd9d 100644 --- a/drivers/mmc/host/sdhci-iproc.c +++ b/drivers/mmc/host/sdhci-iproc.c @@ -261,8 +261,17 @@ static const struct sdhci_iproc_data bcm2835_data = {  	.mmc_caps = 0x00000000,  }; +static const struct sdhci_pltfm_data sdhci_bcm2711_pltfm_data = { +	.ops = &sdhci_iproc_32only_ops, +}; + +static const struct sdhci_iproc_data bcm2711_data = { +	.pdata = &sdhci_bcm2711_pltfm_data, +}; +  static const struct of_device_id sdhci_iproc_of_match[] = {  	{ .compatible = "brcm,bcm2835-sdhci", .data = &bcm2835_data }, +	{ .compatible = "brcm,bcm2711-emmc2", .data = &bcm2711_data },  	{ .compatible = "brcm,sdhci-iproc-cygnus", .data = &iproc_cygnus_data},  	{ .compatible = "brcm,sdhci-iproc", .data = &iproc_data },  	{ } diff --git a/drivers/mmc/host/sdhci-msm.c b/drivers/mmc/host/sdhci-msm.c index 9cf14b359c14..b75c82d8d6c1 100644 --- a/drivers/mmc/host/sdhci-msm.c +++ b/drivers/mmc/host/sdhci-msm.c @@ -1917,8 +1917,6 @@ static int sdhci_msm_probe(struct platform_device *pdev)  	/* Setup IRQ for handling power/voltage tasks with PMIC */  	msm_host->pwr_irq = platform_get_irq_byname(pdev, "pwr_irq");  	if (msm_host->pwr_irq < 0) { -		dev_err(&pdev->dev, "Get pwr_irq failed (%d)\n", -			msm_host->pwr_irq);  		ret = msm_host->pwr_irq;  		goto clk_disable;  	} diff --git a/drivers/mmc/host/sdhci-of-arasan.c b/drivers/mmc/host/sdhci-of-arasan.c index b12abf9b15f2..7023cbec4017 100644 --- a/drivers/mmc/host/sdhci-of-arasan.c +++ b/drivers/mmc/host/sdhci-of-arasan.c @@ -114,6 +114,12 @@ static const struct sdhci_arasan_soc_ctl_map rk3399_soc_ctl_map = {  	.hiword_update = true,  }; +static const struct sdhci_arasan_soc_ctl_map intel_lgm_emmc_soc_ctl_map = { +	.baseclkfreq = { .reg = 0xa0, .width = 8, .shift = 2 }, +	.clockmultiplier = { .reg = 0, .width = -1, .shift = -1 }, +	.hiword_update = false, +}; +  /**   * sdhci_arasan_syscon_write - Write to a field in soc_ctl registers   * @@ -373,6 +379,11 @@ static struct sdhci_arasan_of_data sdhci_arasan_rk3399_data = {  	.pdata = &sdhci_arasan_cqe_pdata,  }; +static struct sdhci_arasan_of_data intel_lgm_emmc_data = { +	.soc_ctl_map = &intel_lgm_emmc_soc_ctl_map, +	.pdata = &sdhci_arasan_cqe_pdata, +}; +  #ifdef CONFIG_PM_SLEEP  /**   * sdhci_arasan_suspend - Suspend method for the driver @@ -474,6 +485,10 @@ static const struct of_device_id sdhci_arasan_of_match[] = {  		.compatible = "rockchip,rk3399-sdhci-5.1",  		.data = &sdhci_arasan_rk3399_data,  	}, +	{ +		.compatible = "intel,lgm-sdhci-5.1-emmc", +		.data = &intel_lgm_emmc_data, +	},  	/* Generic compatible below here */  	{  		.compatible = "arasan,sdhci-8.9a", diff --git a/drivers/mmc/host/sdhci-of-aspeed.c b/drivers/mmc/host/sdhci-of-aspeed.c new file mode 100644 index 000000000000..8962f6664381 --- /dev/null +++ b/drivers/mmc/host/sdhci-of-aspeed.c @@ -0,0 +1,342 @@ +// SPDX-License-Identifier: GPL-2.0-or-later +/* Copyright (C) 2019 ASPEED Technology Inc. */ +/* Copyright (C) 2019 IBM Corp. */ + +#include <linux/clk.h> +#include <linux/delay.h> +#include <linux/device.h> +#include <linux/io.h> +#include <linux/mmc/host.h> +#include <linux/module.h> +#include <linux/of_address.h> +#include <linux/of.h> +#include <linux/of_platform.h> +#include <linux/platform_device.h> +#include <linux/spinlock.h> + +#include "sdhci-pltfm.h" + +#define ASPEED_SDC_INFO		0x00 +#define   ASPEED_SDC_S1MMC8	BIT(25) +#define   ASPEED_SDC_S0MMC8	BIT(24) + +struct aspeed_sdc { +	struct clk *clk; +	struct resource *res; + +	spinlock_t lock; +	void __iomem *regs; +}; + +struct aspeed_sdhci { +	struct aspeed_sdc *parent; +	u32 width_mask; +}; + +static void aspeed_sdc_configure_8bit_mode(struct aspeed_sdc *sdc, +					   struct aspeed_sdhci *sdhci, +					   bool bus8) +{ +	u32 info; + +	/* Set/clear 8 bit mode */ +	spin_lock(&sdc->lock); +	info = readl(sdc->regs + ASPEED_SDC_INFO); +	if (bus8) +		info |= sdhci->width_mask; +	else +		info &= ~sdhci->width_mask; +	writel(info, sdc->regs + ASPEED_SDC_INFO); +	spin_unlock(&sdc->lock); +} + +static void aspeed_sdhci_set_clock(struct sdhci_host *host, unsigned int clock) +{ +	struct sdhci_pltfm_host *pltfm_host; +	unsigned long parent; +	int div; +	u16 clk; + +	pltfm_host = sdhci_priv(host); +	parent = clk_get_rate(pltfm_host->clk); + +	sdhci_writew(host, 0, SDHCI_CLOCK_CONTROL); + +	if (clock == 0) +		return; + +	if (WARN_ON(clock > host->max_clk)) +		clock = host->max_clk; + +	for (div = 1; div < 256; div *= 2) { +		if ((parent / div) <= clock) +			break; +	} +	div >>= 1; + +	clk = div << SDHCI_DIVIDER_SHIFT; + +	sdhci_enable_clk(host, clk); +} + +static unsigned int aspeed_sdhci_get_max_clock(struct sdhci_host *host) +{ +	if (host->mmc->f_max) +		return host->mmc->f_max; + +	return sdhci_pltfm_clk_get_max_clock(host); +} + +static void aspeed_sdhci_set_bus_width(struct sdhci_host *host, int width) +{ +	struct sdhci_pltfm_host *pltfm_priv; +	struct aspeed_sdhci *aspeed_sdhci; +	struct aspeed_sdc *aspeed_sdc; +	u8 ctrl; + +	pltfm_priv = sdhci_priv(host); +	aspeed_sdhci = sdhci_pltfm_priv(pltfm_priv); +	aspeed_sdc = aspeed_sdhci->parent; + +	/* Set/clear 8-bit mode */ +	aspeed_sdc_configure_8bit_mode(aspeed_sdc, aspeed_sdhci, +				       width == MMC_BUS_WIDTH_8); + +	/* Set/clear 1 or 4 bit mode */ +	ctrl = sdhci_readb(host, SDHCI_HOST_CONTROL); +	if (width == MMC_BUS_WIDTH_4) +		ctrl |= SDHCI_CTRL_4BITBUS; +	else +		ctrl &= ~SDHCI_CTRL_4BITBUS; +	sdhci_writeb(host, ctrl, SDHCI_HOST_CONTROL); +} + +static const struct sdhci_ops aspeed_sdhci_ops = { +	.set_clock = aspeed_sdhci_set_clock, +	.get_max_clock = aspeed_sdhci_get_max_clock, +	.set_bus_width = aspeed_sdhci_set_bus_width, +	.get_timeout_clock = sdhci_pltfm_clk_get_max_clock, +	.reset = sdhci_reset, +	.set_uhs_signaling = sdhci_set_uhs_signaling, +}; + +static const struct sdhci_pltfm_data aspeed_sdhci_pdata = { +	.ops = &aspeed_sdhci_ops, +	.quirks = SDHCI_QUIRK_CAP_CLOCK_BASE_BROKEN, +}; + +static inline int aspeed_sdhci_calculate_slot(struct aspeed_sdhci *dev, +					      struct resource *res) +{ +	resource_size_t delta; + +	if (!res || resource_type(res) != IORESOURCE_MEM) +		return -EINVAL; + +	if (res->start < dev->parent->res->start) +		return -EINVAL; + +	delta = res->start - dev->parent->res->start; +	if (delta & (0x100 - 1)) +		return -EINVAL; + +	return (delta / 0x100) - 1; +} + +static int aspeed_sdhci_probe(struct platform_device *pdev) +{ +	struct sdhci_pltfm_host *pltfm_host; +	struct aspeed_sdhci *dev; +	struct sdhci_host *host; +	struct resource *res; +	int slot; +	int ret; + +	host = sdhci_pltfm_init(pdev, &aspeed_sdhci_pdata, sizeof(*dev)); +	if (IS_ERR(host)) +		return PTR_ERR(host); + +	pltfm_host = sdhci_priv(host); +	dev = sdhci_pltfm_priv(pltfm_host); +	dev->parent = dev_get_drvdata(pdev->dev.parent); + +	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	slot = aspeed_sdhci_calculate_slot(dev, res); + +	if (slot < 0) +		return slot; +	else if (slot >= 2) +		return -EINVAL; + +	dev_info(&pdev->dev, "Configuring for slot %d\n", slot); +	dev->width_mask = !slot ? ASPEED_SDC_S0MMC8 : ASPEED_SDC_S1MMC8; + +	sdhci_get_of_property(pdev); + +	pltfm_host->clk = devm_clk_get(&pdev->dev, NULL); +	if (IS_ERR(pltfm_host->clk)) +		return PTR_ERR(pltfm_host->clk); + +	ret = clk_prepare_enable(pltfm_host->clk); +	if (ret) { +		dev_err(&pdev->dev, "Unable to enable SDIO clock\n"); +		goto err_pltfm_free; +	} + +	ret = mmc_of_parse(host->mmc); +	if (ret) +		goto err_sdhci_add; + +	ret = sdhci_add_host(host); +	if (ret) +		goto err_sdhci_add; + +	return 0; + +err_sdhci_add: +	clk_disable_unprepare(pltfm_host->clk); +err_pltfm_free: +	sdhci_pltfm_free(pdev); +	return ret; +} + +static int aspeed_sdhci_remove(struct platform_device *pdev) +{ +	struct sdhci_pltfm_host *pltfm_host; +	struct sdhci_host *host; +	int dead = 0; + +	host = platform_get_drvdata(pdev); +	pltfm_host = sdhci_priv(host); + +	sdhci_remove_host(host, dead); + +	clk_disable_unprepare(pltfm_host->clk); + +	sdhci_pltfm_free(pdev); + +	return 0; +} + +static const struct of_device_id aspeed_sdhci_of_match[] = { +	{ .compatible = "aspeed,ast2400-sdhci", }, +	{ .compatible = "aspeed,ast2500-sdhci", }, +	{ .compatible = "aspeed,ast2600-sdhci", }, +	{ } +}; + +static struct platform_driver aspeed_sdhci_driver = { +	.driver		= { +		.name	= "sdhci-aspeed", +		.of_match_table = aspeed_sdhci_of_match, +	}, +	.probe		= aspeed_sdhci_probe, +	.remove		= aspeed_sdhci_remove, +}; + +static int aspeed_sdc_probe(struct platform_device *pdev) + +{ +	struct device_node *parent, *child; +	struct aspeed_sdc *sdc; +	int ret; + +	sdc = devm_kzalloc(&pdev->dev, sizeof(*sdc), GFP_KERNEL); +	if (!sdc) +		return -ENOMEM; + +	spin_lock_init(&sdc->lock); + +	sdc->clk = devm_clk_get(&pdev->dev, NULL); +	if (IS_ERR(sdc->clk)) +		return PTR_ERR(sdc->clk); + +	ret = clk_prepare_enable(sdc->clk); +	if (ret) { +		dev_err(&pdev->dev, "Unable to enable SDCLK\n"); +		return ret; +	} + +	sdc->res = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	sdc->regs = devm_ioremap_resource(&pdev->dev, sdc->res); +	if (IS_ERR(sdc->regs)) { +		ret = PTR_ERR(sdc->regs); +		goto err_clk; +	} + +	dev_set_drvdata(&pdev->dev, sdc); + +	parent = pdev->dev.of_node; +	for_each_available_child_of_node(parent, child) { +		struct platform_device *cpdev; + +		cpdev = of_platform_device_create(child, NULL, &pdev->dev); +		if (!cpdev) { +			of_node_put(child); +			ret = -ENODEV; +			goto err_clk; +		} +	} + +	return 0; + +err_clk: +	clk_disable_unprepare(sdc->clk); +	return ret; +} + +static int aspeed_sdc_remove(struct platform_device *pdev) +{ +	struct aspeed_sdc *sdc = dev_get_drvdata(&pdev->dev); + +	clk_disable_unprepare(sdc->clk); + +	return 0; +} + +static const struct of_device_id aspeed_sdc_of_match[] = { +	{ .compatible = "aspeed,ast2400-sd-controller", }, +	{ .compatible = "aspeed,ast2500-sd-controller", }, +	{ .compatible = "aspeed,ast2600-sd-controller", }, +	{ } +}; + +MODULE_DEVICE_TABLE(of, aspeed_sdc_of_match); + +static struct platform_driver aspeed_sdc_driver = { +	.driver		= { +		.name	= "sd-controller-aspeed", +		.pm	= &sdhci_pltfm_pmops, +		.of_match_table = aspeed_sdc_of_match, +	}, +	.probe		= aspeed_sdc_probe, +	.remove		= aspeed_sdc_remove, +}; + +static int __init aspeed_sdc_init(void) +{ +	int rc; + +	rc = platform_driver_register(&aspeed_sdhci_driver); +	if (rc < 0) +		return rc; + +	rc = platform_driver_register(&aspeed_sdc_driver); +	if (rc < 0) +		platform_driver_unregister(&aspeed_sdhci_driver); + +	return rc; +} +module_init(aspeed_sdc_init); + +static void __exit aspeed_sdc_exit(void) +{ +	platform_driver_unregister(&aspeed_sdc_driver); +	platform_driver_unregister(&aspeed_sdhci_driver); +} +module_exit(aspeed_sdc_exit); + +MODULE_DESCRIPTION("Driver for the ASPEED SD/SDIO/SDHCI Controllers"); +MODULE_AUTHOR("Ryan Chen <ryan_chen@aspeedtech.com>"); +MODULE_AUTHOR("Andrew Jeffery <andrew@aj.id.au>"); +MODULE_LICENSE("GPL"); diff --git a/drivers/mmc/host/sdhci-of-at91.c b/drivers/mmc/host/sdhci-of-at91.c index d4e7e8b7be77..e7d1920729fb 100644 --- a/drivers/mmc/host/sdhci-of-at91.c +++ b/drivers/mmc/host/sdhci-of-at91.c @@ -357,6 +357,9 @@ static int sdhci_at91_probe(struct platform_device *pdev)  	pm_runtime_set_autosuspend_delay(&pdev->dev, 50);  	pm_runtime_use_autosuspend(&pdev->dev); +	/* HS200 is broken at this moment */ +	host->quirks2 = SDHCI_QUIRK2_BROKEN_HS200; +  	ret = sdhci_add_host(host);  	if (ret)  		goto pm_runtime_disable; diff --git a/drivers/mmc/host/sdhci-of-esdhc.c b/drivers/mmc/host/sdhci-of-esdhc.c index 4dd43b1adf2c..1d1953dfc54b 100644 --- a/drivers/mmc/host/sdhci-of-esdhc.c +++ b/drivers/mmc/host/sdhci-of-esdhc.c @@ -495,7 +495,12 @@ static int esdhc_of_enable_dma(struct sdhci_host *host)  		dma_set_mask_and_coherent(dev, DMA_BIT_MASK(40));  	value = sdhci_readl(host, ESDHC_DMA_SYSCTL); -	value |= ESDHC_DMA_SNOOP; + +	if (of_dma_is_coherent(dev->of_node)) +		value |= ESDHC_DMA_SNOOP; +	else +		value &= ~ESDHC_DMA_SNOOP; +  	sdhci_writel(host, value, ESDHC_DMA_SYSCTL);  	return 0;  } @@ -999,6 +1004,7 @@ static struct soc_device_attribute soc_incorrect_hostver[] = {  static struct soc_device_attribute soc_fixup_sdhc_clkdivs[] = {  	{ .family = "QorIQ LX2160A", .revision = "1.0", },  	{ .family = "QorIQ LX2160A", .revision = "2.0", }, +	{ .family = "QorIQ LS1028A", .revision = "1.0", },  	{ },  }; diff --git a/drivers/mmc/host/sdhci-pci-core.c b/drivers/mmc/host/sdhci-pci-core.c index 7d06e2860c36..eaffa85bc728 100644 --- a/drivers/mmc/host/sdhci-pci-core.c +++ b/drivers/mmc/host/sdhci-pci-core.c @@ -1672,6 +1672,7 @@ static const struct pci_device_id pci_ids[] = {  	SDHCI_PCI_DEVICE(INTEL, EHL_SD,    intel_byt_sd),  	SDHCI_PCI_DEVICE(INTEL, CML_EMMC,  intel_glk_emmc),  	SDHCI_PCI_DEVICE(INTEL, CML_SD,    intel_byt_sd), +	SDHCI_PCI_DEVICE(INTEL, CMLH_SD,   intel_byt_sd),  	SDHCI_PCI_DEVICE(O2, 8120,     o2),  	SDHCI_PCI_DEVICE(O2, 8220,     o2),  	SDHCI_PCI_DEVICE(O2, 8221,     o2), @@ -1684,6 +1685,8 @@ static const struct pci_device_id pci_ids[] = {  	SDHCI_PCI_DEVICE(O2, SEABIRD1, o2),  	SDHCI_PCI_DEVICE(ARASAN, PHY_EMMC, arasan),  	SDHCI_PCI_DEVICE(SYNOPSYS, DWC_MSHC, snps), +	SDHCI_PCI_DEVICE(GLI, 9750, gl9750), +	SDHCI_PCI_DEVICE(GLI, 9755, gl9755),  	SDHCI_PCI_DEVICE_CLASS(AMD, SYSTEM_SDHCI, PCI_CLASS_MASK, amd),  	/* Generic SD host controller */  	{PCI_DEVICE_CLASS(SYSTEM_SDHCI, PCI_CLASS_MASK)}, @@ -1759,8 +1762,7 @@ static const struct sdhci_ops sdhci_pci_ops = {  #ifdef CONFIG_PM_SLEEP  static int sdhci_pci_suspend(struct device *dev)  { -	struct pci_dev *pdev = to_pci_dev(dev); -	struct sdhci_pci_chip *chip = pci_get_drvdata(pdev); +	struct sdhci_pci_chip *chip = dev_get_drvdata(dev);  	if (!chip)  		return 0; @@ -1773,8 +1775,7 @@ static int sdhci_pci_suspend(struct device *dev)  static int sdhci_pci_resume(struct device *dev)  { -	struct pci_dev *pdev = to_pci_dev(dev); -	struct sdhci_pci_chip *chip = pci_get_drvdata(pdev); +	struct sdhci_pci_chip *chip = dev_get_drvdata(dev);  	if (!chip)  		return 0; @@ -1789,8 +1790,7 @@ static int sdhci_pci_resume(struct device *dev)  #ifdef CONFIG_PM  static int sdhci_pci_runtime_suspend(struct device *dev)  { -	struct pci_dev *pdev = to_pci_dev(dev); -	struct sdhci_pci_chip *chip = pci_get_drvdata(pdev); +	struct sdhci_pci_chip *chip = dev_get_drvdata(dev);  	if (!chip)  		return 0; @@ -1803,8 +1803,7 @@ static int sdhci_pci_runtime_suspend(struct device *dev)  static int sdhci_pci_runtime_resume(struct device *dev)  { -	struct pci_dev *pdev = to_pci_dev(dev); -	struct sdhci_pci_chip *chip = pci_get_drvdata(pdev); +	struct sdhci_pci_chip *chip = dev_get_drvdata(dev);  	if (!chip)  		return 0; diff --git a/drivers/mmc/host/sdhci-pci-gli.c b/drivers/mmc/host/sdhci-pci-gli.c new file mode 100644 index 000000000000..5eea8d70a85d --- /dev/null +++ b/drivers/mmc/host/sdhci-pci-gli.c @@ -0,0 +1,352 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (C) 2019 Genesys Logic, Inc. + * + * Authors: Ben Chuang <ben.chuang@genesyslogic.com.tw> + * + * Version: v0.9.0 (2019-08-08) + */ + +#include <linux/bitfield.h> +#include <linux/bits.h> +#include <linux/pci.h> +#include <linux/mmc/mmc.h> +#include <linux/delay.h> +#include "sdhci.h" +#include "sdhci-pci.h" + +/*  Genesys Logic extra registers */ +#define SDHCI_GLI_9750_WT         0x800 +#define   SDHCI_GLI_9750_WT_EN      BIT(0) +#define   GLI_9750_WT_EN_ON	    0x1 +#define   GLI_9750_WT_EN_OFF	    0x0 + +#define SDHCI_GLI_9750_DRIVING      0x860 +#define   SDHCI_GLI_9750_DRIVING_1    GENMASK(11, 0) +#define   SDHCI_GLI_9750_DRIVING_2    GENMASK(27, 26) +#define   GLI_9750_DRIVING_1_VALUE    0xFFF +#define   GLI_9750_DRIVING_2_VALUE    0x3 + +#define SDHCI_GLI_9750_PLL	      0x864 +#define   SDHCI_GLI_9750_PLL_TX2_INV    BIT(23) +#define   SDHCI_GLI_9750_PLL_TX2_DLY    GENMASK(22, 20) +#define   GLI_9750_PLL_TX2_INV_VALUE    0x1 +#define   GLI_9750_PLL_TX2_DLY_VALUE    0x0 + +#define SDHCI_GLI_9750_SW_CTRL      0x874 +#define   SDHCI_GLI_9750_SW_CTRL_4    GENMASK(7, 6) +#define   GLI_9750_SW_CTRL_4_VALUE    0x3 + +#define SDHCI_GLI_9750_MISC            0x878 +#define   SDHCI_GLI_9750_MISC_TX1_INV    BIT(2) +#define   SDHCI_GLI_9750_MISC_RX_INV     BIT(3) +#define   SDHCI_GLI_9750_MISC_TX1_DLY    GENMASK(6, 4) +#define   GLI_9750_MISC_TX1_INV_VALUE    0x0 +#define   GLI_9750_MISC_RX_INV_ON        0x1 +#define   GLI_9750_MISC_RX_INV_OFF       0x0 +#define   GLI_9750_MISC_RX_INV_VALUE     GLI_9750_MISC_RX_INV_OFF +#define   GLI_9750_MISC_TX1_DLY_VALUE    0x5 + +#define SDHCI_GLI_9750_TUNING_CONTROL	          0x540 +#define   SDHCI_GLI_9750_TUNING_CONTROL_EN          BIT(4) +#define   GLI_9750_TUNING_CONTROL_EN_ON             0x1 +#define   GLI_9750_TUNING_CONTROL_EN_OFF            0x0 +#define   SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_1    BIT(16) +#define   SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_2    GENMASK(20, 19) +#define   GLI_9750_TUNING_CONTROL_GLITCH_1_VALUE    0x1 +#define   GLI_9750_TUNING_CONTROL_GLITCH_2_VALUE    0x2 + +#define SDHCI_GLI_9750_TUNING_PARAMETERS           0x544 +#define   SDHCI_GLI_9750_TUNING_PARAMETERS_RX_DLY    GENMASK(2, 0) +#define   GLI_9750_TUNING_PARAMETERS_RX_DLY_VALUE    0x1 + +#define GLI_MAX_TUNING_LOOP 40 + +/* Genesys Logic chipset */ +static inline void gl9750_wt_on(struct sdhci_host *host) +{ +	u32 wt_value; +	u32 wt_enable; + +	wt_value = sdhci_readl(host, SDHCI_GLI_9750_WT); +	wt_enable = FIELD_GET(SDHCI_GLI_9750_WT_EN, wt_value); + +	if (wt_enable == GLI_9750_WT_EN_ON) +		return; + +	wt_value &= ~SDHCI_GLI_9750_WT_EN; +	wt_value |= FIELD_PREP(SDHCI_GLI_9750_WT_EN, GLI_9750_WT_EN_ON); + +	sdhci_writel(host, wt_value, SDHCI_GLI_9750_WT); +} + +static inline void gl9750_wt_off(struct sdhci_host *host) +{ +	u32 wt_value; +	u32 wt_enable; + +	wt_value = sdhci_readl(host, SDHCI_GLI_9750_WT); +	wt_enable = FIELD_GET(SDHCI_GLI_9750_WT_EN, wt_value); + +	if (wt_enable == GLI_9750_WT_EN_OFF) +		return; + +	wt_value &= ~SDHCI_GLI_9750_WT_EN; +	wt_value |= FIELD_PREP(SDHCI_GLI_9750_WT_EN, GLI_9750_WT_EN_OFF); + +	sdhci_writel(host, wt_value, SDHCI_GLI_9750_WT); +} + +static void gli_set_9750(struct sdhci_host *host) +{ +	u32 driving_value; +	u32 pll_value; +	u32 sw_ctrl_value; +	u32 misc_value; +	u32 parameter_value; +	u32 control_value; +	u16 ctrl2; + +	gl9750_wt_on(host); + +	driving_value = sdhci_readl(host, SDHCI_GLI_9750_DRIVING); +	pll_value = sdhci_readl(host, SDHCI_GLI_9750_PLL); +	sw_ctrl_value = sdhci_readl(host, SDHCI_GLI_9750_SW_CTRL); +	misc_value = sdhci_readl(host, SDHCI_GLI_9750_MISC); +	parameter_value = sdhci_readl(host, SDHCI_GLI_9750_TUNING_PARAMETERS); +	control_value = sdhci_readl(host, SDHCI_GLI_9750_TUNING_CONTROL); + +	driving_value &= ~(SDHCI_GLI_9750_DRIVING_1); +	driving_value &= ~(SDHCI_GLI_9750_DRIVING_2); +	driving_value |= FIELD_PREP(SDHCI_GLI_9750_DRIVING_1, +				    GLI_9750_DRIVING_1_VALUE); +	driving_value |= FIELD_PREP(SDHCI_GLI_9750_DRIVING_2, +				    GLI_9750_DRIVING_2_VALUE); +	sdhci_writel(host, driving_value, SDHCI_GLI_9750_DRIVING); + +	sw_ctrl_value &= ~SDHCI_GLI_9750_SW_CTRL_4; +	sw_ctrl_value |= FIELD_PREP(SDHCI_GLI_9750_SW_CTRL_4, +				    GLI_9750_SW_CTRL_4_VALUE); +	sdhci_writel(host, sw_ctrl_value, SDHCI_GLI_9750_SW_CTRL); + +	/* reset the tuning flow after reinit and before starting tuning */ +	pll_value &= ~SDHCI_GLI_9750_PLL_TX2_INV; +	pll_value &= ~SDHCI_GLI_9750_PLL_TX2_DLY; +	pll_value |= FIELD_PREP(SDHCI_GLI_9750_PLL_TX2_INV, +				GLI_9750_PLL_TX2_INV_VALUE); +	pll_value |= FIELD_PREP(SDHCI_GLI_9750_PLL_TX2_DLY, +				GLI_9750_PLL_TX2_DLY_VALUE); + +	misc_value &= ~SDHCI_GLI_9750_MISC_TX1_INV; +	misc_value &= ~SDHCI_GLI_9750_MISC_RX_INV; +	misc_value &= ~SDHCI_GLI_9750_MISC_TX1_DLY; +	misc_value |= FIELD_PREP(SDHCI_GLI_9750_MISC_TX1_INV, +				 GLI_9750_MISC_TX1_INV_VALUE); +	misc_value |= FIELD_PREP(SDHCI_GLI_9750_MISC_RX_INV, +				 GLI_9750_MISC_RX_INV_VALUE); +	misc_value |= FIELD_PREP(SDHCI_GLI_9750_MISC_TX1_DLY, +				 GLI_9750_MISC_TX1_DLY_VALUE); + +	parameter_value &= ~SDHCI_GLI_9750_TUNING_PARAMETERS_RX_DLY; +	parameter_value |= FIELD_PREP(SDHCI_GLI_9750_TUNING_PARAMETERS_RX_DLY, +				      GLI_9750_TUNING_PARAMETERS_RX_DLY_VALUE); + +	control_value &= ~SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_1; +	control_value &= ~SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_2; +	control_value |= FIELD_PREP(SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_1, +				    GLI_9750_TUNING_CONTROL_GLITCH_1_VALUE); +	control_value |= FIELD_PREP(SDHCI_GLI_9750_TUNING_CONTROL_GLITCH_2, +				    GLI_9750_TUNING_CONTROL_GLITCH_2_VALUE); + +	sdhci_writel(host, pll_value, SDHCI_GLI_9750_PLL); +	sdhci_writel(host, misc_value, SDHCI_GLI_9750_MISC); + +	/* disable tuned clk */ +	ctrl2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); +	ctrl2 &= ~SDHCI_CTRL_TUNED_CLK; +	sdhci_writew(host, ctrl2, SDHCI_HOST_CONTROL2); + +	/* enable tuning parameters control */ +	control_value &= ~SDHCI_GLI_9750_TUNING_CONTROL_EN; +	control_value |= FIELD_PREP(SDHCI_GLI_9750_TUNING_CONTROL_EN, +				    GLI_9750_TUNING_CONTROL_EN_ON); +	sdhci_writel(host, control_value, SDHCI_GLI_9750_TUNING_CONTROL); + +	/* write tuning parameters */ +	sdhci_writel(host, parameter_value, SDHCI_GLI_9750_TUNING_PARAMETERS); + +	/* disable tuning parameters control */ +	control_value &= ~SDHCI_GLI_9750_TUNING_CONTROL_EN; +	control_value |= FIELD_PREP(SDHCI_GLI_9750_TUNING_CONTROL_EN, +				    GLI_9750_TUNING_CONTROL_EN_OFF); +	sdhci_writel(host, control_value, SDHCI_GLI_9750_TUNING_CONTROL); + +	/* clear tuned clk */ +	ctrl2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); +	ctrl2 &= ~SDHCI_CTRL_TUNED_CLK; +	sdhci_writew(host, ctrl2, SDHCI_HOST_CONTROL2); + +	gl9750_wt_off(host); +} + +static void gli_set_9750_rx_inv(struct sdhci_host *host, bool b) +{ +	u32 misc_value; + +	gl9750_wt_on(host); + +	misc_value = sdhci_readl(host, SDHCI_GLI_9750_MISC); +	misc_value &= ~SDHCI_GLI_9750_MISC_RX_INV; +	if (b) { +		misc_value |= FIELD_PREP(SDHCI_GLI_9750_MISC_RX_INV, +					 GLI_9750_MISC_RX_INV_ON); +	} else { +		misc_value |= FIELD_PREP(SDHCI_GLI_9750_MISC_RX_INV, +					 GLI_9750_MISC_RX_INV_OFF); +	} +	sdhci_writel(host, misc_value, SDHCI_GLI_9750_MISC); + +	gl9750_wt_off(host); +} + +static int __sdhci_execute_tuning_9750(struct sdhci_host *host, u32 opcode) +{ +	int i; +	int rx_inv; + +	for (rx_inv = 0; rx_inv < 2; rx_inv++) { +		gli_set_9750_rx_inv(host, !!rx_inv); +		sdhci_start_tuning(host); + +		for (i = 0; i < GLI_MAX_TUNING_LOOP; i++) { +			u16 ctrl; + +			sdhci_send_tuning(host, opcode); + +			if (!host->tuning_done) { +				sdhci_abort_tuning(host, opcode); +				break; +			} + +			ctrl = sdhci_readw(host, SDHCI_HOST_CONTROL2); +			if (!(ctrl & SDHCI_CTRL_EXEC_TUNING)) { +				if (ctrl & SDHCI_CTRL_TUNED_CLK) +					return 0; /* Success! */ +				break; +			} +		} +	} +	if (!host->tuning_done) { +		pr_info("%s: Tuning timeout, falling back to fixed sampling clock\n", +			mmc_hostname(host->mmc)); +		return -ETIMEDOUT; +	} + +	pr_info("%s: Tuning failed, falling back to fixed sampling clock\n", +		mmc_hostname(host->mmc)); +	sdhci_reset_tuning(host); + +	return -EAGAIN; +} + +static int gl9750_execute_tuning(struct sdhci_host *host, u32 opcode) +{ +	host->mmc->retune_period = 0; +	if (host->tuning_mode == SDHCI_TUNING_MODE_1) +		host->mmc->retune_period = host->tuning_count; + +	gli_set_9750(host); +	host->tuning_err = __sdhci_execute_tuning_9750(host, opcode); +	sdhci_end_tuning(host); + +	return 0; +} + +static int gli_probe_slot_gl9750(struct sdhci_pci_slot *slot) +{ +	struct sdhci_host *host = slot->host; + +	slot->host->mmc->caps2 |= MMC_CAP2_NO_SDIO; +	sdhci_enable_v4_mode(host); + +	return 0; +} + +static int gli_probe_slot_gl9755(struct sdhci_pci_slot *slot) +{ +	struct sdhci_host *host = slot->host; + +	slot->host->mmc->caps2 |= MMC_CAP2_NO_SDIO; +	sdhci_enable_v4_mode(host); + +	return 0; +} + +static void sdhci_gli_voltage_switch(struct sdhci_host *host) +{ +	/* +	 * According to Section 3.6.1 signal voltage switch procedure in +	 * SD Host Controller Simplified Spec. 4.20, steps 6~8 are as +	 * follows: +	 * (6) Set 1.8V Signal Enable in the Host Control 2 register. +	 * (7) Wait 5ms. 1.8V voltage regulator shall be stable within this +	 *     period. +	 * (8) If 1.8V Signal Enable is cleared by Host Controller, go to +	 *     step (12). +	 * +	 * Wait 5ms after set 1.8V signal enable in Host Control 2 register +	 * to ensure 1.8V signal enable bit is set by GL9750/GL9755. +	 */ +	usleep_range(5000, 5500); +} + +static void sdhci_gl9750_reset(struct sdhci_host *host, u8 mask) +{ +	sdhci_reset(host, mask); +	gli_set_9750(host); +} + +static u32 sdhci_gl9750_readl(struct sdhci_host *host, int reg) +{ +	u32 value; + +	value = readl(host->ioaddr + reg); +	if (unlikely(reg == SDHCI_MAX_CURRENT && !(value & 0xff))) +		value |= 0xc8; + +	return value; +} + +static const struct sdhci_ops sdhci_gl9755_ops = { +	.set_clock		= sdhci_set_clock, +	.enable_dma		= sdhci_pci_enable_dma, +	.set_bus_width		= sdhci_set_bus_width, +	.reset			= sdhci_reset, +	.set_uhs_signaling	= sdhci_set_uhs_signaling, +	.voltage_switch		= sdhci_gli_voltage_switch, +}; + +const struct sdhci_pci_fixes sdhci_gl9755 = { +	.quirks		= SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, +	.quirks2	= SDHCI_QUIRK2_BROKEN_DDR50, +	.probe_slot	= gli_probe_slot_gl9755, +	.ops            = &sdhci_gl9755_ops, +}; + +static const struct sdhci_ops sdhci_gl9750_ops = { +	.read_l                 = sdhci_gl9750_readl, +	.set_clock		= sdhci_set_clock, +	.enable_dma		= sdhci_pci_enable_dma, +	.set_bus_width		= sdhci_set_bus_width, +	.reset			= sdhci_gl9750_reset, +	.set_uhs_signaling	= sdhci_set_uhs_signaling, +	.voltage_switch		= sdhci_gli_voltage_switch, +	.platform_execute_tuning = gl9750_execute_tuning, +}; + +const struct sdhci_pci_fixes sdhci_gl9750 = { +	.quirks		= SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, +	.quirks2	= SDHCI_QUIRK2_BROKEN_DDR50, +	.probe_slot	= gli_probe_slot_gl9750, +	.ops            = &sdhci_gl9750_ops, +}; diff --git a/drivers/mmc/host/sdhci-pci-o2micro.c b/drivers/mmc/host/sdhci-pci-o2micro.c index 9dc4548271b4..fa8105087d68 100644 --- a/drivers/mmc/host/sdhci-pci-o2micro.c +++ b/drivers/mmc/host/sdhci-pci-o2micro.c @@ -11,6 +11,7 @@  #include <linux/mmc/host.h>  #include <linux/mmc/mmc.h>  #include <linux/delay.h> +#include <linux/iopoll.h>  #include "sdhci.h"  #include "sdhci-pci.h" @@ -51,13 +52,136 @@  #define O2_SD_VENDOR_SETTING2	0x1C8  #define O2_SD_HW_TUNING_DISABLE	BIT(4) -#define O2_PLL_WDT_CONTROL1	0x1CC +#define O2_PLL_DLL_WDT_CONTROL1	0x1CC  #define  O2_PLL_FORCE_ACTIVE	BIT(18)  #define  O2_PLL_LOCK_STATUS	BIT(14)  #define  O2_PLL_SOFT_RESET	BIT(12) +#define  O2_DLL_LOCK_STATUS	BIT(11)  #define O2_SD_DETECT_SETTING 0x324 +static const u32 dmdn_table[] = {0x2B1C0000, +	0x2C1A0000, 0x371B0000, 0x35100000}; +#define DMDN_SZ ARRAY_SIZE(dmdn_table) + +struct o2_host { +	u8 dll_adjust_count; +}; + +static void sdhci_o2_wait_card_detect_stable(struct sdhci_host *host) +{ +	ktime_t timeout; +	u32 scratch32; + +	/* Wait max 50 ms */ +	timeout = ktime_add_ms(ktime_get(), 50); +	while (1) { +		bool timedout = ktime_after(ktime_get(), timeout); + +		scratch32 = sdhci_readl(host, SDHCI_PRESENT_STATE); +		if ((scratch32 & SDHCI_CARD_PRESENT) >> SDHCI_CARD_PRES_SHIFT +		    == (scratch32 & SDHCI_CD_LVL) >> SDHCI_CD_LVL_SHIFT) +			break; + +		if (timedout) { +			pr_err("%s: Card Detect debounce never finished.\n", +			       mmc_hostname(host->mmc)); +			sdhci_dumpregs(host); +			return; +		} +		udelay(10); +	} +} + +static void sdhci_o2_enable_internal_clock(struct sdhci_host *host) +{ +	ktime_t timeout; +	u16 scratch; +	u32 scratch32; + +	/* PLL software reset */ +	scratch32 = sdhci_readl(host, O2_PLL_DLL_WDT_CONTROL1); +	scratch32 |= O2_PLL_SOFT_RESET; +	sdhci_writel(host, scratch32, O2_PLL_DLL_WDT_CONTROL1); +	udelay(1); +	scratch32 &= ~(O2_PLL_SOFT_RESET); +	sdhci_writel(host, scratch32, O2_PLL_DLL_WDT_CONTROL1); + +	/* PLL force active */ +	scratch32 |= O2_PLL_FORCE_ACTIVE; +	sdhci_writel(host, scratch32, O2_PLL_DLL_WDT_CONTROL1); + +	/* Wait max 20 ms */ +	timeout = ktime_add_ms(ktime_get(), 20); +	while (1) { +		bool timedout = ktime_after(ktime_get(), timeout); + +		scratch = sdhci_readw(host, O2_PLL_DLL_WDT_CONTROL1); +		if (scratch & O2_PLL_LOCK_STATUS) +			break; +		if (timedout) { +			pr_err("%s: Internal clock never stabilised.\n", +			       mmc_hostname(host->mmc)); +			sdhci_dumpregs(host); +			goto out; +		} +		udelay(10); +	} + +	/* Wait for card detect finish */ +	udelay(1); +	sdhci_o2_wait_card_detect_stable(host); + +out: +	/* Cancel PLL force active */ +	scratch32 = sdhci_readl(host, O2_PLL_DLL_WDT_CONTROL1); +	scratch32 &= ~O2_PLL_FORCE_ACTIVE; +	sdhci_writel(host, scratch32, O2_PLL_DLL_WDT_CONTROL1); +} + +static int sdhci_o2_get_cd(struct mmc_host *mmc) +{ +	struct sdhci_host *host = mmc_priv(mmc); + +	if (!(sdhci_readw(host, O2_PLL_DLL_WDT_CONTROL1) & O2_PLL_LOCK_STATUS)) +		sdhci_o2_enable_internal_clock(host); + +	return !!(sdhci_readl(host, SDHCI_PRESENT_STATE) & SDHCI_CARD_PRESENT); +} + +static void o2_pci_set_baseclk(struct sdhci_pci_chip *chip, u32 value) +{ +	u32 scratch_32; + +	pci_read_config_dword(chip->pdev, +			      O2_SD_PLL_SETTING, &scratch_32); + +	scratch_32 &= 0x0000FFFF; +	scratch_32 |= value; + +	pci_write_config_dword(chip->pdev, +			       O2_SD_PLL_SETTING, scratch_32); +} + +static u32 sdhci_o2_pll_dll_wdt_control(struct sdhci_host *host) +{ +	return sdhci_readl(host, O2_PLL_DLL_WDT_CONTROL1); +} + +/* + * This function is used to detect dll lock status. + * Since the dll lock status bit will toggle randomly + * with very short interval which needs to be polled + * as fast as possible. Set sleep_us as 1 microsecond. + */ +static int sdhci_o2_wait_dll_detect_lock(struct sdhci_host *host) +{ +	u32	scratch32 = 0; + +	return readx_poll_timeout(sdhci_o2_pll_dll_wdt_control, host, +		scratch32, !(scratch32 & O2_DLL_LOCK_STATUS), 1, 1000000); +} +  static void sdhci_o2_set_tuning_mode(struct sdhci_host *host)  {  	u16 reg; @@ -95,6 +219,83 @@ static void __sdhci_o2_execute_tuning(struct sdhci_host *host, u32 opcode)  	sdhci_reset_tuning(host);  } +/* + * This function is used to fix o2 dll shift issue. + * It isn't necessary to detect card present before recovery. + * Firstly, it is used by bht emmc card, which is embedded. + * Second, before call recovery card present will be detected + * outside of the execute tuning function. + */ +static int sdhci_o2_dll_recovery(struct sdhci_host *host) +{ +	int ret = 0; +	u8 scratch_8 = 0; +	u32 scratch_32 = 0; +	struct sdhci_pci_slot *slot = sdhci_priv(host); +	struct sdhci_pci_chip *chip = slot->chip; +	struct o2_host *o2_host = sdhci_pci_priv(slot); + +	/* UnLock WP */ +	pci_read_config_byte(chip->pdev, +			O2_SD_LOCK_WP, &scratch_8); +	scratch_8 &= 0x7f; +	pci_write_config_byte(chip->pdev, O2_SD_LOCK_WP, scratch_8); +	while (o2_host->dll_adjust_count < DMDN_SZ && !ret) { +		/* Disable clock */ +		sdhci_writeb(host, 0, SDHCI_CLOCK_CONTROL); + +		/* PLL software reset */ +		scratch_32 = sdhci_readl(host, O2_PLL_DLL_WDT_CONTROL1); +		scratch_32 |= O2_PLL_SOFT_RESET; +		sdhci_writel(host, scratch_32, O2_PLL_DLL_WDT_CONTROL1); + +		pci_read_config_dword(chip->pdev, +					    O2_SD_FUNC_REG4, +					    &scratch_32); +		/* Enable Base Clk setting change */ +		scratch_32 |= O2_SD_FREG4_ENABLE_CLK_SET; +		pci_write_config_dword(chip->pdev, O2_SD_FUNC_REG4, scratch_32); +		o2_pci_set_baseclk(chip, dmdn_table[o2_host->dll_adjust_count]); + +		/* Enable internal clock */ +		scratch_8 = SDHCI_CLOCK_INT_EN; +		sdhci_writeb(host, scratch_8, SDHCI_CLOCK_CONTROL); + +		if (sdhci_o2_get_cd(host->mmc)) { +			/* +			 * need wait at least 5ms for dll status stable, +			 * after enable internal clock +			 */ +			usleep_range(5000, 6000); +			if (sdhci_o2_wait_dll_detect_lock(host)) { +				scratch_8 |= SDHCI_CLOCK_CARD_EN; +				sdhci_writeb(host, scratch_8, +					SDHCI_CLOCK_CONTROL); +				ret = 1; +			} else { +				pr_warn("%s: DLL unlocked when dll_adjust_count is %d.\n", +					mmc_hostname(host->mmc), +					o2_host->dll_adjust_count); +			} +		} else { +			pr_err("%s: card present detect failed.\n", +				mmc_hostname(host->mmc)); +			break; +		} + +		o2_host->dll_adjust_count++; +	} +	if (!ret && o2_host->dll_adjust_count == DMDN_SZ) +		pr_err("%s: DLL adjust over max times\n", +		mmc_hostname(host->mmc)); +	/* Lock WP */ +	pci_read_config_byte(chip->pdev, +				   O2_SD_LOCK_WP, &scratch_8); +	scratch_8 |= 0x80; +	pci_write_config_byte(chip->pdev, O2_SD_LOCK_WP, scratch_8); +	return ret; +} +  static int sdhci_o2_execute_tuning(struct mmc_host *mmc, u32 opcode)  {  	struct sdhci_host *host = mmc_priv(mmc); @@ -109,7 +310,16 @@ static int sdhci_o2_execute_tuning(struct mmc_host *mmc, u32 opcode)  	if (WARN_ON(opcode != MMC_SEND_TUNING_BLOCK_HS200))  		return -EINVAL; - +	/* +	 * Judge the tuning reason, whether caused by dll shift +	 * If cause by dll shift, should call sdhci_o2_dll_recovery +	 */ +	if (!sdhci_o2_wait_dll_detect_lock(host)) +		if (!sdhci_o2_dll_recovery(host)) { +			pr_err("%s: o2 dll recovery failed\n", +				mmc_hostname(host->mmc)); +			return -EINVAL; +		}  	/*  	 * o2 sdhci host didn't support 8bit emmc tuning  	 */ @@ -136,19 +346,6 @@ static int sdhci_o2_execute_tuning(struct mmc_host *mmc, u32 opcode)  	return 0;  } -static void o2_pci_set_baseclk(struct sdhci_pci_chip *chip, u32 value) -{ -	u32 scratch_32; -	pci_read_config_dword(chip->pdev, -			      O2_SD_PLL_SETTING, &scratch_32); - -	scratch_32 &= 0x0000FFFF; -	scratch_32 |= value; - -	pci_write_config_dword(chip->pdev, -			       O2_SD_PLL_SETTING, scratch_32); -} -  static void o2_pci_led_enable(struct sdhci_pci_chip *chip)  {  	int ret; @@ -284,92 +481,13 @@ static void sdhci_pci_o2_enable_msi(struct sdhci_pci_chip *chip,  	host->irq = pci_irq_vector(chip->pdev, 0);  } -static void sdhci_o2_wait_card_detect_stable(struct sdhci_host *host) -{ -	ktime_t timeout; -	u32 scratch32; - -	/* Wait max 50 ms */ -	timeout = ktime_add_ms(ktime_get(), 50); -	while (1) { -		bool timedout = ktime_after(ktime_get(), timeout); - -		scratch32 = sdhci_readl(host, SDHCI_PRESENT_STATE); -		if ((scratch32 & SDHCI_CARD_PRESENT) >> SDHCI_CARD_PRES_SHIFT -		    == (scratch32 & SDHCI_CD_LVL) >> SDHCI_CD_LVL_SHIFT) -			break; - -		if (timedout) { -			pr_err("%s: Card Detect debounce never finished.\n", -			       mmc_hostname(host->mmc)); -			sdhci_dumpregs(host); -			return; -		} -		udelay(10); -	} -} - -static void sdhci_o2_enable_internal_clock(struct sdhci_host *host) -{ -	ktime_t timeout; -	u16 scratch; -	u32 scratch32; - -	/* PLL software reset */ -	scratch32 = sdhci_readl(host, O2_PLL_WDT_CONTROL1); -	scratch32 |= O2_PLL_SOFT_RESET; -	sdhci_writel(host, scratch32, O2_PLL_WDT_CONTROL1); -	udelay(1); -	scratch32 &= ~(O2_PLL_SOFT_RESET); -	sdhci_writel(host, scratch32, O2_PLL_WDT_CONTROL1); - -	/* PLL force active */ -	scratch32 |= O2_PLL_FORCE_ACTIVE; -	sdhci_writel(host, scratch32, O2_PLL_WDT_CONTROL1); - -	/* Wait max 20 ms */ -	timeout = ktime_add_ms(ktime_get(), 20); -	while (1) { -		bool timedout = ktime_after(ktime_get(), timeout); - -		scratch = sdhci_readw(host, O2_PLL_WDT_CONTROL1); -		if (scratch & O2_PLL_LOCK_STATUS) -			break; -		if (timedout) { -			pr_err("%s: Internal clock never stabilised.\n", -			       mmc_hostname(host->mmc)); -			sdhci_dumpregs(host); -			goto out; -		} -		udelay(10); -	} - -	/* Wait for card detect finish */ -	udelay(1); -	sdhci_o2_wait_card_detect_stable(host); - -out: -	/* Cancel PLL force active */ -	scratch32 = sdhci_readl(host, O2_PLL_WDT_CONTROL1); -	scratch32 &= ~O2_PLL_FORCE_ACTIVE; -	sdhci_writel(host, scratch32, O2_PLL_WDT_CONTROL1); -} - -static int sdhci_o2_get_cd(struct mmc_host *mmc) -{ -	struct sdhci_host *host = mmc_priv(mmc); - -	sdhci_o2_enable_internal_clock(host); - -	return !!(sdhci_readl(host, SDHCI_PRESENT_STATE) & SDHCI_CARD_PRESENT); -} -  static void sdhci_o2_enable_clk(struct sdhci_host *host, u16 clk)  {  	/* Enable internal clock */  	clk |= SDHCI_CLOCK_INT_EN;  	sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL); +	sdhci_o2_enable_internal_clock(host);  	if (sdhci_o2_get_cd(host->mmc)) {  		clk |= SDHCI_CLOCK_CARD_EN;  		sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL); @@ -395,12 +513,14 @@ int sdhci_pci_o2_probe_slot(struct sdhci_pci_slot *slot)  {  	struct sdhci_pci_chip *chip;  	struct sdhci_host *host; +	struct o2_host *o2_host = sdhci_pci_priv(slot);  	u32 reg, caps;  	int ret;  	chip = slot->chip;  	host = slot->host; +	o2_host->dll_adjust_count = 0;  	caps = sdhci_readl(host, SDHCI_CAPABILITIES);  	/* @@ -432,7 +552,6 @@ int sdhci_pci_o2_probe_slot(struct sdhci_pci_slot *slot)  					mmc_hostname(host->mmc));  				host->flags &= ~SDHCI_SIGNALING_330;  				host->flags |= SDHCI_SIGNALING_180; -				host->quirks2 |= SDHCI_QUIRK2_CLEAR_TRANSFERMODE_REG_BEFORE_CMD;  				host->mmc->caps2 |= MMC_CAP2_NO_SD;  				host->mmc->caps2 |= MMC_CAP2_NO_SDIO;  				pci_write_config_dword(chip->pdev, @@ -682,9 +801,11 @@ static const struct sdhci_ops sdhci_pci_o2_ops = {  const struct sdhci_pci_fixes sdhci_o2 = {  	.probe = sdhci_pci_o2_probe,  	.quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, +	.quirks2 = SDHCI_QUIRK2_CLEAR_TRANSFERMODE_REG_BEFORE_CMD,  	.probe_slot = sdhci_pci_o2_probe_slot,  #ifdef CONFIG_PM_SLEEP  	.resume = sdhci_pci_o2_resume,  #endif  	.ops = &sdhci_pci_o2_ops, +	.priv_size = sizeof(struct o2_host),  }; diff --git a/drivers/mmc/host/sdhci-pci.h b/drivers/mmc/host/sdhci-pci.h index cdd15f357d01..558202fe64c6 100644 --- a/drivers/mmc/host/sdhci-pci.h +++ b/drivers/mmc/host/sdhci-pci.h @@ -54,6 +54,7 @@  #define PCI_DEVICE_ID_INTEL_EHL_SD	0x4b48  #define PCI_DEVICE_ID_INTEL_CML_EMMC	0x02c4  #define PCI_DEVICE_ID_INTEL_CML_SD	0x02f5 +#define PCI_DEVICE_ID_INTEL_CMLH_SD	0x06f5  #define PCI_DEVICE_ID_SYSKONNECT_8000	0x8000  #define PCI_DEVICE_ID_VIA_95D0		0x95d0 @@ -67,6 +68,9 @@  #define PCI_DEVICE_ID_SYNOPSYS_DWC_MSHC 0xc202 +#define PCI_DEVICE_ID_GLI_9755		0x9755 +#define PCI_DEVICE_ID_GLI_9750		0x9750 +  /*   * PCI device class and mask   */ @@ -187,5 +191,7 @@ int sdhci_pci_enable_dma(struct sdhci_host *host);  extern const struct sdhci_pci_fixes sdhci_arasan;  extern const struct sdhci_pci_fixes sdhci_snps;  extern const struct sdhci_pci_fixes sdhci_o2; +extern const struct sdhci_pci_fixes sdhci_gl9750; +extern const struct sdhci_pci_fixes sdhci_gl9755;  #endif /* __SDHCI_PCI_H */ diff --git a/drivers/mmc/host/sdhci-pltfm.c b/drivers/mmc/host/sdhci-pltfm.c index d268b3b8850a..328b132bbe57 100644 --- a/drivers/mmc/host/sdhci-pltfm.c +++ b/drivers/mmc/host/sdhci-pltfm.c @@ -118,12 +118,10 @@ struct sdhci_host *sdhci_pltfm_init(struct platform_device *pdev,  				    size_t priv_size)  {  	struct sdhci_host *host; -	struct resource *iomem;  	void __iomem *ioaddr;  	int irq, ret; -	iomem = platform_get_resource(pdev, IORESOURCE_MEM, 0); -	ioaddr = devm_ioremap_resource(&pdev->dev, iomem); +	ioaddr = devm_platform_ioremap_resource(pdev, 0);  	if (IS_ERR(ioaddr)) {  		ret = PTR_ERR(ioaddr);  		goto err; @@ -131,7 +129,6 @@ struct sdhci_host *sdhci_pltfm_init(struct platform_device *pdev,  	irq = platform_get_irq(pdev, 0);  	if (irq < 0) { -		dev_err(&pdev->dev, "failed to get IRQ number\n");  		ret = irq;  		goto err;  	} diff --git a/drivers/mmc/host/sdhci-s3c.c b/drivers/mmc/host/sdhci-s3c.c index f5753aef7151..51e096f27388 100644 --- a/drivers/mmc/host/sdhci-s3c.c +++ b/drivers/mmc/host/sdhci-s3c.c @@ -490,10 +490,8 @@ static int sdhci_s3c_probe(struct platform_device *pdev)  	}  	irq = platform_get_irq(pdev, 0); -	if (irq < 0) { -		dev_err(dev, "no irq specified\n"); +	if (irq < 0)  		return irq; -	}  	host = sdhci_alloc_host(dev, sizeof(struct sdhci_s3c));  	if (IS_ERR(host)) { @@ -611,6 +609,7 @@ static int sdhci_s3c_probe(struct platform_device *pdev)  	switch (pdata->max_width) {  	case 8:  		host->mmc->caps |= MMC_CAP_8_BIT_DATA; +		/* Fall through */  	case 4:  		host->mmc->caps |= MMC_CAP_4_BIT_DATA;  		break; diff --git a/drivers/mmc/host/sdhci-sprd.c b/drivers/mmc/host/sdhci-sprd.c index 83a4767ca680..d07b9793380f 100644 --- a/drivers/mmc/host/sdhci-sprd.c +++ b/drivers/mmc/host/sdhci-sprd.c @@ -217,10 +217,11 @@ static inline void _sdhci_sprd_set_clock(struct sdhci_host *host,  	struct sdhci_sprd_host *sprd_host = TO_SPRD_HOST(host);  	u32 div, val, mask; -	div = sdhci_sprd_calc_div(sprd_host->base_rate, clk); +	sdhci_writew(host, 0, SDHCI_CLOCK_CONTROL); -	clk |= ((div & 0x300) >> 2) | ((div & 0xFF) << 8); -	sdhci_enable_clk(host, clk); +	div = sdhci_sprd_calc_div(sprd_host->base_rate, clk); +	div = ((div & 0x300) >> 2) | ((div & 0xFF) << 8); +	sdhci_enable_clk(host, div);  	/* enable auto gate sdhc_enable_auto_gate */  	val = sdhci_readl(host, SDHCI_SPRD_REG_32_BUSY_POSI); @@ -373,6 +374,11 @@ static unsigned int sdhci_sprd_get_max_timeout_count(struct sdhci_host *host)  	return 1 << 31;  } +static unsigned int sdhci_sprd_get_ro(struct sdhci_host *host) +{ +	return 0; +} +  static struct sdhci_ops sdhci_sprd_ops = {  	.read_l = sdhci_sprd_readl,  	.write_l = sdhci_sprd_writel, @@ -385,6 +391,7 @@ static struct sdhci_ops sdhci_sprd_ops = {  	.set_uhs_signaling = sdhci_sprd_set_uhs_signaling,  	.hw_reset = sdhci_sprd_hw_reset,  	.get_max_timeout_count = sdhci_sprd_get_max_timeout_count, +	.get_ro = sdhci_sprd_get_ro,  };  static void sdhci_sprd_request(struct mmc_host *mmc, struct mmc_request *mrq) @@ -501,9 +508,12 @@ static void sdhci_sprd_phy_param_parse(struct sdhci_sprd_host *sprd_host,  }  static const struct sdhci_pltfm_data sdhci_sprd_pdata = { -	.quirks = SDHCI_QUIRK_DATA_TIMEOUT_USES_SDCLK, +	.quirks = SDHCI_QUIRK_BROKEN_CARD_DETECTION | +		  SDHCI_QUIRK_DATA_TIMEOUT_USES_SDCLK | +		  SDHCI_QUIRK_MISSING_CAPS,  	.quirks2 = SDHCI_QUIRK2_BROKEN_HS200 | -		   SDHCI_QUIRK2_USE_32BIT_BLK_CNT, +		   SDHCI_QUIRK2_USE_32BIT_BLK_CNT | +		   SDHCI_QUIRK2_PRESET_VALUE_BROKEN,  	.ops = &sdhci_sprd_ops,  }; @@ -605,6 +615,16 @@ static int sdhci_sprd_probe(struct platform_device *pdev)  	sdhci_enable_v4_mode(host); +	/* +	 * Supply the existing CAPS, but clear the UHS-I modes. This +	 * will allow these modes to be specified only by device +	 * tree properties through mmc_of_parse(). +	 */ +	host->caps = sdhci_readl(host, SDHCI_CAPABILITIES); +	host->caps1 = sdhci_readl(host, SDHCI_CAPABILITIES_1); +	host->caps1 &= ~(SDHCI_SUPPORT_SDR50 | SDHCI_SUPPORT_SDR104 | +			 SDHCI_SUPPORT_DDR50); +  	ret = sdhci_setup_host(host);  	if (ret)  		goto pm_runtime_disable; diff --git a/drivers/mmc/host/sdhci-tegra.c b/drivers/mmc/host/sdhci-tegra.c index f4d4761cf20a..7bc950520fd9 100644 --- a/drivers/mmc/host/sdhci-tegra.c +++ b/drivers/mmc/host/sdhci-tegra.c @@ -4,6 +4,7 @@   */  #include <linux/delay.h> +#include <linux/dma-mapping.h>  #include <linux/err.h>  #include <linux/module.h>  #include <linux/init.h> @@ -104,6 +105,7 @@  struct sdhci_tegra_soc_data {  	const struct sdhci_pltfm_data *pdata; +	u64 dma_mask;  	u32 nvquirks;  	u8 min_tap_delay;  	u8 max_tap_delay; @@ -258,6 +260,16 @@ static void tegra210_sdhci_writew(struct sdhci_host *host, u16 val, int reg)  	}  } +static unsigned int tegra_sdhci_get_ro(struct sdhci_host *host) +{ +	/* +	 * Write-enable shall be assumed if GPIO is missing in a board's +	 * device-tree because SDHCI's WRITE_PROTECT bit doesn't work on +	 * Tegra. +	 */ +	return mmc_gpio_get_ro(host->mmc); +} +  static bool tegra_sdhci_is_pad_and_regulator_valid(struct sdhci_host *host)  {  	struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host); @@ -1223,10 +1235,25 @@ static const struct cqhci_host_ops sdhci_tegra_cqhci_ops = {  	.update_dcmd_desc = sdhci_tegra_update_dcmd_desc,  }; +static int tegra_sdhci_set_dma_mask(struct sdhci_host *host) +{ +	struct sdhci_pltfm_host *platform = sdhci_priv(host); +	struct sdhci_tegra *tegra = sdhci_pltfm_priv(platform); +	const struct sdhci_tegra_soc_data *soc = tegra->soc_data; +	struct device *dev = mmc_dev(host->mmc); + +	if (soc->dma_mask) +		return dma_set_mask_and_coherent(dev, soc->dma_mask); + +	return 0; +} +  static const struct sdhci_ops tegra_sdhci_ops = { +	.get_ro     = tegra_sdhci_get_ro,  	.read_w     = tegra_sdhci_readw,  	.write_l    = tegra_sdhci_writel,  	.set_clock  = tegra_sdhci_set_clock, +	.set_dma_mask = tegra_sdhci_set_dma_mask,  	.set_bus_width = sdhci_set_bus_width,  	.reset      = tegra_sdhci_reset,  	.platform_execute_tuning = tegra_sdhci_execute_tuning, @@ -1246,6 +1273,7 @@ static const struct sdhci_pltfm_data sdhci_tegra20_pdata = {  static const struct sdhci_tegra_soc_data soc_data_tegra20 = {  	.pdata = &sdhci_tegra20_pdata, +	.dma_mask = DMA_BIT_MASK(32),  	.nvquirks = NVQUIRK_FORCE_SDHCI_SPEC_200 |  		    NVQUIRK_ENABLE_BLOCK_GAP_DET,  }; @@ -1272,6 +1300,7 @@ static const struct sdhci_pltfm_data sdhci_tegra30_pdata = {  static const struct sdhci_tegra_soc_data soc_data_tegra30 = {  	.pdata = &sdhci_tegra30_pdata, +	.dma_mask = DMA_BIT_MASK(32),  	.nvquirks = NVQUIRK_ENABLE_SDHCI_SPEC_300 |  		    NVQUIRK_ENABLE_SDR50 |  		    NVQUIRK_ENABLE_SDR104 | @@ -1279,10 +1308,12 @@ static const struct sdhci_tegra_soc_data soc_data_tegra30 = {  };  static const struct sdhci_ops tegra114_sdhci_ops = { +	.get_ro     = tegra_sdhci_get_ro,  	.read_w     = tegra_sdhci_readw,  	.write_w    = tegra_sdhci_writew,  	.write_l    = tegra_sdhci_writel,  	.set_clock  = tegra_sdhci_set_clock, +	.set_dma_mask = tegra_sdhci_set_dma_mask,  	.set_bus_width = sdhci_set_bus_width,  	.reset      = tegra_sdhci_reset,  	.platform_execute_tuning = tegra_sdhci_execute_tuning, @@ -1304,6 +1335,7 @@ static const struct sdhci_pltfm_data sdhci_tegra114_pdata = {  static const struct sdhci_tegra_soc_data soc_data_tegra114 = {  	.pdata = &sdhci_tegra114_pdata, +	.dma_mask = DMA_BIT_MASK(32),  };  static const struct sdhci_pltfm_data sdhci_tegra124_pdata = { @@ -1313,29 +1345,22 @@ static const struct sdhci_pltfm_data sdhci_tegra124_pdata = {  		  SDHCI_QUIRK_NO_HISPD_BIT |  		  SDHCI_QUIRK_BROKEN_ADMA_ZEROLEN_DESC |  		  SDHCI_QUIRK_CAP_CLOCK_BASE_BROKEN, -	.quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN | -		   /* -		    * The TRM states that the SD/MMC controller found on -		    * Tegra124 can address 34 bits (the maximum supported by -		    * the Tegra memory controller), but tests show that DMA -		    * to or from above 4 GiB doesn't work. This is possibly -		    * caused by missing programming, though it's not obvious -		    * what sequence is required. Mark 64-bit DMA broken for -		    * now to fix this for existing users (e.g. Nyan boards). -		    */ -		   SDHCI_QUIRK2_BROKEN_64_BIT_DMA, +	.quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN,  	.ops  = &tegra114_sdhci_ops,  };  static const struct sdhci_tegra_soc_data soc_data_tegra124 = {  	.pdata = &sdhci_tegra124_pdata, +	.dma_mask = DMA_BIT_MASK(34),  };  static const struct sdhci_ops tegra210_sdhci_ops = { +	.get_ro     = tegra_sdhci_get_ro,  	.read_w     = tegra_sdhci_readw,  	.write_w    = tegra210_sdhci_writew,  	.write_l    = tegra_sdhci_writel,  	.set_clock  = tegra_sdhci_set_clock, +	.set_dma_mask = tegra_sdhci_set_dma_mask,  	.set_bus_width = sdhci_set_bus_width,  	.reset      = tegra_sdhci_reset,  	.set_uhs_signaling = tegra_sdhci_set_uhs_signaling, @@ -1356,6 +1381,7 @@ static const struct sdhci_pltfm_data sdhci_tegra210_pdata = {  static const struct sdhci_tegra_soc_data soc_data_tegra210 = {  	.pdata = &sdhci_tegra210_pdata, +	.dma_mask = DMA_BIT_MASK(34),  	.nvquirks = NVQUIRK_NEEDS_PAD_CONTROL |  		    NVQUIRK_HAS_PADCALIB |  		    NVQUIRK_DIS_CARD_CLK_CONFIG_TAP | @@ -1366,9 +1392,11 @@ static const struct sdhci_tegra_soc_data soc_data_tegra210 = {  };  static const struct sdhci_ops tegra186_sdhci_ops = { +	.get_ro     = tegra_sdhci_get_ro,  	.read_w     = tegra_sdhci_readw,  	.write_l    = tegra_sdhci_writel,  	.set_clock  = tegra_sdhci_set_clock, +	.set_dma_mask = tegra_sdhci_set_dma_mask,  	.set_bus_width = sdhci_set_bus_width,  	.reset      = tegra_sdhci_reset,  	.set_uhs_signaling = tegra_sdhci_set_uhs_signaling, @@ -1384,20 +1412,13 @@ static const struct sdhci_pltfm_data sdhci_tegra186_pdata = {  		  SDHCI_QUIRK_NO_HISPD_BIT |  		  SDHCI_QUIRK_BROKEN_ADMA_ZEROLEN_DESC |  		  SDHCI_QUIRK_CAP_CLOCK_BASE_BROKEN, -	.quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN | -		   /* SDHCI controllers on Tegra186 support 40-bit addressing. -		    * IOVA addresses are 48-bit wide on Tegra186. -		    * With 64-bit dma mask used for SDHCI, accesses can -		    * be broken. Disable 64-bit dma, which would fall back -		    * to 32-bit dma mask. Ideally 40-bit dma mask would work, -		    * But it is not supported as of now. -		    */ -		   SDHCI_QUIRK2_BROKEN_64_BIT_DMA, +	.quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN,  	.ops  = &tegra186_sdhci_ops,  };  static const struct sdhci_tegra_soc_data soc_data_tegra186 = {  	.pdata = &sdhci_tegra186_pdata, +	.dma_mask = DMA_BIT_MASK(40),  	.nvquirks = NVQUIRK_NEEDS_PAD_CONTROL |  		    NVQUIRK_HAS_PADCALIB |  		    NVQUIRK_DIS_CARD_CLK_CONFIG_TAP | @@ -1410,6 +1431,7 @@ static const struct sdhci_tegra_soc_data soc_data_tegra186 = {  static const struct sdhci_tegra_soc_data soc_data_tegra194 = {  	.pdata = &sdhci_tegra186_pdata, +	.dma_mask = DMA_BIT_MASK(39),  	.nvquirks = NVQUIRK_NEEDS_PAD_CONTROL |  		    NVQUIRK_HAS_PADCALIB |  		    NVQUIRK_DIS_CARD_CLK_CONFIG_TAP | diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c index a5dc5aae973e..b056400e34b1 100644 --- a/drivers/mmc/host/sdhci.c +++ b/drivers/mmc/host/sdhci.c @@ -668,10 +668,10 @@ void sdhci_adma_write_desc(struct sdhci_host *host, void **desc,  	/* 32-bit and 64-bit descriptors have these members in same position */  	dma_desc->cmd = cpu_to_le16(cmd);  	dma_desc->len = cpu_to_le16(len); -	dma_desc->addr_lo = cpu_to_le32((u32)addr); +	dma_desc->addr_lo = cpu_to_le32(lower_32_bits(addr));  	if (host->flags & SDHCI_USE_64_BIT_DMA) -		dma_desc->addr_hi = cpu_to_le32((u64)addr >> 32); +		dma_desc->addr_hi = cpu_to_le32(upper_32_bits(addr));  	*desc += host->desc_sz;  } @@ -816,6 +816,13 @@ static void sdhci_adma_table_post(struct sdhci_host *host,  	}  } +static void sdhci_set_adma_addr(struct sdhci_host *host, dma_addr_t addr) +{ +	sdhci_writel(host, lower_32_bits(addr), SDHCI_ADMA_ADDRESS); +	if (host->flags & SDHCI_USE_64_BIT_DMA) +		sdhci_writel(host, upper_32_bits(addr), SDHCI_ADMA_ADDRESS_HI); +} +  static dma_addr_t sdhci_sdma_address(struct sdhci_host *host)  {  	if (host->bounce_buffer) @@ -826,13 +833,10 @@ static dma_addr_t sdhci_sdma_address(struct sdhci_host *host)  static void sdhci_set_sdma_addr(struct sdhci_host *host, dma_addr_t addr)  { -	if (host->v4_mode) { -		sdhci_writel(host, addr, SDHCI_ADMA_ADDRESS); -		if (host->flags & SDHCI_USE_64_BIT_DMA) -			sdhci_writel(host, (u64)addr >> 32, SDHCI_ADMA_ADDRESS_HI); -	} else { +	if (host->v4_mode) +		sdhci_set_adma_addr(host, addr); +	else  		sdhci_writel(host, addr, SDHCI_DMA_ADDRESS); -	}  }  static unsigned int sdhci_target_timeout(struct sdhci_host *host, @@ -1095,12 +1099,7 @@ static void sdhci_prepare_data(struct sdhci_host *host, struct mmc_command *cmd)  			host->flags &= ~SDHCI_REQ_USE_DMA;  		} else if (host->flags & SDHCI_USE_ADMA) {  			sdhci_adma_table_pre(host, data, sg_cnt); - -			sdhci_writel(host, host->adma_addr, SDHCI_ADMA_ADDRESS); -			if (host->flags & SDHCI_USE_64_BIT_DMA) -				sdhci_writel(host, -					     (u64)host->adma_addr >> 32, -					     SDHCI_ADMA_ADDRESS_HI); +			sdhci_set_adma_addr(host, host->adma_addr);  		} else {  			WARN_ON(sg_cnt != 1);  			sdhci_set_sdma_addr(host, sdhci_sdma_address(host)); @@ -1636,8 +1635,8 @@ void sdhci_enable_clk(struct sdhci_host *host, u16 clk)  	clk |= SDHCI_CLOCK_INT_EN;  	sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL); -	/* Wait max 20 ms */ -	timeout = ktime_add_ms(ktime_get(), 20); +	/* Wait max 150 ms */ +	timeout = ktime_add_ms(ktime_get(), 150);  	while (1) {  		bool timedout = ktime_after(ktime_get(), timeout); @@ -1653,6 +1652,29 @@ void sdhci_enable_clk(struct sdhci_host *host, u16 clk)  		udelay(10);  	} +	if (host->version >= SDHCI_SPEC_410 && host->v4_mode) { +		clk |= SDHCI_CLOCK_PLL_EN; +		clk &= ~SDHCI_CLOCK_INT_STABLE; +		sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL); + +		/* Wait max 150 ms */ +		timeout = ktime_add_ms(ktime_get(), 150); +		while (1) { +			bool timedout = ktime_after(ktime_get(), timeout); + +			clk = sdhci_readw(host, SDHCI_CLOCK_CONTROL); +			if (clk & SDHCI_CLOCK_INT_STABLE) +				break; +			if (timedout) { +				pr_err("%s: PLL clock never stabilised.\n", +				       mmc_hostname(host->mmc)); +				sdhci_dumpregs(host); +				return; +			} +			udelay(10); +		} +	} +  	clk |= SDHCI_CLOCK_CARD_EN;  	sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL);  } @@ -1849,7 +1871,9 @@ void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing)  		ctrl_2 |= SDHCI_CTRL_UHS_SDR104;  	else if (timing == MMC_TIMING_UHS_SDR12)  		ctrl_2 |= SDHCI_CTRL_UHS_SDR12; -	else if (timing == MMC_TIMING_UHS_SDR25) +	else if (timing == MMC_TIMING_SD_HS || +		 timing == MMC_TIMING_MMC_HS || +		 timing == MMC_TIMING_UHS_SDR25)  		ctrl_2 |= SDHCI_CTRL_UHS_SDR25;  	else if (timing == MMC_TIMING_UHS_SDR50)  		ctrl_2 |= SDHCI_CTRL_UHS_SDR50; @@ -2120,11 +2144,6 @@ void sdhci_enable_sdio_irq(struct mmc_host *mmc, int enable)  		pm_runtime_get_noresume(host->mmc->parent);  	spin_lock_irqsave(&host->lock, flags); -	if (enable) -		host->flags |= SDHCI_SDIO_IRQ_ENABLED; -	else -		host->flags &= ~SDHCI_SDIO_IRQ_ENABLED; -  	sdhci_enable_sdio_irq_nolock(host, enable);  	spin_unlock_irqrestore(&host->lock, flags); @@ -2139,8 +2158,7 @@ static void sdhci_ack_sdio_irq(struct mmc_host *mmc)  	unsigned long flags;  	spin_lock_irqsave(&host->lock, flags); -	if (host->flags & SDHCI_SDIO_IRQ_ENABLED) -		sdhci_enable_sdio_irq_nolock(host, true); +	sdhci_enable_sdio_irq_nolock(host, true);  	spin_unlock_irqrestore(&host->lock, flags);  } @@ -2305,7 +2323,7 @@ void sdhci_reset_tuning(struct sdhci_host *host)  }  EXPORT_SYMBOL_GPL(sdhci_reset_tuning); -static void sdhci_abort_tuning(struct sdhci_host *host, u32 opcode) +void sdhci_abort_tuning(struct sdhci_host *host, u32 opcode)  {  	sdhci_reset_tuning(host); @@ -2316,6 +2334,7 @@ static void sdhci_abort_tuning(struct sdhci_host *host, u32 opcode)  	mmc_abort_tuning(host->mmc, opcode);  } +EXPORT_SYMBOL_GPL(sdhci_abort_tuning);  /*   * We use sdhci_send_tuning() because mmc_send_tuning() is not a good fit. SDHCI @@ -2855,6 +2874,7 @@ static void sdhci_cmd_irq(struct sdhci_host *host, u32 intmask, u32 *intmask_p)  static void sdhci_adma_show_error(struct sdhci_host *host)  {  	void *desc = host->adma_table; +	dma_addr_t dma = host->adma_addr;  	sdhci_dumpregs(host); @@ -2862,18 +2882,21 @@ static void sdhci_adma_show_error(struct sdhci_host *host)  		struct sdhci_adma2_64_desc *dma_desc = desc;  		if (host->flags & SDHCI_USE_64_BIT_DMA) -			DBG("%p: DMA 0x%08x%08x, LEN 0x%04x, Attr=0x%02x\n", -			    desc, le32_to_cpu(dma_desc->addr_hi), +			SDHCI_DUMP("%08llx: DMA 0x%08x%08x, LEN 0x%04x, Attr=0x%02x\n", +			    (unsigned long long)dma, +			    le32_to_cpu(dma_desc->addr_hi),  			    le32_to_cpu(dma_desc->addr_lo),  			    le16_to_cpu(dma_desc->len),  			    le16_to_cpu(dma_desc->cmd));  		else -			DBG("%p: DMA 0x%08x, LEN 0x%04x, Attr=0x%02x\n", -			    desc, le32_to_cpu(dma_desc->addr_lo), +			SDHCI_DUMP("%08llx: DMA 0x%08x, LEN 0x%04x, Attr=0x%02x\n", +			    (unsigned long long)dma, +			    le32_to_cpu(dma_desc->addr_lo),  			    le16_to_cpu(dma_desc->len),  			    le16_to_cpu(dma_desc->cmd));  		desc += host->desc_sz; +		dma += host->desc_sz;  		if (dma_desc->cmd & cpu_to_le16(ADMA2_END))  			break; @@ -2949,7 +2972,8 @@ static void sdhci_data_irq(struct sdhci_host *host, u32 intmask)  			!= MMC_BUS_TEST_R)  		host->data->error = -EILSEQ;  	else if (intmask & SDHCI_INT_ADMA_ERROR) { -		pr_err("%s: ADMA error\n", mmc_hostname(host->mmc)); +		pr_err("%s: ADMA error: 0x%08x\n", mmc_hostname(host->mmc), +		       intmask);  		sdhci_adma_show_error(host);  		host->data->error = -EIO;  		if (host->ops->adma_workaround) @@ -3024,7 +3048,7 @@ static irqreturn_t sdhci_irq(int irq, void *dev_id)  	spin_lock(&host->lock); -	if (host->runtime_suspended && !sdhci_sdio_irq_enabled(host)) { +	if (host->runtime_suspended) {  		spin_unlock(&host->lock);  		return IRQ_NONE;  	} @@ -3358,7 +3382,7 @@ int sdhci_runtime_resume_host(struct sdhci_host *host, int soft_reset)  	host->runtime_suspended = false;  	/* Enable SDIO IRQ */ -	if (host->flags & SDHCI_SDIO_IRQ_ENABLED) +	if (sdio_irq_claimed(mmc))  		sdhci_enable_sdio_irq_nolock(host, true);  	/* Enable Card Detection */ @@ -3565,7 +3589,8 @@ static int sdhci_set_dma_mask(struct sdhci_host *host)  	return ret;  } -void __sdhci_read_caps(struct sdhci_host *host, u16 *ver, u32 *caps, u32 *caps1) +void __sdhci_read_caps(struct sdhci_host *host, const u16 *ver, +		       const u32 *caps, const u32 *caps1)  {  	u16 v;  	u64 dt_caps_mask = 0; @@ -3756,18 +3781,14 @@ int sdhci_setup_host(struct sdhci_host *host)  		host->flags &= ~SDHCI_USE_ADMA;  	} -	/* -	 * It is assumed that a 64-bit capable device has set a 64-bit DMA mask -	 * and *must* do 64-bit DMA.  A driver has the opportunity to change -	 * that during the first call to ->enable_dma().  Similarly -	 * SDHCI_QUIRK2_BROKEN_64_BIT_DMA must be left to the drivers to -	 * implement. -	 */  	if (sdhci_can_64bit_dma(host))  		host->flags |= SDHCI_USE_64_BIT_DMA;  	if (host->flags & (SDHCI_USE_SDMA | SDHCI_USE_ADMA)) { -		ret = sdhci_set_dma_mask(host); +		if (host->ops->set_dma_mask) +			ret = host->ops->set_dma_mask(host); +		else +			ret = sdhci_set_dma_mask(host);  		if (!ret && host->ops->enable_dma)  			ret = host->ops->enable_dma(host); diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h index 902f855efe8f..0ed3e0eaef5f 100644 --- a/drivers/mmc/host/sdhci.h +++ b/drivers/mmc/host/sdhci.h @@ -114,6 +114,7 @@  #define  SDHCI_DIV_HI_MASK	0x300  #define  SDHCI_PROG_CLOCK_MODE	0x0020  #define  SDHCI_CLOCK_CARD_EN	0x0004 +#define  SDHCI_CLOCK_PLL_EN	0x0008  #define  SDHCI_CLOCK_INT_STABLE	0x0002  #define  SDHCI_CLOCK_INT_EN	0x0001 @@ -511,7 +512,6 @@ struct sdhci_host {  #define SDHCI_AUTO_CMD12	(1<<6)	/* Auto CMD12 support */  #define SDHCI_AUTO_CMD23	(1<<7)	/* Auto CMD23 support */  #define SDHCI_PV_ENABLED	(1<<8)	/* Preset value enabled */ -#define SDHCI_SDIO_IRQ_ENABLED	(1<<9)	/* SDIO irq enabled */  #define SDHCI_USE_64_BIT_DMA	(1<<12)	/* Use 64-bit DMA */  #define SDHCI_HS400_TUNING	(1<<13)	/* Tuning for HS400 */  #define SDHCI_SIGNALING_330	(1<<14)	/* Host is capable of 3.3V signaling */ @@ -622,6 +622,7 @@ struct sdhci_ops {  	u32		(*irq)(struct sdhci_host *host, u32 intmask); +	int		(*set_dma_mask)(struct sdhci_host *host);  	int		(*enable_dma)(struct sdhci_host *host);  	unsigned int	(*get_max_clock)(struct sdhci_host *host);  	unsigned int	(*get_min_clock)(struct sdhci_host *host); @@ -738,8 +739,8 @@ static inline void *sdhci_priv(struct sdhci_host *host)  }  void sdhci_card_detect(struct sdhci_host *host); -void __sdhci_read_caps(struct sdhci_host *host, u16 *ver, u32 *caps, -		       u32 *caps1); +void __sdhci_read_caps(struct sdhci_host *host, const u16 *ver, +		       const u32 *caps, const u32 *caps1);  int sdhci_setup_host(struct sdhci_host *host);  void sdhci_cleanup_host(struct sdhci_host *host);  int __sdhci_add_host(struct sdhci_host *host); @@ -752,11 +753,6 @@ static inline void sdhci_read_caps(struct sdhci_host *host)  	__sdhci_read_caps(host, NULL, NULL, NULL);  } -static inline bool sdhci_sdio_irq_enabled(struct sdhci_host *host) -{ -	return !!(host->flags & SDHCI_SDIO_IRQ_ENABLED); -} -  u16 sdhci_calc_clk(struct sdhci_host *host, unsigned int clock,  		   unsigned int *actual_clock);  void sdhci_set_clock(struct sdhci_host *host, unsigned int clock); @@ -796,5 +792,6 @@ void sdhci_start_tuning(struct sdhci_host *host);  void sdhci_end_tuning(struct sdhci_host *host);  void sdhci_reset_tuning(struct sdhci_host *host);  void sdhci_send_tuning(struct sdhci_host *host, u32 opcode); +void sdhci_abort_tuning(struct sdhci_host *host, u32 opcode);  #endif /* __SDHCI_HW_H */ diff --git a/drivers/mmc/host/sdhci_f_sdh30.c b/drivers/mmc/host/sdhci_f_sdh30.c index e369cbf1ff02..f8b939e63e02 100644 --- a/drivers/mmc/host/sdhci_f_sdh30.c +++ b/drivers/mmc/host/sdhci_f_sdh30.c @@ -119,10 +119,8 @@ static int sdhci_f_sdh30_probe(struct platform_device *pdev)  	u32 reg = 0;  	irq = platform_get_irq(pdev, 0); -	if (irq < 0) { -		dev_err(dev, "%s: no irq specified\n", __func__); +	if (irq < 0)  		return irq; -	}  	host = sdhci_alloc_host(dev, sizeof(struct f_sdhost_priv));  	if (IS_ERR(host)) diff --git a/drivers/mmc/host/tmio_mmc.c b/drivers/mmc/host/tmio_mmc.c index 8539e10784b4..93e83ad25976 100644 --- a/drivers/mmc/host/tmio_mmc.c +++ b/drivers/mmc/host/tmio_mmc.c @@ -172,8 +172,6 @@ static int tmio_mmc_probe(struct platform_device *pdev)  	host->mmc->f_max = pdata->hclk;  	host->mmc->f_min = pdata->hclk / 512; -	pm_runtime_enable(&pdev->dev); -  	ret = tmio_mmc_host_probe(host);  	if (ret)  		goto host_free; @@ -193,7 +191,6 @@ host_remove:  	tmio_mmc_host_remove(host);  host_free:  	tmio_mmc_host_free(host); -	pm_runtime_disable(&pdev->dev);  cell_disable:  	if (cell->disable)  		cell->disable(pdev); @@ -210,8 +207,6 @@ static int tmio_mmc_remove(struct platform_device *pdev)  	if (cell->disable)  		cell->disable(pdev); -	pm_runtime_disable(&pdev->dev); -  	return 0;  } diff --git a/drivers/mmc/host/tmio_mmc.h b/drivers/mmc/host/tmio_mmc.h index c5ba13fae399..2f0b092d6dcc 100644 --- a/drivers/mmc/host/tmio_mmc.h +++ b/drivers/mmc/host/tmio_mmc.h @@ -163,6 +163,7 @@ struct tmio_mmc_host {  	unsigned long		last_req_ts;  	struct mutex		ios_lock;	/* protect set_ios() context */  	bool			native_hotplug; +	bool			runtime_synced;  	bool			sdio_irq_enabled;  	/* Mandatory callback */ diff --git a/drivers/mmc/host/tmio_mmc_core.c b/drivers/mmc/host/tmio_mmc_core.c index 2cb3f951c3e2..9b6e1001e77c 100644 --- a/drivers/mmc/host/tmio_mmc_core.c +++ b/drivers/mmc/host/tmio_mmc_core.c @@ -1153,15 +1153,6 @@ void tmio_mmc_host_free(struct tmio_mmc_host *host)  }  EXPORT_SYMBOL_GPL(tmio_mmc_host_free); -/** - * tmio_mmc_host_probe() - Common probe for all implementations - * @_host: Host to probe - * - * Perform tasks common to all implementations probe functions. - * - * The caller should have called pm_runtime_enable() prior to calling - * the common probe function. - */  int tmio_mmc_host_probe(struct tmio_mmc_host *_host)  {  	struct platform_device *pdev = _host->pdev; @@ -1257,19 +1248,22 @@ int tmio_mmc_host_probe(struct tmio_mmc_host *_host)  	/* See if we also get DMA */  	tmio_mmc_request_dma(_host, pdata); -	pm_runtime_set_active(&pdev->dev);  	pm_runtime_set_autosuspend_delay(&pdev->dev, 50);  	pm_runtime_use_autosuspend(&pdev->dev); +	pm_runtime_enable(&pdev->dev); +	pm_runtime_get_sync(&pdev->dev);  	ret = mmc_add_host(mmc);  	if (ret)  		goto remove_host;  	dev_pm_qos_expose_latency_limit(&pdev->dev, 100); +	pm_runtime_put(&pdev->dev);  	return 0;  remove_host: +	pm_runtime_put_noidle(&pdev->dev);  	tmio_mmc_host_remove(_host);  	return ret;  } @@ -1280,12 +1274,11 @@ void tmio_mmc_host_remove(struct tmio_mmc_host *host)  	struct platform_device *pdev = host->pdev;  	struct mmc_host *mmc = host->mmc; +	pm_runtime_get_sync(&pdev->dev); +  	if (host->pdata->flags & TMIO_MMC_SDIO_IRQ)  		sd_ctrl_write16(host, CTL_TRANSACTION_CTL, 0x0000); -	if (!host->native_hotplug) -		pm_runtime_get_sync(&pdev->dev); -  	dev_pm_qos_hide_latency_limit(&pdev->dev);  	mmc_remove_host(mmc); @@ -1294,7 +1287,10 @@ void tmio_mmc_host_remove(struct tmio_mmc_host *host)  	tmio_mmc_release_dma(host);  	pm_runtime_dont_use_autosuspend(&pdev->dev); +	if (host->native_hotplug) +		pm_runtime_put_noidle(&pdev->dev);  	pm_runtime_put_sync(&pdev->dev); +	pm_runtime_disable(&pdev->dev);  }  EXPORT_SYMBOL_GPL(tmio_mmc_host_remove); @@ -1337,6 +1333,11 @@ int tmio_mmc_host_runtime_resume(struct device *dev)  {  	struct tmio_mmc_host *host = dev_get_drvdata(dev); +	if (!host->runtime_synced) { +		host->runtime_synced = true; +		return 0; +	} +  	tmio_mmc_clk_enable(host);  	tmio_mmc_hw_reset(host->mmc); diff --git a/drivers/mmc/host/uniphier-sd.c b/drivers/mmc/host/uniphier-sd.c index 49aad9a79c18..0c72ec5546c3 100644 --- a/drivers/mmc/host/uniphier-sd.c +++ b/drivers/mmc/host/uniphier-sd.c @@ -557,10 +557,8 @@ static int uniphier_sd_probe(struct platform_device *pdev)  	int irq, ret;  	irq = platform_get_irq(pdev, 0); -	if (irq < 0) { -		dev_err(dev, "failed to get IRQ number"); +	if (irq < 0)  		return irq; -	}  	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);  	if (!priv) @@ -631,7 +629,6 @@ static int uniphier_sd_probe(struct platform_device *pdev)  	host->clk_disable = uniphier_sd_clk_disable;  	host->set_clock = uniphier_sd_set_clock; -	pm_runtime_enable(&pdev->dev);  	ret = uniphier_sd_clk_enable(host);  	if (ret)  		goto free_host; @@ -653,7 +650,6 @@ static int uniphier_sd_probe(struct platform_device *pdev)  free_host:  	tmio_mmc_host_free(host); -	pm_runtime_disable(&pdev->dev);  	return ret;  } @@ -664,7 +660,6 @@ static int uniphier_sd_remove(struct platform_device *pdev)  	tmio_mmc_host_remove(host);  	uniphier_sd_clk_disable(host); -	pm_runtime_disable(&pdev->dev);  	return 0;  } | 

