| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * OpenFirmware bindings for Secure Digital Host Controller Interface. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2007 Freescale Semiconductor, Inc. | 
					
						
							|  |  |  |  * Copyright (c) 2009 MontaVista Software, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Authors: Xiaobo Xie <X.Xie@freescale.com> | 
					
						
							|  |  |  |  *	    Anton Vorontsov <avorontsov@ru.mvista.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation; either version 2 of the License, or (at | 
					
						
							|  |  |  |  * your option) any later version. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/init.h>
 | 
					
						
							|  |  |  | #include <linux/io.h>
 | 
					
						
							|  |  |  | #include <linux/interrupt.h>
 | 
					
						
							|  |  |  | #include <linux/delay.h>
 | 
					
						
							|  |  |  | #include <linux/of.h>
 | 
					
						
							|  |  |  | #include <linux/of_platform.h>
 | 
					
						
							|  |  |  | #include <linux/mmc/host.h>
 | 
					
						
							| 
									
										
										
										
											2009-09-22 16:45:15 -07:00
										 |  |  | #include <asm/machdep.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #include "sdhci-of.h"
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | #include "sdhci.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #ifdef CONFIG_MMC_SDHCI_BIG_ENDIAN_32BIT_BYTE_SWAPPER
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  |  * These accessors are designed for big endian hosts doing I/O to | 
					
						
							|  |  |  |  * little endian controllers incorporating a 32-bit hardware byte swapper. | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | u32 sdhci_be32bs_readl(struct sdhci_host *host, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	return in_be32(host->ioaddr + reg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | u16 sdhci_be32bs_readw(struct sdhci_host *host, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | 	return in_be16(host->ioaddr + (reg ^ 0x2)); | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | u8 sdhci_be32bs_readb(struct sdhci_host *host, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	return in_8(host->ioaddr + (reg ^ 0x3)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | void sdhci_be32bs_writel(struct sdhci_host *host, u32 val, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	out_be32(host->ioaddr + reg, val); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | void sdhci_be32bs_writew(struct sdhci_host *host, u16 val, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct sdhci_of_host *of_host = sdhci_priv(host); | 
					
						
							|  |  |  | 	int base = reg & ~0x3; | 
					
						
							|  |  |  | 	int shift = (reg & 0x2) * 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (reg) { | 
					
						
							|  |  |  | 	case SDHCI_TRANSFER_MODE: | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * Postpone this write, we must do it together with a | 
					
						
							|  |  |  | 		 * command write that is down below. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		of_host->xfer_mode_shadow = val; | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	case SDHCI_COMMAND: | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | 		sdhci_be32bs_writel(host, val << 16 | of_host->xfer_mode_shadow, | 
					
						
							|  |  |  | 				    SDHCI_TRANSFER_MODE); | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	clrsetbits_be32(host->ioaddr + base, 0xffff << shift, val << shift); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | void sdhci_be32bs_writeb(struct sdhci_host *host, u8 val, int reg) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  | 	int base = reg & ~0x3; | 
					
						
							|  |  |  | 	int shift = (reg & 0x3) * 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	clrsetbits_be32(host->ioaddr + base , 0xff << shift, val << shift); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #endif /* CONFIG_MMC_SDHCI_BIG_ENDIAN_32BIT_BYTE_SWAPPER */
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_PM
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int sdhci_of_suspend(struct of_device *ofdev, pm_message_t state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sdhci_host *host = dev_get_drvdata(&ofdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return mmc_suspend_host(host->mmc, state); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int sdhci_of_resume(struct of_device *ofdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sdhci_host *host = dev_get_drvdata(&ofdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return mmc_resume_host(host->mmc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define sdhci_of_suspend NULL
 | 
					
						
							|  |  |  | #define sdhci_of_resume NULL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-22 16:45:15 -07:00
										 |  |  | static bool __devinit sdhci_of_wp_inverted(struct device_node *np) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (of_get_property(np, "sdhci,wp-inverted", NULL)) | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Old device trees don't have the wp-inverted property. */ | 
					
						
							|  |  |  | 	return machine_is(mpc837x_rdb) || machine_is(mpc837x_mds); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | static int __devinit sdhci_of_probe(struct of_device *ofdev, | 
					
						
							|  |  |  | 				 const struct of_device_id *match) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device_node *np = ofdev->node; | 
					
						
							|  |  |  | 	struct sdhci_of_data *sdhci_of_data = match->data; | 
					
						
							|  |  |  | 	struct sdhci_host *host; | 
					
						
							|  |  |  | 	struct sdhci_of_host *of_host; | 
					
						
							|  |  |  | 	const u32 *clk; | 
					
						
							|  |  |  | 	int size; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!of_device_is_available(np)) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	host = sdhci_alloc_host(&ofdev->dev, sizeof(*of_host)); | 
					
						
							| 
									
										
										
										
											2009-08-06 15:07:41 -07:00
										 |  |  | 	if (IS_ERR(host)) | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	of_host = sdhci_priv(host); | 
					
						
							|  |  |  | 	dev_set_drvdata(&ofdev->dev, host); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	host->ioaddr = of_iomap(np, 0); | 
					
						
							|  |  |  | 	if (!host->ioaddr) { | 
					
						
							|  |  |  | 		ret = -ENOMEM; | 
					
						
							|  |  |  | 		goto err_addr_map; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	host->irq = irq_of_parse_and_map(np, 0); | 
					
						
							|  |  |  | 	if (!host->irq) { | 
					
						
							|  |  |  | 		ret = -EINVAL; | 
					
						
							|  |  |  | 		goto err_no_irq; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	host->hw_name = dev_name(&ofdev->dev); | 
					
						
							|  |  |  | 	if (sdhci_of_data) { | 
					
						
							|  |  |  | 		host->quirks = sdhci_of_data->quirks; | 
					
						
							|  |  |  | 		host->ops = &sdhci_of_data->ops; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-18 00:14:08 +04:00
										 |  |  | 	if (of_get_property(np, "sdhci,1-bit-only", NULL)) | 
					
						
							|  |  |  | 		host->quirks |= SDHCI_QUIRK_FORCE_1_BIT_DATA; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-22 16:45:15 -07:00
										 |  |  | 	if (sdhci_of_wp_inverted(np)) | 
					
						
							|  |  |  | 		host->quirks |= SDHCI_QUIRK_INVERTED_WRITE_PROTECT; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 	clk = of_get_property(np, "clock-frequency", &size); | 
					
						
							|  |  |  | 	if (clk && size == sizeof(*clk) && *clk) | 
					
						
							|  |  |  | 		of_host->clock = *clk; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = sdhci_add_host(host); | 
					
						
							|  |  |  | 	if (ret) | 
					
						
							|  |  |  | 		goto err_add_host; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | err_add_host: | 
					
						
							|  |  |  | 	irq_dispose_mapping(host->irq); | 
					
						
							|  |  |  | err_no_irq: | 
					
						
							|  |  |  | 	iounmap(host->ioaddr); | 
					
						
							|  |  |  | err_addr_map: | 
					
						
							|  |  |  | 	sdhci_free_host(host); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __devexit sdhci_of_remove(struct of_device *ofdev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sdhci_host *host = dev_get_drvdata(&ofdev->dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sdhci_remove_host(host, 0); | 
					
						
							|  |  |  | 	sdhci_free_host(host); | 
					
						
							|  |  |  | 	irq_dispose_mapping(host->irq); | 
					
						
							|  |  |  | 	iounmap(host->ioaddr); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct of_device_id sdhci_of_match[] = { | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #ifdef CONFIG_MMC_SDHCI_OF_ESDHC
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 	{ .compatible = "fsl,mpc8379-esdhc", .data = &sdhci_esdhc, }, | 
					
						
							|  |  |  | 	{ .compatible = "fsl,mpc8536-esdhc", .data = &sdhci_esdhc, }, | 
					
						
							| 
									
										
										
										
											2009-05-08 08:52:49 -05:00
										 |  |  | 	{ .compatible = "fsl,esdhc", .data = &sdhci_esdhc, }, | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef CONFIG_MMC_SDHCI_OF_HLWD
 | 
					
						
							|  |  |  | 	{ .compatible = "nintendo,hollywood-sdhci", .data = &sdhci_hlwd, }, | 
					
						
							| 
									
										
										
										
											2009-12-17 15:27:20 -08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-03-17 00:14:05 +03:00
										 |  |  | 	{ .compatible = "generic-sdhci", }, | 
					
						
							|  |  |  | 	{}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | MODULE_DEVICE_TABLE(of, sdhci_of_match); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct of_platform_driver sdhci_of_driver = { | 
					
						
							|  |  |  | 	.driver.name = "sdhci-of", | 
					
						
							|  |  |  | 	.match_table = sdhci_of_match, | 
					
						
							|  |  |  | 	.probe = sdhci_of_probe, | 
					
						
							|  |  |  | 	.remove = __devexit_p(sdhci_of_remove), | 
					
						
							|  |  |  | 	.suspend = sdhci_of_suspend, | 
					
						
							|  |  |  | 	.resume	= sdhci_of_resume, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init sdhci_of_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return of_register_platform_driver(&sdhci_of_driver); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | module_init(sdhci_of_init); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit sdhci_of_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	of_unregister_platform_driver(&sdhci_of_driver); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | module_exit(sdhci_of_exit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Secure Digital Host Controller Interface OF driver"); | 
					
						
							|  |  |  | MODULE_AUTHOR("Xiaobo Xie <X.Xie@freescale.com>, " | 
					
						
							|  |  |  | 	      "Anton Vorontsov <avorontsov@ru.mvista.com>"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |