blob: 483846ff8293b8bf261cd35a576815c366223d1b [file] [log] [blame]
Kevin Wells19d95e12010-07-27 08:44:37 -07001/*
Roland Stiggef5c42272012-04-22 12:01:19 +02002 * Platform support for LPC32xx SoC
Kevin Wells19d95e12010-07-27 08:44:37 -07003 *
4 * Author: Kevin Wells <kevin.wells@nxp.com>
5 *
Roland Stiggef5c42272012-04-22 12:01:19 +02006 * Copyright (C) 2012 Roland Stigge <stigge@antcom.de>
Kevin Wells19d95e12010-07-27 08:44:37 -07007 * Copyright (C) 2010 NXP Semiconductors
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 */
19
20#include <linux/init.h>
21#include <linux/platform_device.h>
Kay Sieversedbaa602011-12-21 16:26:03 -080022#include <linux/device.h>
Kevin Wells19d95e12010-07-27 08:44:37 -070023#include <linux/interrupt.h>
24#include <linux/irq.h>
25#include <linux/dma-mapping.h>
26#include <linux/device.h>
27#include <linux/spi/spi.h>
28#include <linux/spi/eeprom.h>
Kevin Wells19d95e12010-07-27 08:44:37 -070029#include <linux/gpio.h>
30#include <linux/amba/bus.h>
31#include <linux/amba/clcd.h>
32#include <linux/amba/pl022.h>
Roland Stiggef5c42272012-04-22 12:01:19 +020033#include <linux/of.h>
34#include <linux/of_address.h>
35#include <linux/of_irq.h>
36#include <linux/of_platform.h>
37#include <linux/clk.h>
38#include <linux/amba/pl08x.h>
Kevin Wells19d95e12010-07-27 08:44:37 -070039
40#include <asm/setup.h>
41#include <asm/mach-types.h>
42#include <asm/mach/arch.h>
43
44#include <mach/hardware.h>
45#include <mach/platform.h>
Roland Stiggec20b9092012-03-12 22:27:28 +010046#include <mach/board.h>
Linus Walleij9c587c02011-08-22 08:45:15 +010047#include <mach/gpio-lpc32xx.h>
Kevin Wells19d95e12010-07-27 08:44:37 -070048#include "common.h"
49
50/*
51 * Mapped GPIOLIB GPIOs
52 */
53#define SPI0_CS_GPIO LPC32XX_GPIO(LPC32XX_GPIO_P3_GRP, 5)
54#define LCD_POWER_GPIO LPC32XX_GPIO(LPC32XX_GPO_P3_GRP, 0)
55#define BKL_POWER_GPIO LPC32XX_GPIO(LPC32XX_GPO_P3_GRP, 4)
Kevin Wells19d95e12010-07-27 08:44:37 -070056
57/*
58 * AMBA LCD controller
59 */
60static struct clcd_panel conn_lcd_panel = {
61 .mode = {
62 .name = "QVGA portrait",
63 .refresh = 60,
64 .xres = 240,
65 .yres = 320,
66 .pixclock = 191828,
67 .left_margin = 22,
68 .right_margin = 11,
69 .upper_margin = 2,
70 .lower_margin = 1,
71 .hsync_len = 5,
72 .vsync_len = 2,
73 .sync = 0,
74 .vmode = FB_VMODE_NONINTERLACED,
75 },
76 .width = -1,
77 .height = -1,
78 .tim2 = (TIM2_IVS | TIM2_IHS),
79 .cntl = (CNTL_BGR | CNTL_LCDTFT | CNTL_LCDVCOMP(1) |
80 CNTL_LCDBPP16_565),
81 .bpp = 16,
82};
83#define PANEL_SIZE (3 * SZ_64K)
84
85static int lpc32xx_clcd_setup(struct clcd_fb *fb)
86{
87 dma_addr_t dma;
88
89 fb->fb.screen_base = dma_alloc_writecombine(&fb->dev->dev,
90 PANEL_SIZE, &dma, GFP_KERNEL);
91 if (!fb->fb.screen_base) {
92 printk(KERN_ERR "CLCD: unable to map framebuffer\n");
93 return -ENOMEM;
94 }
95
96 fb->fb.fix.smem_start = dma;
97 fb->fb.fix.smem_len = PANEL_SIZE;
98 fb->panel = &conn_lcd_panel;
99
100 if (gpio_request(LCD_POWER_GPIO, "LCD power"))
101 printk(KERN_ERR "Error requesting gpio %u",
102 LCD_POWER_GPIO);
103 else if (gpio_direction_output(LCD_POWER_GPIO, 1))
104 printk(KERN_ERR "Error setting gpio %u to output",
105 LCD_POWER_GPIO);
106
107 if (gpio_request(BKL_POWER_GPIO, "LCD backlight power"))
108 printk(KERN_ERR "Error requesting gpio %u",
109 BKL_POWER_GPIO);
110 else if (gpio_direction_output(BKL_POWER_GPIO, 1))
111 printk(KERN_ERR "Error setting gpio %u to output",
112 BKL_POWER_GPIO);
113
114 return 0;
115}
116
117static int lpc32xx_clcd_mmap(struct clcd_fb *fb, struct vm_area_struct *vma)
118{
119 return dma_mmap_writecombine(&fb->dev->dev, vma,
120 fb->fb.screen_base, fb->fb.fix.smem_start,
121 fb->fb.fix.smem_len);
122}
123
124static void lpc32xx_clcd_remove(struct clcd_fb *fb)
125{
126 dma_free_writecombine(&fb->dev->dev, fb->fb.fix.smem_len,
127 fb->fb.screen_base, fb->fb.fix.smem_start);
128}
129
130/*
131 * On some early LCD modules (1307.0), the backlight logic is inverted.
132 * For those board variants, swap the disable and enable states for
133 * BKL_POWER_GPIO.
134*/
135static void clcd_disable(struct clcd_fb *fb)
136{
137 gpio_set_value(BKL_POWER_GPIO, 0);
138 gpio_set_value(LCD_POWER_GPIO, 0);
139}
140
141static void clcd_enable(struct clcd_fb *fb)
142{
143 gpio_set_value(BKL_POWER_GPIO, 1);
144 gpio_set_value(LCD_POWER_GPIO, 1);
145}
146
147static struct clcd_board lpc32xx_clcd_data = {
148 .name = "Phytec LCD",
149 .check = clcdfb_check,
150 .decode = clcdfb_decode,
151 .disable = clcd_disable,
152 .enable = clcd_enable,
153 .setup = lpc32xx_clcd_setup,
154 .mmap = lpc32xx_clcd_mmap,
155 .remove = lpc32xx_clcd_remove,
156};
157
Kevin Wells19d95e12010-07-27 08:44:37 -0700158/*
159 * AMBA SSP (SPI)
160 */
161static void phy3250_spi_cs_set(u32 control)
162{
163 gpio_set_value(SPI0_CS_GPIO, (int) control);
164}
165
166static struct pl022_config_chip spi0_chip_info = {
Kevin Wells19d95e12010-07-27 08:44:37 -0700167 .com_mode = INTERRUPT_TRANSFER,
168 .iface = SSP_INTERFACE_MOTOROLA_SPI,
169 .hierarchy = SSP_MASTER,
170 .slave_tx_disable = 0,
Kevin Wells19d95e12010-07-27 08:44:37 -0700171 .rx_lev_trig = SSP_RX_4_OR_MORE_ELEM,
172 .tx_lev_trig = SSP_TX_4_OR_MORE_EMPTY_LOC,
Kevin Wells19d95e12010-07-27 08:44:37 -0700173 .ctrl_len = SSP_BITS_8,
174 .wait_state = SSP_MWIRE_WAIT_ZERO,
175 .duplex = SSP_MICROWIRE_CHANNEL_FULL_DUPLEX,
176 .cs_control = phy3250_spi_cs_set,
177};
178
179static struct pl022_ssp_controller lpc32xx_ssp0_data = {
180 .bus_id = 0,
181 .num_chipselect = 1,
182 .enable_dma = 0,
183};
184
Roland Stiggef5c42272012-04-22 12:01:19 +0200185static struct pl022_ssp_controller lpc32xx_ssp1_data = {
186 .bus_id = 1,
187 .num_chipselect = 1,
188 .enable_dma = 0,
189};
Kevin Wells19d95e12010-07-27 08:44:37 -0700190
191/* AT25 driver registration */
192static int __init phy3250_spi_board_register(void)
193{
194#if defined(CONFIG_SPI_SPIDEV) || defined(CONFIG_SPI_SPIDEV_MODULE)
195 static struct spi_board_info info[] = {
196 {
197 .modalias = "spidev",
198 .max_speed_hz = 5000000,
199 .bus_num = 0,
200 .chip_select = 0,
201 .controller_data = &spi0_chip_info,
202 },
203 };
204
205#else
206 static struct spi_eeprom eeprom = {
207 .name = "at25256a",
208 .byte_len = 0x8000,
209 .page_size = 64,
210 .flags = EE_ADDR2,
211 };
212
213 static struct spi_board_info info[] = {
214 {
215 .modalias = "at25",
216 .max_speed_hz = 5000000,
217 .bus_num = 0,
218 .chip_select = 0,
Kevin Wellsbde435a2010-09-16 06:18:50 -0700219 .mode = SPI_MODE_0,
Kevin Wells19d95e12010-07-27 08:44:37 -0700220 .platform_data = &eeprom,
221 .controller_data = &spi0_chip_info,
222 },
223 };
224#endif
225 return spi_register_board_info(info, ARRAY_SIZE(info));
226}
227arch_initcall(phy3250_spi_board_register);
228
Roland Stiggef5c42272012-04-22 12:01:19 +0200229static struct pl08x_platform_data pl08x_pd = {
Kevin Wells19d95e12010-07-27 08:44:37 -0700230};
231
Roland Stiggef5c42272012-04-22 12:01:19 +0200232static const struct of_dev_auxdata lpc32xx_auxdata_lookup[] __initconst = {
233 OF_DEV_AUXDATA("arm,pl022", 0x20084000, "dev:ssp0", &lpc32xx_ssp0_data),
234 OF_DEV_AUXDATA("arm,pl022", 0x2008C000, "dev:ssp1", &lpc32xx_ssp1_data),
235 OF_DEV_AUXDATA("arm,pl110", 0x31040000, "dev:clcd", &lpc32xx_clcd_data),
236 OF_DEV_AUXDATA("arm,pl080", 0x31000000, "pl08xdmac", &pl08x_pd),
237 { }
Kevin Wells19d95e12010-07-27 08:44:37 -0700238};
239
Roland Stiggef5c42272012-04-22 12:01:19 +0200240static void __init lpc3250_machine_init(void)
Kevin Wells19d95e12010-07-27 08:44:37 -0700241{
242 u32 tmp;
Kevin Wells19d95e12010-07-27 08:44:37 -0700243
244 /* Setup SLC NAND controller muxing */
245 __raw_writel(LPC32XX_CLKPWR_NANDCLK_SEL_SLC,
246 LPC32XX_CLKPWR_NAND_CLK_CTRL);
247
248 /* Setup LCD muxing to RGB565 */
249 tmp = __raw_readl(LPC32XX_CLKPWR_LCDCLK_CTRL) &
250 ~(LPC32XX_CLKPWR_LCDCTRL_LCDTYPE_MSK |
251 LPC32XX_CLKPWR_LCDCTRL_PSCALE_MSK);
252 tmp |= LPC32XX_CLKPWR_LCDCTRL_LCDTYPE_TFT16;
253 __raw_writel(tmp, LPC32XX_CLKPWR_LCDCLK_CTRL);
254
Roland Stiggef5c42272012-04-22 12:01:19 +0200255 /* Set up USB power */
256 tmp = __raw_readl(LPC32XX_CLKPWR_USB_CTRL);
257 tmp |= LPC32XX_CLKPWR_USBCTRL_HCLK_EN |
258 LPC32XX_CLKPWR_USBCTRL_USBI2C_EN;
259 __raw_writel(tmp, LPC32XX_CLKPWR_USB_CTRL);
260
Kevin Wells19d95e12010-07-27 08:44:37 -0700261 /* Set up I2C pull levels */
262 tmp = __raw_readl(LPC32XX_CLKPWR_I2C_CLK_CTRL);
263 tmp |= LPC32XX_CLKPWR_I2CCLK_USBI2CHI_DRIVE |
264 LPC32XX_CLKPWR_I2CCLK_I2C2HI_DRIVE;
265 __raw_writel(tmp, LPC32XX_CLKPWR_I2C_CLK_CTRL);
266
267 /* Disable IrDA pulsing support on UART6 */
268 tmp = __raw_readl(LPC32XX_UARTCTL_CTRL);
269 tmp |= LPC32XX_UART_UART6_IRDAMOD_BYPASS;
270 __raw_writel(tmp, LPC32XX_UARTCTL_CTRL);
271
272 /* Enable DMA for I2S1 channel */
273 tmp = __raw_readl(LPC32XX_CLKPWR_I2S_CLK_CTRL);
274 tmp = LPC32XX_CLKPWR_I2SCTRL_I2S1_USE_DMA;
275 __raw_writel(tmp, LPC32XX_CLKPWR_I2S_CLK_CTRL);
276
277 lpc32xx_serial_init();
278
279 /*
280 * AMBA peripheral clocks need to be enabled prior to AMBA device
281 * detection or a data fault will occur, so enable the clocks
Roland Stiggef5c42272012-04-22 12:01:19 +0200282 * here.
Kevin Wells19d95e12010-07-27 08:44:37 -0700283 */
Kevin Wells19d95e12010-07-27 08:44:37 -0700284 tmp = __raw_readl(LPC32XX_CLKPWR_LCDCLK_CTRL);
285 __raw_writel((tmp | LPC32XX_CLKPWR_LCDCTRL_CLK_EN),
286 LPC32XX_CLKPWR_LCDCLK_CTRL);
Roland Stiggef5c42272012-04-22 12:01:19 +0200287
Kevin Wells19d95e12010-07-27 08:44:37 -0700288 tmp = __raw_readl(LPC32XX_CLKPWR_SSP_CLK_CTRL);
289 __raw_writel((tmp | LPC32XX_CLKPWR_SSPCTRL_SSPCLK0_EN),
290 LPC32XX_CLKPWR_SSP_CLK_CTRL);
Kevin Wells19d95e12010-07-27 08:44:37 -0700291
Roland Stiggef5c42272012-04-22 12:01:19 +0200292 tmp = __raw_readl(LPC32XX_CLKPWR_DMA_CLK_CTRL);
293 __raw_writel((tmp | LPC32XX_CLKPWR_DMACLKCTRL_CLK_EN),
294 LPC32XX_CLKPWR_DMA_CLK_CTRL);
Kevin Wells19d95e12010-07-27 08:44:37 -0700295
296 /* Test clock needed for UDA1380 initial init */
297 __raw_writel(LPC32XX_CLKPWR_TESTCLK2_SEL_MOSC |
298 LPC32XX_CLKPWR_TESTCLK_TESTCLK2_EN,
299 LPC32XX_CLKPWR_TEST_CLK_SEL);
300
Roland Stiggef5c42272012-04-22 12:01:19 +0200301 of_platform_populate(NULL, of_default_bus_match_table,
302 lpc32xx_auxdata_lookup, NULL);
303
304 /* Register GPIOs used on this board */
305 if (gpio_request(SPI0_CS_GPIO, "spi0 cs"))
306 printk(KERN_ERR "Error requesting gpio %u",
307 SPI0_CS_GPIO);
308 else if (gpio_direction_output(SPI0_CS_GPIO, 1))
309 printk(KERN_ERR "Error setting gpio %u to output",
310 SPI0_CS_GPIO);
Kevin Wells19d95e12010-07-27 08:44:37 -0700311}
312
313static int __init lpc32xx_display_uid(void)
314{
315 u32 uid[4];
316
317 lpc32xx_get_uid(uid);
318
319 printk(KERN_INFO "LPC32XX unique ID: %08x%08x%08x%08x\n",
320 uid[3], uid[2], uid[1], uid[0]);
321
322 return 1;
323}
324arch_initcall(lpc32xx_display_uid);
325
Roland Stiggef5c42272012-04-22 12:01:19 +0200326static char const *lpc32xx_dt_compat[] __initdata = {
327 "nxp,lpc3220",
328 "nxp,lpc3230",
329 "nxp,lpc3240",
330 "nxp,lpc3250",
331 NULL
332};
333
334DT_MACHINE_START(LPC32XX_DT, "LPC32XX SoC (Flattened Device Tree)")
Nicolas Pitrebdec5dd2011-07-05 22:38:14 -0400335 .atag_offset = 0x100,
Kevin Wells19d95e12010-07-27 08:44:37 -0700336 .map_io = lpc32xx_map_io,
337 .init_irq = lpc32xx_init_irq,
338 .timer = &lpc32xx_timer,
Roland Stiggef5c42272012-04-22 12:01:19 +0200339 .init_machine = lpc3250_machine_init,
340 .dt_compat = lpc32xx_dt_compat,
Russell Kingb23fcd92011-11-05 12:17:40 +0000341 .restart = lpc23xx_restart,
Kevin Wells19d95e12010-07-27 08:44:37 -0700342MACHINE_END