blob: 9f1a214626201bf2a969eefe60081b24dfa521be [file] [log] [blame]
Ben Dooks5718df92008-10-21 14:07:09 +01001/* linux/arch/arm/mach-s3c6410/mach-smdk6410.c
2 *
3 * Copyright 2008 Openmoko, Inc.
4 * Copyright 2008 Simtec Electronics
5 * Ben Dooks <ben@simtec.co.uk>
6 * http://armlinux.simtec.co.uk/
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 *
12*/
13
14#include <linux/kernel.h>
15#include <linux/types.h>
16#include <linux/interrupt.h>
17#include <linux/list.h>
18#include <linux/timer.h>
19#include <linux/init.h>
20#include <linux/serial_core.h>
21#include <linux/platform_device.h>
22#include <linux/io.h>
Ben Dooks096941e2008-10-31 16:14:59 +000023#include <linux/i2c.h>
Ben Dooks438a5d42008-11-19 15:41:34 +000024#include <linux/fb.h>
25#include <linux/gpio.h>
26#include <linux/delay.h>
Mark Brown3056ea02009-01-27 16:18:01 +000027#include <linux/smsc911x.h>
Ben Dooks438a5d42008-11-19 15:41:34 +000028
Mark Brownecc558a2009-02-17 15:59:38 +000029#ifdef CONFIG_SMDK6410_WM1190_EV1
30#include <linux/mfd/wm8350/core.h>
31#include <linux/mfd/wm8350/pmic.h>
32#endif
Ben Dooks438a5d42008-11-19 15:41:34 +000033
34#include <video/platform_lcd.h>
Ben Dooks5718df92008-10-21 14:07:09 +010035
36#include <asm/mach/arch.h>
37#include <asm/mach/map.h>
38#include <asm/mach/irq.h>
39
40#include <mach/hardware.h>
Ben Dooks438a5d42008-11-19 15:41:34 +000041#include <mach/regs-fb.h>
Ben Dooks5718df92008-10-21 14:07:09 +010042#include <mach/map.h>
43
44#include <asm/irq.h>
45#include <asm/mach-types.h>
46
47#include <plat/regs-serial.h>
Ben Dooksd6662c32008-12-12 00:24:40 +000048#include <plat/regs-modem.h>
49#include <plat/regs-gpio.h>
50#include <plat/regs-sys.h>
Ben Dooksd85fa242008-10-31 16:14:52 +000051#include <plat/iic.h>
Ben Dooks438a5d42008-11-19 15:41:34 +000052#include <plat/fb.h>
Mark Brown3056ea02009-01-27 16:18:01 +000053#include <plat/gpio-cfg.h>
Ben Dooks5718df92008-10-21 14:07:09 +010054
55#include <plat/s3c6410.h>
56#include <plat/clock.h>
57#include <plat/devs.h>
58#include <plat/cpu.h>
59
60#define UCON S3C2410_UCON_DEFAULT | S3C2410_UCON_UCLK
61#define ULCON S3C2410_LCON_CS8 | S3C2410_LCON_PNONE | S3C2410_LCON_STOPB
62#define UFCON S3C2410_UFCON_RXTRIG8 | S3C2410_UFCON_FIFOMODE
63
64static struct s3c2410_uartcfg smdk6410_uartcfgs[] __initdata = {
65 [0] = {
66 .hwport = 0,
67 .flags = 0,
Matt Hsubd258e52009-06-29 19:03:41 +080068 .ucon = UCON,
69 .ulcon = ULCON,
70 .ufcon = UFCON,
Ben Dooks5718df92008-10-21 14:07:09 +010071 },
72 [1] = {
73 .hwport = 1,
74 .flags = 0,
Matt Hsubd258e52009-06-29 19:03:41 +080075 .ucon = UCON,
76 .ulcon = ULCON,
77 .ufcon = UFCON,
78 },
79 [2] = {
80 .hwport = 2,
81 .flags = 0,
82 .ucon = UCON,
83 .ulcon = ULCON,
84 .ufcon = UFCON,
85 },
86 [3] = {
87 .hwport = 3,
88 .flags = 0,
89 .ucon = UCON,
90 .ulcon = ULCON,
91 .ufcon = UFCON,
Ben Dooks5718df92008-10-21 14:07:09 +010092 },
93};
94
Ben Dooks438a5d42008-11-19 15:41:34 +000095/* framebuffer and LCD setup. */
96
97/* GPF15 = LCD backlight control
98 * GPF13 => Panel power
99 * GPN5 = LCD nRESET signal
100 * PWM_TOUT1 => backlight brightness
101 */
102
103static void smdk6410_lcd_power_set(struct plat_lcd_data *pd,
104 unsigned int power)
105{
106 if (power) {
107 gpio_direction_output(S3C64XX_GPF(13), 1);
108 gpio_direction_output(S3C64XX_GPF(15), 1);
109
110 /* fire nRESET on power up */
111 gpio_direction_output(S3C64XX_GPN(5), 0);
112 msleep(10);
113 gpio_direction_output(S3C64XX_GPN(5), 1);
114 msleep(1);
115 } else {
116 gpio_direction_output(S3C64XX_GPF(15), 0);
117 gpio_direction_output(S3C64XX_GPF(13), 0);
118 }
119}
120
121static struct plat_lcd_data smdk6410_lcd_power_data = {
122 .set_power = smdk6410_lcd_power_set,
123};
124
125static struct platform_device smdk6410_lcd_powerdev = {
126 .name = "platform-lcd",
127 .dev.parent = &s3c_device_fb.dev,
128 .dev.platform_data = &smdk6410_lcd_power_data,
129};
130
131static struct s3c_fb_pd_win smdk6410_fb_win0 = {
132 /* this is to ensure we use win0 */
133 .win_mode = {
134 .pixclock = 41094,
135 .left_margin = 8,
136 .right_margin = 13,
137 .upper_margin = 7,
138 .lower_margin = 5,
139 .hsync_len = 3,
140 .vsync_len = 1,
141 .xres = 800,
142 .yres = 480,
143 },
144 .max_bpp = 32,
145 .default_bpp = 16,
146};
147
148/* 405566 clocks per frame => 60Hz refresh requires 24333960Hz clock */
149static struct s3c_fb_platdata smdk6410_lcd_pdata __initdata = {
150 .setup_gpio = s3c64xx_fb_gpio_setup_24bpp,
151 .win[0] = &smdk6410_fb_win0,
152 .vidcon0 = VIDCON0_VIDOUT_RGB | VIDCON0_PNRMODE_RGB,
153 .vidcon1 = VIDCON1_INV_HSYNC | VIDCON1_INV_VSYNC,
154};
155
Mark Brown3056ea02009-01-27 16:18:01 +0000156static struct resource smdk6410_smsc911x_resources[] = {
157 [0] = {
158 .start = 0x18000000,
159 .end = 0x18000000 + SZ_64K - 1,
160 .flags = IORESOURCE_MEM,
161 },
162 [1] = {
163 .start = S3C_EINT(10),
164 .end = S3C_EINT(10),
165 .flags = IORESOURCE_IRQ | IRQ_TYPE_LEVEL_LOW,
166 },
167};
168
169static struct smsc911x_platform_config smdk6410_smsc911x_pdata = {
170 .irq_polarity = SMSC911X_IRQ_POLARITY_ACTIVE_LOW,
171 .irq_type = SMSC911X_IRQ_TYPE_OPEN_DRAIN,
172 .flags = SMSC911X_USE_32BIT | SMSC911X_FORCE_INTERNAL_PHY,
173 .phy_interface = PHY_INTERFACE_MODE_MII,
174};
175
176
177static struct platform_device smdk6410_smsc911x = {
178 .name = "smsc911x",
179 .id = -1,
180 .num_resources = ARRAY_SIZE(smdk6410_smsc911x_resources),
181 .resource = &smdk6410_smsc911x_resources[0],
182 .dev = {
183 .platform_data = &smdk6410_smsc911x_pdata,
184 },
185};
186
Mark Brown027191a2009-01-23 16:29:43 +0000187static struct map_desc smdk6410_iodesc[] = {};
Ben Dooks5718df92008-10-21 14:07:09 +0100188
189static struct platform_device *smdk6410_devices[] __initdata = {
Ben Dooksb24636c2008-11-03 20:14:53 +0000190#ifdef CONFIG_SMDK6410_SD_CH0
Ben Dooks39057f22008-10-31 16:14:29 +0000191 &s3c_device_hsmmc0,
Ben Dooksb24636c2008-11-03 20:14:53 +0000192#endif
193#ifdef CONFIG_SMDK6410_SD_CH1
194 &s3c_device_hsmmc1,
195#endif
Ben Dooksd85fa242008-10-31 16:14:52 +0000196 &s3c_device_i2c0,
Ben Dooksd7ea3742008-10-31 16:14:57 +0000197 &s3c_device_i2c1,
Ben Dooks438a5d42008-11-19 15:41:34 +0000198 &s3c_device_fb,
Ben Dooks98fd63b2009-03-06 19:49:49 +0000199 &s3c_device_usb,
Ben Dooks06fa1d32009-05-16 22:11:20 +0100200 &s3c_device_usb_hsotg,
Ben Dooks438a5d42008-11-19 15:41:34 +0000201 &smdk6410_lcd_powerdev,
Mark Brown3056ea02009-01-27 16:18:01 +0000202
203 &smdk6410_smsc911x,
Ben Dooks5718df92008-10-21 14:07:09 +0100204};
205
Mark Brownecc558a2009-02-17 15:59:38 +0000206#ifdef CONFIG_SMDK6410_WM1190_EV1
207/* S3C64xx internal logic & PLL */
208static struct regulator_init_data wm8350_dcdc1_data = {
209 .constraints = {
210 .name = "PVDD_INT/PVDD_PLL",
211 .min_uV = 1200000,
212 .max_uV = 1200000,
213 .always_on = 1,
214 .apply_uV = 1,
215 },
216};
217
218/* Memory */
219static struct regulator_init_data wm8350_dcdc3_data = {
220 .constraints = {
221 .name = "PVDD_MEM",
222 .min_uV = 1800000,
223 .max_uV = 1800000,
224 .always_on = 1,
225 .state_mem = {
226 .uV = 1800000,
227 .mode = REGULATOR_MODE_NORMAL,
228 .enabled = 1,
229 },
230 .initial_state = PM_SUSPEND_MEM,
231 },
232};
233
234/* USB, EXT, PCM, ADC/DAC, USB, MMC */
235static struct regulator_init_data wm8350_dcdc4_data = {
236 .constraints = {
237 .name = "PVDD_HI/PVDD_EXT/PVDD_SYS/PVCCM2MTV",
238 .min_uV = 3000000,
239 .max_uV = 3000000,
240 .always_on = 1,
241 },
242};
243
244/* ARM core */
Mark Browne3980b62009-04-15 17:11:53 +0100245static struct regulator_consumer_supply dcdc6_consumers[] = {
246 {
247 .supply = "vddarm",
248 }
249};
250
Mark Brownecc558a2009-02-17 15:59:38 +0000251static struct regulator_init_data wm8350_dcdc6_data = {
252 .constraints = {
253 .name = "PVDD_ARM",
254 .min_uV = 1000000,
255 .max_uV = 1300000,
256 .always_on = 1,
Mark Browne3980b62009-04-15 17:11:53 +0100257 .valid_ops_mask = REGULATOR_CHANGE_VOLTAGE,
Mark Brownecc558a2009-02-17 15:59:38 +0000258 },
Mark Browne3980b62009-04-15 17:11:53 +0100259 .num_consumer_supplies = ARRAY_SIZE(dcdc6_consumers),
260 .consumer_supplies = dcdc6_consumers,
Mark Brownecc558a2009-02-17 15:59:38 +0000261};
262
263/* Alive */
264static struct regulator_init_data wm8350_ldo1_data = {
265 .constraints = {
266 .name = "PVDD_ALIVE",
267 .min_uV = 1200000,
268 .max_uV = 1200000,
269 .always_on = 1,
270 .apply_uV = 1,
271 },
272};
273
274/* OTG */
275static struct regulator_init_data wm8350_ldo2_data = {
276 .constraints = {
277 .name = "PVDD_OTG",
278 .min_uV = 3300000,
279 .max_uV = 3300000,
Mark Brownf53aee22009-04-09 16:30:40 +0100280 .always_on = 1,
Mark Brownecc558a2009-02-17 15:59:38 +0000281 },
282};
283
284/* LCD */
285static struct regulator_init_data wm8350_ldo3_data = {
286 .constraints = {
287 .name = "PVDD_LCD",
288 .min_uV = 3000000,
289 .max_uV = 3000000,
Mark Brownf53aee22009-04-09 16:30:40 +0100290 .always_on = 1,
Mark Brownecc558a2009-02-17 15:59:38 +0000291 },
292};
293
294/* OTGi/1190-EV1 HPVDD & AVDD */
295static struct regulator_init_data wm8350_ldo4_data = {
296 .constraints = {
297 .name = "PVDD_OTGI/HPVDD/AVDD",
298 .min_uV = 1200000,
299 .max_uV = 1200000,
300 .apply_uV = 1,
Mark Brownf53aee22009-04-09 16:30:40 +0100301 .always_on = 1,
Mark Brownecc558a2009-02-17 15:59:38 +0000302 },
303};
304
305static struct {
306 int regulator;
307 struct regulator_init_data *initdata;
308} wm1190_regulators[] = {
309 { WM8350_DCDC_1, &wm8350_dcdc1_data },
310 { WM8350_DCDC_3, &wm8350_dcdc3_data },
311 { WM8350_DCDC_4, &wm8350_dcdc4_data },
312 { WM8350_DCDC_6, &wm8350_dcdc6_data },
313 { WM8350_LDO_1, &wm8350_ldo1_data },
314 { WM8350_LDO_2, &wm8350_ldo2_data },
315 { WM8350_LDO_3, &wm8350_ldo3_data },
316 { WM8350_LDO_4, &wm8350_ldo4_data },
317};
318
319static int __init smdk6410_wm8350_init(struct wm8350 *wm8350)
320{
321 int i;
322
Mark Browna3323b72009-11-03 14:42:04 +0000323 /* Configure the IRQ line */
324 s3c_gpio_setpull(S3C64XX_GPN(12), S3C_GPIO_PULL_UP);
325
Mark Brownecc558a2009-02-17 15:59:38 +0000326 /* Instantiate the regulators */
327 for (i = 0; i < ARRAY_SIZE(wm1190_regulators); i++)
328 wm8350_register_regulator(wm8350,
329 wm1190_regulators[i].regulator,
330 wm1190_regulators[i].initdata);
331
332 return 0;
333}
334
335static struct wm8350_platform_data __initdata smdk6410_wm8350_pdata = {
336 .init = smdk6410_wm8350_init,
Mark Browndb9256f2009-04-09 19:00:19 +0100337 .irq_high = 1,
Mark Brownecc558a2009-02-17 15:59:38 +0000338};
339#endif
340
Ben Dooks096941e2008-10-31 16:14:59 +0000341static struct i2c_board_info i2c_devs0[] __initdata = {
342 { I2C_BOARD_INFO("24c08", 0x50), },
Mark Brown77897472009-01-23 16:29:41 +0000343 { I2C_BOARD_INFO("wm8580", 0x1b), },
Mark Brownecc558a2009-02-17 15:59:38 +0000344
345#ifdef CONFIG_SMDK6410_WM1190_EV1
346 { I2C_BOARD_INFO("wm8350", 0x1a),
347 .platform_data = &smdk6410_wm8350_pdata,
348 .irq = S3C_EINT(12),
349 },
350#endif
Ben Dooks096941e2008-10-31 16:14:59 +0000351};
352
353static struct i2c_board_info i2c_devs1[] __initdata = {
354 { I2C_BOARD_INFO("24c128", 0x57), }, /* Samsung S524AD0XD1 */
Ben Dooks5718df92008-10-21 14:07:09 +0100355};
356
Ben Dooks5718df92008-10-21 14:07:09 +0100357static void __init smdk6410_map_io(void)
358{
Ben Dooksd6662c32008-12-12 00:24:40 +0000359 u32 tmp;
360
Ben Dooks5718df92008-10-21 14:07:09 +0100361 s3c64xx_init_io(smdk6410_iodesc, ARRAY_SIZE(smdk6410_iodesc));
362 s3c24xx_init_clocks(12000000);
363 s3c24xx_init_uarts(smdk6410_uartcfgs, ARRAY_SIZE(smdk6410_uartcfgs));
Ben Dooksd6662c32008-12-12 00:24:40 +0000364
365 /* set the LCD type */
366
367 tmp = __raw_readl(S3C64XX_SPCON);
368 tmp &= ~S3C64XX_SPCON_LCD_SEL_MASK;
369 tmp |= S3C64XX_SPCON_LCD_SEL_RGB;
370 __raw_writel(tmp, S3C64XX_SPCON);
371
372 /* remove the lcd bypass */
373 tmp = __raw_readl(S3C64XX_MODEM_MIFPCON);
374 tmp &= ~MIFPCON_LCD_BYPASS;
375 __raw_writel(tmp, S3C64XX_MODEM_MIFPCON);
Ben Dooks5718df92008-10-21 14:07:09 +0100376}
377
378static void __init smdk6410_machine_init(void)
379{
Ben Dooksd85fa242008-10-31 16:14:52 +0000380 s3c_i2c0_set_platdata(NULL);
Ben Dooksd7ea3742008-10-31 16:14:57 +0000381 s3c_i2c1_set_platdata(NULL);
Ben Dooks438a5d42008-11-19 15:41:34 +0000382 s3c_fb_set_platdata(&smdk6410_lcd_pdata);
Ben Dooks096941e2008-10-31 16:14:59 +0000383
Mark Brownb7f9a942009-04-08 16:12:35 +0100384 gpio_request(S3C64XX_GPN(5), "LCD power");
385 gpio_request(S3C64XX_GPF(13), "LCD power");
386 gpio_request(S3C64XX_GPF(15), "LCD power");
387
Ben Dooks096941e2008-10-31 16:14:59 +0000388 i2c_register_board_info(0, i2c_devs0, ARRAY_SIZE(i2c_devs0));
389 i2c_register_board_info(1, i2c_devs1, ARRAY_SIZE(i2c_devs1));
390
Ben Dooks5718df92008-10-21 14:07:09 +0100391 platform_add_devices(smdk6410_devices, ARRAY_SIZE(smdk6410_devices));
392}
393
394MACHINE_START(SMDK6410, "SMDK6410")
395 /* Maintainer: Ben Dooks <ben@fluff.org> */
396 .phys_io = S3C_PA_UART & 0xfff00000,
397 .io_pg_offst = (((u32)S3C_VA_UART) >> 18) & 0xfffc,
398 .boot_params = S3C64XX_PA_SDRAM + 0x100,
399
400 .init_irq = s3c6410_init_irq,
401 .map_io = smdk6410_map_io,
402 .init_machine = smdk6410_machine_init,
403 .timer = &s3c24xx_timer,
404MACHINE_END