| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  platform.c: platform support for PNX833X. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright 2008 NXP Semiconductors | 
					
						
							|  |  |  |  *	  Chris Steel <chris.steel@nxp.com> | 
					
						
							|  |  |  |  *    Daniel Laird <daniel.j.laird@nxp.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Based on software written by: | 
					
						
							|  |  |  |  *      Nikita Youshchenko <yoush@debian.org>, based on PNX8550 code. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  *  it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  *  the Free Software Foundation; either version 2 of the License, or | 
					
						
							|  |  |  |  *  (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  *  GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  *  along with this program; if not, write to the Free Software | 
					
						
							|  |  |  |  *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <linux/device.h>
 | 
					
						
							|  |  |  | #include <linux/dma-mapping.h>
 | 
					
						
							|  |  |  | #include <linux/platform_device.h>
 | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/init.h>
 | 
					
						
							|  |  |  | #include <linux/resource.h>
 | 
					
						
							|  |  |  | #include <linux/serial.h>
 | 
					
						
							|  |  |  | #include <linux/serial_pnx8xxx.h>
 | 
					
						
							|  |  |  | #include <linux/mtd/nand.h>
 | 
					
						
							|  |  |  | #include <linux/mtd/partitions.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_I2C_PNX0105
 | 
					
						
							|  |  |  | /* Until i2c driver available in kernel.*/ | 
					
						
							|  |  |  | #include <linux/i2c-pnx0105.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <irq.h>
 | 
					
						
							|  |  |  | #include <irq-mapping.h>
 | 
					
						
							|  |  |  | #include <pnx833x.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | static u64 uart_dmamask     = DMA_BIT_MASK(32); | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct resource pnx833x_uart_resources[] = { | 
					
						
							|  |  |  | 	[0] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_UART0_PORTS_START, | 
					
						
							|  |  |  | 		.end		= PNX833X_UART0_PORTS_END, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[1] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_PIC_UART0_INT, | 
					
						
							|  |  |  | 		.end		= PNX833X_PIC_UART0_INT, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[2] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_UART1_PORTS_START, | 
					
						
							|  |  |  | 		.end		= PNX833X_UART1_PORTS_END, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[3] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_PIC_UART1_INT, | 
					
						
							|  |  |  | 		.end		= PNX833X_PIC_UART1_INT, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct pnx8xxx_port pnx8xxx_ports[] = { | 
					
						
							|  |  |  | 	[0] = { | 
					
						
							|  |  |  | 		.port   = { | 
					
						
							|  |  |  | 			.type		= PORT_PNX8XXX, | 
					
						
							|  |  |  | 			.iotype		= UPIO_MEM, | 
					
						
							|  |  |  | 			.membase	= (void __iomem *)PNX833X_UART0_PORTS_START, | 
					
						
							|  |  |  | 			.mapbase	= PNX833X_UART0_PORTS_START, | 
					
						
							|  |  |  | 			.irq		= PNX833X_PIC_UART0_INT, | 
					
						
							|  |  |  | 			.uartclk	= 3692300, | 
					
						
							|  |  |  | 			.fifosize	= 16, | 
					
						
							|  |  |  | 			.flags		= UPF_BOOT_AUTOCONF, | 
					
						
							|  |  |  | 			.line		= 0, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[1] = { | 
					
						
							|  |  |  | 		.port   = { | 
					
						
							|  |  |  | 			.type		= PORT_PNX8XXX, | 
					
						
							|  |  |  | 			.iotype		= UPIO_MEM, | 
					
						
							|  |  |  | 			.membase	= (void __iomem *)PNX833X_UART1_PORTS_START, | 
					
						
							|  |  |  | 			.mapbase	= PNX833X_UART1_PORTS_START, | 
					
						
							|  |  |  | 			.irq		= PNX833X_PIC_UART1_INT, | 
					
						
							|  |  |  | 			.uartclk	= 3692300, | 
					
						
							|  |  |  | 			.fifosize	= 16, | 
					
						
							|  |  |  | 			.flags		= UPF_BOOT_AUTOCONF, | 
					
						
							|  |  |  | 			.line		= 1, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_uart_device = { | 
					
						
							|  |  |  | 	.name		= "pnx8xxx-uart", | 
					
						
							|  |  |  | 	.id		= -1, | 
					
						
							|  |  |  | 	.dev = { | 
					
						
							|  |  |  | 		.dma_mask		= &uart_dmamask, | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | 		.coherent_dma_mask	= DMA_BIT_MASK(32), | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 		.platform_data		= pnx8xxx_ports, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.num_resources	= ARRAY_SIZE(pnx833x_uart_resources), | 
					
						
							|  |  |  | 	.resource	= pnx833x_uart_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | static u64 ehci_dmamask     = DMA_BIT_MASK(32); | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct resource pnx833x_usb_ehci_resources[] = { | 
					
						
							|  |  |  | 	[0] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_USB_PORTS_START, | 
					
						
							|  |  |  | 		.end		= PNX833X_USB_PORTS_END, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[1] = { | 
					
						
							|  |  |  | 		.start		= PNX833X_PIC_USB_INT, | 
					
						
							|  |  |  | 		.end		= PNX833X_PIC_USB_INT, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_usb_ehci_device = { | 
					
						
							|  |  |  | 	.name		= "pnx833x-ehci", | 
					
						
							|  |  |  | 	.id		= -1, | 
					
						
							|  |  |  | 	.dev = { | 
					
						
							|  |  |  | 		.dma_mask		= &ehci_dmamask, | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | 		.coherent_dma_mask	= DMA_BIT_MASK(32), | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	.num_resources	= ARRAY_SIZE(pnx833x_usb_ehci_resources), | 
					
						
							|  |  |  | 	.resource	= pnx833x_usb_ehci_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_I2C_PNX0105
 | 
					
						
							|  |  |  | static struct resource pnx833x_i2c0_resources[] = { | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.start		= PNX833X_I2C0_PORTS_START, | 
					
						
							|  |  |  | 		.end		= PNX833X_I2C0_PORTS_END, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.start		= PNX833X_PIC_I2C0_INT, | 
					
						
							|  |  |  | 		.end		= PNX833X_PIC_I2C0_INT, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct resource pnx833x_i2c1_resources[] = { | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.start		= PNX833X_I2C1_PORTS_START, | 
					
						
							|  |  |  | 		.end		= PNX833X_I2C1_PORTS_END, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.start		= PNX833X_PIC_I2C1_INT, | 
					
						
							|  |  |  | 		.end		= PNX833X_PIC_I2C1_INT, | 
					
						
							|  |  |  | 		.flags		= IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct i2c_pnx0105_dev pnx833x_i2c_dev[] = { | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.base = PNX833X_I2C0_PORTS_START, | 
					
						
							|  |  |  | 		.irq = -1, /* should be PNX833X_PIC_I2C0_INT but polling is faster */ | 
					
						
							|  |  |  | 		.clock = 6,	/* 0 == 400 kHz, 4 == 100 kHz(Maximum HDMI), 6 = 50kHz(Prefered HDCP) */ | 
					
						
							|  |  |  | 		.bus_addr = 0,	/* no slave support */ | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		.base = PNX833X_I2C1_PORTS_START, | 
					
						
							|  |  |  | 		.irq = -1,	/* on high freq, polling is faster */ | 
					
						
							|  |  |  | 		/*.irq = PNX833X_PIC_I2C1_INT,*/ | 
					
						
							|  |  |  | 		.clock = 4,	/* 0 == 400 kHz, 4 == 100 kHz. 100 kHz seems a safe default for now */ | 
					
						
							|  |  |  | 		.bus_addr = 0,	/* no slave support */ | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_i2c0_device = { | 
					
						
							|  |  |  | 	.name		= "i2c-pnx0105", | 
					
						
							|  |  |  | 	.id		= 0, | 
					
						
							|  |  |  | 	.dev = { | 
					
						
							|  |  |  | 		.platform_data = &pnx833x_i2c_dev[0], | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.num_resources  = ARRAY_SIZE(pnx833x_i2c0_resources), | 
					
						
							|  |  |  | 	.resource	= pnx833x_i2c0_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_i2c1_device = { | 
					
						
							|  |  |  | 	.name		= "i2c-pnx0105", | 
					
						
							|  |  |  | 	.id		= 1, | 
					
						
							|  |  |  | 	.dev = { | 
					
						
							|  |  |  | 		.platform_data = &pnx833x_i2c_dev[1], | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.num_resources  = ARRAY_SIZE(pnx833x_i2c1_resources), | 
					
						
							|  |  |  | 	.resource	= pnx833x_i2c1_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | static u64 ethernet_dmamask = DMA_BIT_MASK(32); | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct resource pnx833x_ethernet_resources[] = { | 
					
						
							|  |  |  | 	[0] = { | 
					
						
							|  |  |  | 		.start = PNX8335_IP3902_PORTS_START, | 
					
						
							|  |  |  | 		.end   = PNX8335_IP3902_PORTS_END, | 
					
						
							|  |  |  | 		.flags = IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[1] = { | 
					
						
							|  |  |  | 		.start = PNX8335_PIC_ETHERNET_INT, | 
					
						
							|  |  |  | 		.end   = PNX8335_PIC_ETHERNET_INT, | 
					
						
							|  |  |  | 		.flags = IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_ethernet_device = { | 
					
						
							|  |  |  | 	.name = "ip3902-eth", | 
					
						
							|  |  |  | 	.id   = -1, | 
					
						
							|  |  |  | 	.dev  = { | 
					
						
							|  |  |  | 		.dma_mask          = ðernet_dmamask, | 
					
						
							| 
									
										
										
										
											2009-04-06 19:01:15 -07:00
										 |  |  | 		.coherent_dma_mask = DMA_BIT_MASK(32), | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	.num_resources = ARRAY_SIZE(pnx833x_ethernet_resources), | 
					
						
							|  |  |  | 	.resource      = pnx833x_ethernet_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct resource pnx833x_sata_resources[] = { | 
					
						
							|  |  |  | 	[0] = { | 
					
						
							|  |  |  | 		.start = PNX8335_SATA_PORTS_START, | 
					
						
							|  |  |  | 		.end   = PNX8335_SATA_PORTS_END, | 
					
						
							|  |  |  | 		.flags = IORESOURCE_MEM, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	[1] = { | 
					
						
							|  |  |  | 		.start = PNX8335_PIC_SATA_INT, | 
					
						
							|  |  |  | 		.end   = PNX8335_PIC_SATA_INT, | 
					
						
							|  |  |  | 		.flags = IORESOURCE_IRQ, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_sata_device = { | 
					
						
							|  |  |  | 	.name          = "pnx833x-sata", | 
					
						
							|  |  |  | 	.id            = -1, | 
					
						
							|  |  |  | 	.num_resources = ARRAY_SIZE(pnx833x_sata_resources), | 
					
						
							|  |  |  | 	.resource      = pnx833x_sata_resources, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *part_probes[] = { | 
					
						
							|  |  |  | 	"cmdlinepart", | 
					
						
							|  |  |  | 	NULL | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | pnx833x_flash_nand_cmd_ctrl(struct mtd_info *mtd, int cmd, unsigned int ctrl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct nand_chip *this = mtd->priv; | 
					
						
							|  |  |  | 	unsigned long nandaddr = (unsigned long)this->IO_ADDR_W; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (cmd == NAND_CMD_NONE) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ctrl & NAND_CLE) | 
					
						
							|  |  |  | 		writeb(cmd, (void __iomem *)(nandaddr + PNX8335_NAND_CLE_MASK)); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		writeb(cmd, (void __iomem *)(nandaddr + PNX8335_NAND_ALE_MASK)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_nand_data pnx833x_flash_nand_data = { | 
					
						
							|  |  |  | 	.chip = { | 
					
						
							| 
									
										
										
										
											2010-08-12 03:53:54 +01:00
										 |  |  | 		.nr_chips		= 1, | 
					
						
							| 
									
										
										
										
											2008-06-16 15:49:21 +01:00
										 |  |  | 		.chip_delay		= 25, | 
					
						
							|  |  |  | 		.part_probe_types 	= part_probes, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.ctrl = { | 
					
						
							|  |  |  | 		.cmd_ctrl 		= pnx833x_flash_nand_cmd_ctrl | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Set start to be the correct address (PNX8335_NAND_BASE with no 0xb!!), | 
					
						
							|  |  |  |  * 12 bytes more seems to be the standard that allows for NAND access. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct resource pnx833x_flash_nand_resource = { | 
					
						
							|  |  |  | 	.start 	= PNX8335_NAND_BASE, | 
					
						
							|  |  |  | 	.end 	= PNX8335_NAND_BASE + 12, | 
					
						
							|  |  |  | 	.flags 	= IORESOURCE_MEM, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device pnx833x_flash_nand = { | 
					
						
							|  |  |  | 	.name	        = "gen_nand", | 
					
						
							|  |  |  | 	.id		        = -1, | 
					
						
							|  |  |  | 	.num_resources	= 1, | 
					
						
							|  |  |  | 	.resource	    = &pnx833x_flash_nand_resource, | 
					
						
							|  |  |  | 	.dev            = { | 
					
						
							|  |  |  | 		.platform_data = &pnx833x_flash_nand_data, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct platform_device *pnx833x_platform_devices[] __initdata = { | 
					
						
							|  |  |  | 	&pnx833x_uart_device, | 
					
						
							|  |  |  | 	&pnx833x_usb_ehci_device, | 
					
						
							|  |  |  | #ifdef CONFIG_I2C_PNX0105
 | 
					
						
							|  |  |  | 	&pnx833x_i2c0_device, | 
					
						
							|  |  |  | 	&pnx833x_i2c1_device, | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	&pnx833x_ethernet_device, | 
					
						
							|  |  |  | 	&pnx833x_sata_device, | 
					
						
							|  |  |  | 	&pnx833x_flash_nand, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init pnx833x_platform_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = platform_add_devices(pnx833x_platform_devices, | 
					
						
							|  |  |  | 				   ARRAY_SIZE(pnx833x_platform_devices)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | arch_initcall(pnx833x_platform_init); |