320 lines
7.8 KiB
C
320 lines
7.8 KiB
C
|
/*
|
||
|
* 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>
|
||
|
|
||
|
static u64 uart_dmamask = DMA_32BIT_MASK;
|
||
|
|
||
|
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,
|
||
|
.coherent_dma_mask = DMA_32BIT_MASK,
|
||
|
.platform_data = pnx8xxx_ports,
|
||
|
},
|
||
|
.num_resources = ARRAY_SIZE(pnx833x_uart_resources),
|
||
|
.resource = pnx833x_uart_resources,
|
||
|
};
|
||
|
|
||
|
static u64 ehci_dmamask = DMA_32BIT_MASK;
|
||
|
|
||
|
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,
|
||
|
.coherent_dma_mask = DMA_32BIT_MASK,
|
||
|
},
|
||
|
.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
|
||
|
|
||
|
static u64 ethernet_dmamask = DMA_32BIT_MASK;
|
||
|
|
||
|
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,
|
||
|
.coherent_dma_mask = DMA_32BIT_MASK,
|
||
|
},
|
||
|
.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 = {
|
||
|
.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);
|