2017-12-25 20:54:33 +01:00
// SPDX-License-Identifier: GPL-2.0
//
// Copyright 2010 Darius Augulis <augulis.darius@gmail.com>
// Copyright 2008 Openmoko, Inc.
// Copyright 2008 Simtec Electronics
// Ben Dooks <ben@simtec.co.uk>
// http://armlinux.simtec.co.uk/
2010-07-28 23:03:43 +03:00
2010-10-19 13:52:29 +09:00
# include <linux/init.h>
2010-07-28 23:03:43 +03:00
# include <linux/interrupt.h>
2010-10-19 13:52:29 +09:00
# include <linux/fb.h>
# include <linux/gpio.h>
# include <linux/kernel.h>
2010-07-28 23:03:43 +03:00
# include <linux/list.h>
2010-08-01 23:38:43 +03:00
# include <linux/dm9000.h>
2010-10-19 13:52:32 +09:00
# include <linux/mtd/mtd.h>
# include <linux/mtd/partitions.h>
2010-07-28 23:03:43 +03:00
# include <linux/platform_device.h>
2010-10-19 13:52:29 +09:00
# include <linux/serial_core.h>
2014-02-14 10:32:45 +09:00
# include <linux/serial_s3c.h>
2010-10-19 13:52:29 +09:00
# include <linux/types.h>
2010-07-28 23:03:43 +03:00
# include <asm/mach-types.h>
# include <asm/mach/arch.h>
# include <asm/mach/map.h>
2010-10-19 13:52:29 +09:00
2010-07-28 23:03:43 +03:00
# include <mach/map.h>
2010-10-19 13:52:29 +09:00
# include <mach/regs-gpio.h>
2014-01-14 14:24:24 +01:00
# include <mach/gpio-samsung.h>
2015-02-27 22:06:58 +01:00
# include <mach/irqs.h>
2010-10-19 13:52:29 +09:00
2010-10-19 13:52:34 +09:00
# include <plat/adc.h>
2010-07-28 23:03:43 +03:00
# include <plat/cpu.h>
2010-08-02 01:39:25 +03:00
# include <plat/devs.h>
2010-10-19 13:52:29 +09:00
# include <plat/fb.h>
2012-08-24 15:22:12 +02:00
# include <linux/platform_data/mtd-nand-s3c2410.h>
# include <linux/platform_data/touchscreen-s3c2410.h>
2010-07-28 23:03:43 +03:00
2010-10-19 13:52:29 +09:00
# include <video/platform_lcd.h>
2012-08-08 09:44:49 +09:00
# include <video/samsung_fimd.h>
2013-01-09 18:47:04 -08:00
# include <plat/samsung-time.h>
2010-07-28 23:03:43 +03:00
2011-12-22 23:27:42 +01:00
# include "common.h"
2013-01-02 13:24:12 -08:00
# include "regs-modem.h"
2013-01-02 13:26:25 -08:00
# include "regs-srom.h"
2011-12-22 23:27:42 +01:00
2010-11-16 18:08:50 +09:00
# define UCON S3C2410_UCON_DEFAULT
2010-09-09 21:40:22 +09:00
# define ULCON (S3C2410_LCON_CS8 | S3C2410_LCON_PNONE | S3C2410_LCON_STOPB)
# define UFCON (S3C2410_UFCON_RXTRIG8 | S3C2410_UFCON_FIFOMODE)
2010-07-28 23:03:43 +03:00
static struct s3c2410_uartcfg real6410_uartcfgs [ ] __initdata = {
[ 0 ] = {
2010-09-09 21:40:22 +09:00
. hwport = 0 ,
. flags = 0 ,
. ucon = UCON ,
. ulcon = ULCON ,
. ufcon = UFCON ,
2010-07-28 23:03:43 +03:00
} ,
[ 1 ] = {
2010-09-09 21:40:22 +09:00
. hwport = 1 ,
. flags = 0 ,
. ucon = UCON ,
. ulcon = ULCON ,
. ufcon = UFCON ,
2010-07-28 23:03:43 +03:00
} ,
[ 2 ] = {
2010-09-09 21:40:22 +09:00
. hwport = 2 ,
. flags = 0 ,
. ucon = UCON ,
. ulcon = ULCON ,
. ufcon = UFCON ,
2010-07-28 23:03:43 +03:00
} ,
[ 3 ] = {
2010-09-09 21:40:22 +09:00
. hwport = 3 ,
. flags = 0 ,
. ucon = UCON ,
. ulcon = ULCON ,
. ufcon = UFCON ,
2010-07-28 23:03:43 +03:00
} ,
} ;
2010-08-01 23:38:43 +03:00
/* DM9000AEP 10/100 ethernet controller */
static struct resource real6410_dm9k_resource [ ] = {
2012-05-12 16:12:29 +09:00
[ 0 ] = DEFINE_RES_MEM ( S3C64XX_PA_XM0CSN1 , 2 ) ,
[ 1 ] = DEFINE_RES_MEM ( S3C64XX_PA_XM0CSN1 + 4 , 2 ) ,
[ 2 ] = DEFINE_RES_NAMED ( S3C_EINT ( 7 ) , 1 , NULL , IORESOURCE_IRQ \
| IORESOURCE_IRQ_HIGHLEVEL ) ,
2010-08-01 23:38:43 +03:00
} ;
static struct dm9000_plat_data real6410_dm9k_pdata = {
2010-09-09 21:40:22 +09:00
. flags = ( DM9000_PLATF_16BITONLY | DM9000_PLATF_NO_EEPROM ) ,
2010-08-01 23:38:43 +03:00
} ;
static struct platform_device real6410_device_eth = {
2010-09-09 21:40:22 +09:00
. name = " dm9000 " ,
. id = - 1 ,
. num_resources = ARRAY_SIZE ( real6410_dm9k_resource ) ,
. resource = real6410_dm9k_resource ,
. dev = {
. platform_data = & real6410_dm9k_pdata ,
} ,
2010-08-01 23:38:43 +03:00
} ;
2012-03-24 21:58:47 +05:30
static struct s3c_fb_pd_win real6410_lcd_type0_fb_win = {
. max_bpp = 32 ,
. default_bpp = 16 ,
2012-03-24 21:58:48 +05:30
. xres = 480 ,
. yres = 272 ,
} ;
static struct fb_videomode real6410_lcd_type0_timing = {
/* 4.3" 480x272 */
. left_margin = 3 ,
. right_margin = 2 ,
. upper_margin = 1 ,
. lower_margin = 1 ,
. hsync_len = 40 ,
. vsync_len = 1 ,
2010-10-19 13:52:29 +09:00
} ;
2012-03-24 21:58:47 +05:30
static struct s3c_fb_pd_win real6410_lcd_type1_fb_win = {
. max_bpp = 32 ,
. default_bpp = 16 ,
2012-03-24 21:58:48 +05:30
. xres = 800 ,
. yres = 480 ,
} ;
static struct fb_videomode real6410_lcd_type1_timing = {
/* 7.0" 800x480 */
. left_margin = 8 ,
. right_margin = 13 ,
. upper_margin = 7 ,
. lower_margin = 5 ,
. hsync_len = 3 ,
. vsync_len = 1 ,
. xres = 800 ,
. yres = 480 ,
2012-03-24 21:58:47 +05:30
} ;
static struct s3c_fb_platdata real6410_lcd_pdata [ ] __initdata = {
2010-10-19 13:52:29 +09:00
{
2012-03-24 21:58:47 +05:30
. setup_gpio = s3c64xx_fb_gpio_setup_24bpp ,
2012-03-24 21:58:48 +05:30
. vtiming = & real6410_lcd_type0_timing ,
2012-03-24 21:58:47 +05:30
. win [ 0 ] = & real6410_lcd_type0_fb_win ,
. vidcon0 = VIDCON0_VIDOUT_RGB | VIDCON0_PNRMODE_RGB ,
. vidcon1 = VIDCON1_INV_HSYNC | VIDCON1_INV_VSYNC ,
2010-10-19 13:52:29 +09:00
} , {
2012-03-24 21:58:47 +05:30
. setup_gpio = s3c64xx_fb_gpio_setup_24bpp ,
2012-03-24 21:58:48 +05:30
. vtiming = & real6410_lcd_type1_timing ,
2012-03-24 21:58:47 +05:30
. win [ 0 ] = & real6410_lcd_type1_fb_win ,
. vidcon0 = VIDCON0_VIDOUT_RGB | VIDCON0_PNRMODE_RGB ,
. vidcon1 = VIDCON1_INV_HSYNC | VIDCON1_INV_VSYNC ,
2010-10-19 13:52:29 +09:00
} ,
2012-03-24 21:58:47 +05:30
{ } ,
2010-10-19 13:52:29 +09:00
} ;
2010-10-19 13:52:32 +09:00
static struct mtd_partition real6410_nand_part [ ] = {
[ 0 ] = {
. name = " uboot " ,
. size = SZ_1M ,
. offset = 0 ,
} ,
[ 1 ] = {
. name = " kernel " ,
. size = SZ_2M ,
. offset = SZ_1M ,
} ,
[ 2 ] = {
. name = " rootfs " ,
. size = MTDPART_SIZ_FULL ,
. offset = SZ_1M + SZ_2M ,
} ,
} ;
static struct s3c2410_nand_set real6410_nand_sets [ ] = {
[ 0 ] = {
. name = " nand " ,
. nr_chips = 1 ,
. nr_partitions = ARRAY_SIZE ( real6410_nand_part ) ,
. partitions = real6410_nand_part ,
} ,
} ;
static struct s3c2410_platform_nand real6410_nand_info = {
. tacls = 25 ,
. twrph0 = 55 ,
. twrph1 = 40 ,
. nr_sets = ARRAY_SIZE ( real6410_nand_sets ) ,
. sets = real6410_nand_sets ,
2016-10-20 19:42:44 -02:00
. ecc_mode = NAND_ECC_SOFT ,
2010-10-19 13:52:32 +09:00
} ;
2010-08-01 23:38:43 +03:00
static struct platform_device * real6410_devices [ ] __initdata = {
& real6410_device_eth ,
2010-08-02 01:39:25 +03:00
& s3c_device_hsmmc0 ,
& s3c_device_hsmmc1 ,
2010-10-19 13:52:29 +09:00
& s3c_device_fb ,
2010-10-19 13:52:32 +09:00
& s3c_device_nand ,
2010-10-19 13:52:34 +09:00
& s3c_device_adc ,
2010-10-19 13:52:37 +09:00
& s3c_device_ohci ,
2010-10-19 13:52:34 +09:00
} ;
2010-07-28 23:03:43 +03:00
static void __init real6410_map_io ( void )
{
2010-10-19 13:52:29 +09:00
u32 tmp ;
2010-07-28 23:03:43 +03:00
s3c64xx_init_io ( NULL , 0 ) ;
s3c24xx_init_clocks ( 12000000 ) ;
s3c24xx_init_uarts ( real6410_uartcfgs , ARRAY_SIZE ( real6410_uartcfgs ) ) ;
2013-01-09 18:47:04 -08:00
samsung_set_timer_source ( SAMSUNG_PWM3 , SAMSUNG_PWM4 ) ;
2010-10-19 13:52:29 +09:00
/* set the LCD type */
tmp = __raw_readl ( S3C64XX_SPCON ) ;
tmp & = ~ S3C64XX_SPCON_LCD_SEL_MASK ;
tmp | = S3C64XX_SPCON_LCD_SEL_RGB ;
__raw_writel ( tmp , S3C64XX_SPCON ) ;
/* remove the LCD bypass */
tmp = __raw_readl ( S3C64XX_MODEM_MIFPCON ) ;
tmp & = ~ MIFPCON_LCD_BYPASS ;
__raw_writel ( tmp , S3C64XX_MODEM_MIFPCON ) ;
}
/*
* real6410_features string
*
* 0 - 9 LCD configuration
*
*/
static char real6410_features_str [ 12 ] __initdata = " 0 " ;
static int __init real6410_features_setup ( char * str )
{
if ( str )
strlcpy ( real6410_features_str , str ,
sizeof ( real6410_features_str ) ) ;
return 1 ;
}
__setup ( " real6410= " , real6410_features_setup ) ;
# define FEATURE_SCREEN (1 << 0)
struct real6410_features_t {
int done ;
int lcd_index ;
} ;
static void real6410_parse_features (
struct real6410_features_t * features ,
const char * features_str )
{
const char * fp = features_str ;
features - > done = 0 ;
features - > lcd_index = 0 ;
while ( * fp ) {
char f = * fp + + ;
switch ( f ) {
case ' 0 ' . . . ' 9 ' : /* tft screen */
if ( features - > done & FEATURE_SCREEN ) {
printk ( KERN_INFO " REAL6410: '%c' ignored, "
" screen type already set \n " , f ) ;
} else {
int li = f - ' 0 ' ;
2012-03-24 21:58:47 +05:30
if ( li > = ARRAY_SIZE ( real6410_lcd_pdata ) )
2010-10-19 13:52:29 +09:00
printk ( KERN_INFO " REAL6410: '%c' out "
" of range LCD mode \n " , f ) ;
else {
features - > lcd_index = li ;
}
}
features - > done | = FEATURE_SCREEN ;
break ;
}
}
2010-07-28 23:03:43 +03:00
}
static void __init real6410_machine_init ( void )
{
2010-08-01 23:38:43 +03:00
u32 cs1 ;
2010-10-19 13:52:29 +09:00
struct real6410_features_t features = { 0 } ;
printk ( KERN_INFO " REAL6410: Option string real6410=%s \n " ,
real6410_features_str ) ;
/* Parse the feature string */
real6410_parse_features ( & features , real6410_features_str ) ;
printk ( KERN_INFO " REAL6410: selected LCD display is %dx%d \n " ,
2012-03-24 21:58:48 +05:30
real6410_lcd_pdata [ features . lcd_index ] . win [ 0 ] - > xres ,
real6410_lcd_pdata [ features . lcd_index ] . win [ 0 ] - > yres ) ;
2010-10-19 13:52:29 +09:00
2012-03-24 21:58:47 +05:30
s3c_fb_set_platdata ( & real6410_lcd_pdata [ features . lcd_index ] ) ;
2010-10-19 13:52:32 +09:00
s3c_nand_set_platdata ( & real6410_nand_info ) ;
2015-03-02 09:47:23 +01:00
s3c64xx_ts_set_platdata ( NULL ) ;
2010-08-01 23:38:43 +03:00
/* configure nCS1 width to 16 bits */
cs1 = __raw_readl ( S3C64XX_SROM_BW ) &
~ ( S3C64XX_SROM_BW__CS_MASK < < S3C64XX_SROM_BW__NCS1__SHIFT ) ;
cs1 | = ( ( 1 < < S3C64XX_SROM_BW__DATAWIDTH__SHIFT ) |
( 1 < < S3C64XX_SROM_BW__WAITENABLE__SHIFT ) |
( 1 < < S3C64XX_SROM_BW__BYTEENABLE__SHIFT ) ) < <
S3C64XX_SROM_BW__NCS1__SHIFT ;
__raw_writel ( cs1 , S3C64XX_SROM_BW ) ;
/* set timing for nCS1 suitable for ethernet chip */
__raw_writel ( ( 0 < < S3C64XX_SROM_BCX__PMC__SHIFT ) |
2010-09-09 21:40:22 +09:00
( 6 < < S3C64XX_SROM_BCX__TACP__SHIFT ) |
( 4 < < S3C64XX_SROM_BCX__TCAH__SHIFT ) |
( 1 < < S3C64XX_SROM_BCX__TCOH__SHIFT ) |
( 13 < < S3C64XX_SROM_BCX__TACC__SHIFT ) |
( 4 < < S3C64XX_SROM_BCX__TCOS__SHIFT ) |
( 0 < < S3C64XX_SROM_BCX__TACS__SHIFT ) , S3C64XX_SROM_BC1 ) ;
2010-08-01 23:38:43 +03:00
2010-10-19 13:52:29 +09:00
gpio_request ( S3C64XX_GPF ( 15 ) , " LCD power " ) ;
2010-08-01 23:38:43 +03:00
platform_add_devices ( real6410_devices , ARRAY_SIZE ( real6410_devices ) ) ;
2010-07-28 23:03:43 +03:00
}
MACHINE_START ( REAL6410 , " REAL6410 " )
/* Maintainer: Darius Augulis <augulis.darius@gmail.com> */
2011-07-05 22:38:17 -04:00
. atag_offset = 0x100 ,
2015-02-27 22:06:58 +01:00
. nr_irqs = S3C64XX_NR_IRQS ,
2010-07-28 23:03:43 +03:00
. init_irq = s3c6410_init_irq ,
. map_io = real6410_map_io ,
. init_machine = real6410_machine_init ,
2013-01-09 18:47:04 -08:00
. init_time = samsung_timer_init ,
2012-01-03 14:03:30 +01:00
. restart = s3c64xx_restart ,
2010-07-28 23:03:43 +03:00
MACHINE_END