xref: /rk3399_rockchip-uboot/board/lg/sniper/sniper.c (revision 00caae6d47645e68d6e5277aceb69592b49381a6)
141582e2eSPaul Kocialkowski /*
241582e2eSPaul Kocialkowski  * LG Optimus Black codename sniper board
341582e2eSPaul Kocialkowski  *
441582e2eSPaul Kocialkowski  * Copyright (C) 2015 Paul Kocialkowski <contact@paulk.fr>
541582e2eSPaul Kocialkowski  *
641582e2eSPaul Kocialkowski  * SPDX-License-Identifier:	GPL-2.0+
741582e2eSPaul Kocialkowski  */
841582e2eSPaul Kocialkowski 
941582e2eSPaul Kocialkowski #include <config.h>
1041582e2eSPaul Kocialkowski #include <common.h>
1141582e2eSPaul Kocialkowski #include <dm.h>
1241582e2eSPaul Kocialkowski #include <linux/ctype.h>
1341582e2eSPaul Kocialkowski #include <linux/usb/musb.h>
1441582e2eSPaul Kocialkowski #include <asm/omap_musb.h>
1541582e2eSPaul Kocialkowski #include <asm/arch/mmc_host_def.h>
1641582e2eSPaul Kocialkowski #include <asm/arch/sys_proto.h>
1741582e2eSPaul Kocialkowski #include <asm/arch/mem.h>
1841582e2eSPaul Kocialkowski #include <asm/io.h>
1941582e2eSPaul Kocialkowski #include <ns16550.h>
2041582e2eSPaul Kocialkowski #include <twl4030.h>
2141582e2eSPaul Kocialkowski #include "sniper.h"
2241582e2eSPaul Kocialkowski 
2341582e2eSPaul Kocialkowski DECLARE_GLOBAL_DATA_PTR;
2441582e2eSPaul Kocialkowski 
2541582e2eSPaul Kocialkowski const omap3_sysinfo sysinfo = {
2641582e2eSPaul Kocialkowski 	.mtype = DDR_STACKED,
2741582e2eSPaul Kocialkowski 	.board_string = "sniper",
2841582e2eSPaul Kocialkowski 	.nand_string = "MMC"
2941582e2eSPaul Kocialkowski };
3041582e2eSPaul Kocialkowski 
3141582e2eSPaul Kocialkowski static const struct ns16550_platdata serial_omap_platdata = {
3241582e2eSPaul Kocialkowski 	.base = OMAP34XX_UART3,
3341582e2eSPaul Kocialkowski 	.reg_shift = 2,
3417fa0326SHeiko Schocher 	.clock = V_NS16550_CLK,
3517fa0326SHeiko Schocher 	.fcr = UART_FCR_DEFVAL,
3641582e2eSPaul Kocialkowski };
3741582e2eSPaul Kocialkowski 
3841582e2eSPaul Kocialkowski U_BOOT_DEVICE(sniper_serial) = {
3941582e2eSPaul Kocialkowski 	.name = "ns16550_serial",
4041582e2eSPaul Kocialkowski 	.platdata = &serial_omap_platdata
4141582e2eSPaul Kocialkowski };
4241582e2eSPaul Kocialkowski 
4341582e2eSPaul Kocialkowski static struct musb_hdrc_config musb_config = {
4441582e2eSPaul Kocialkowski 	.multipoint = 1,
4541582e2eSPaul Kocialkowski 	.dyn_fifo = 1,
4641582e2eSPaul Kocialkowski 	.num_eps = 16,
4741582e2eSPaul Kocialkowski 	.ram_bits = 12
4841582e2eSPaul Kocialkowski };
4941582e2eSPaul Kocialkowski 
5041582e2eSPaul Kocialkowski static struct omap_musb_board_data musb_board_data = {
5141582e2eSPaul Kocialkowski 	.interface_type	= MUSB_INTERFACE_ULPI,
5241582e2eSPaul Kocialkowski };
5341582e2eSPaul Kocialkowski 
5441582e2eSPaul Kocialkowski static struct musb_hdrc_platform_data musb_platform_data = {
5541582e2eSPaul Kocialkowski 	.mode = MUSB_PERIPHERAL,
5641582e2eSPaul Kocialkowski 	.config = &musb_config,
5741582e2eSPaul Kocialkowski 	.power = 100,
5841582e2eSPaul Kocialkowski 	.platform_ops = &omap2430_ops,
5941582e2eSPaul Kocialkowski 	.board_data = &musb_board_data,
6041582e2eSPaul Kocialkowski };
6141582e2eSPaul Kocialkowski 
set_muxconf_regs(void)6241582e2eSPaul Kocialkowski void set_muxconf_regs(void)
6341582e2eSPaul Kocialkowski {
6441582e2eSPaul Kocialkowski 	MUX_SNIPER();
6541582e2eSPaul Kocialkowski }
6641582e2eSPaul Kocialkowski 
6741582e2eSPaul Kocialkowski #ifdef CONFIG_SPL_BUILD
get_board_mem_timings(struct board_sdrc_timings * timings)6841582e2eSPaul Kocialkowski void get_board_mem_timings(struct board_sdrc_timings *timings)
6941582e2eSPaul Kocialkowski {
7041582e2eSPaul Kocialkowski 	timings->mcfg = HYNIX_V_MCFG_200(256 << 20);
7141582e2eSPaul Kocialkowski 	timings->ctrla = HYNIX_V_ACTIMA_200;
7241582e2eSPaul Kocialkowski 	timings->ctrlb = HYNIX_V_ACTIMB_200;
7341582e2eSPaul Kocialkowski 	timings->rfr_ctrl = SDP_3430_SDRC_RFR_CTRL_200MHz;
7441582e2eSPaul Kocialkowski 	timings->mr = MICRON_V_MR_165;
7541582e2eSPaul Kocialkowski }
7641582e2eSPaul Kocialkowski #endif
7741582e2eSPaul Kocialkowski 
board_init(void)7841582e2eSPaul Kocialkowski int board_init(void)
7941582e2eSPaul Kocialkowski {
8041582e2eSPaul Kocialkowski 	/* GPMC init */
8141582e2eSPaul Kocialkowski 	gpmc_init();
8241582e2eSPaul Kocialkowski 
8341582e2eSPaul Kocialkowski 	/* MACH number */
8441582e2eSPaul Kocialkowski 	gd->bd->bi_arch_number = 3000;
8541582e2eSPaul Kocialkowski 
8641582e2eSPaul Kocialkowski 	/* ATAGs location */
8741582e2eSPaul Kocialkowski 	gd->bd->bi_boot_params = OMAP34XX_SDRC_CS0 + 0x100;
8841582e2eSPaul Kocialkowski 
8941582e2eSPaul Kocialkowski 	return 0;
9041582e2eSPaul Kocialkowski }
9141582e2eSPaul Kocialkowski 
misc_init_r(void)9241582e2eSPaul Kocialkowski int misc_init_r(void)
9341582e2eSPaul Kocialkowski {
9441582e2eSPaul Kocialkowski 	unsigned char keypad_matrix[64] = { 0 };
9541582e2eSPaul Kocialkowski 	char reboot_mode[2] = { 0 };
9641582e2eSPaul Kocialkowski 	unsigned char keys[3];
9741582e2eSPaul Kocialkowski 	unsigned char data = 0;
9841582e2eSPaul Kocialkowski 	int rc;
9941582e2eSPaul Kocialkowski 
10041582e2eSPaul Kocialkowski 	/* Power button reset init */
10141582e2eSPaul Kocialkowski 
10241582e2eSPaul Kocialkowski 	twl4030_power_reset_init();
10341582e2eSPaul Kocialkowski 
10441582e2eSPaul Kocialkowski 	/* Keypad */
10541582e2eSPaul Kocialkowski 
10641582e2eSPaul Kocialkowski 	twl4030_keypad_scan((unsigned char *)&keypad_matrix);
10741582e2eSPaul Kocialkowski 
10841582e2eSPaul Kocialkowski 	keys[0] = twl4030_keypad_key((unsigned char *)&keypad_matrix, 0, 0);
10941582e2eSPaul Kocialkowski 	keys[1] = twl4030_keypad_key((unsigned char *)&keypad_matrix, 0, 1);
11041582e2eSPaul Kocialkowski 	keys[2] = twl4030_keypad_key((unsigned char *)&keypad_matrix, 0, 2);
11141582e2eSPaul Kocialkowski 
11241582e2eSPaul Kocialkowski 	/* Reboot mode */
11341582e2eSPaul Kocialkowski 
11441582e2eSPaul Kocialkowski 	rc = omap_reboot_mode(reboot_mode, sizeof(reboot_mode));
11541582e2eSPaul Kocialkowski 
11641582e2eSPaul Kocialkowski 	if (keys[0])
11741582e2eSPaul Kocialkowski 		reboot_mode[0] = 'r';
11841582e2eSPaul Kocialkowski 	else if (keys[1])
11941582e2eSPaul Kocialkowski 		reboot_mode[0] = 'b';
12041582e2eSPaul Kocialkowski 
12141582e2eSPaul Kocialkowski 	if (rc < 0 || reboot_mode[0] == 'o') {
12241582e2eSPaul Kocialkowski 		/*
12341582e2eSPaul Kocialkowski 		 * When not rebooting, valid power on reasons are either the
12441582e2eSPaul Kocialkowski 		 * power button, charger plug or USB plug.
12541582e2eSPaul Kocialkowski 		 */
12641582e2eSPaul Kocialkowski 
12741582e2eSPaul Kocialkowski 		data |= twl4030_input_power_button();
12841582e2eSPaul Kocialkowski 		data |= twl4030_input_charger();
12941582e2eSPaul Kocialkowski 		data |= twl4030_input_usb();
13041582e2eSPaul Kocialkowski 
13141582e2eSPaul Kocialkowski 		if (!data)
13241582e2eSPaul Kocialkowski 			twl4030_power_off();
13341582e2eSPaul Kocialkowski 	}
13441582e2eSPaul Kocialkowski 
13541582e2eSPaul Kocialkowski 	if (reboot_mode[0] > 0 && isascii(reboot_mode[0])) {
136*00caae6dSSimon Glass 		if (!env_get("reboot-mode"))
137382bee57SSimon Glass 			env_set("reboot-mode", (char *)reboot_mode);
13841582e2eSPaul Kocialkowski 	}
13941582e2eSPaul Kocialkowski 
14041582e2eSPaul Kocialkowski 	omap_reboot_mode_clear();
14141582e2eSPaul Kocialkowski 
14241582e2eSPaul Kocialkowski 	/* Serial number */
14341582e2eSPaul Kocialkowski 
14441582e2eSPaul Kocialkowski 	omap_die_id_serial();
14541582e2eSPaul Kocialkowski 
14641582e2eSPaul Kocialkowski 	/* MUSB */
14741582e2eSPaul Kocialkowski 
14841582e2eSPaul Kocialkowski 	musb_register(&musb_platform_data, &musb_board_data, (void *)MUSB_BASE);
14941582e2eSPaul Kocialkowski 
15041582e2eSPaul Kocialkowski 	return 0;
15141582e2eSPaul Kocialkowski }
15241582e2eSPaul Kocialkowski 
get_board_rev(void)15341582e2eSPaul Kocialkowski u32 get_board_rev(void)
15441582e2eSPaul Kocialkowski {
15541582e2eSPaul Kocialkowski 	/* Sold devices are expected to be at least revision F. */
15641582e2eSPaul Kocialkowski 	return 6;
15741582e2eSPaul Kocialkowski }
15841582e2eSPaul Kocialkowski 
get_board_serial(struct tag_serialnr * serialnr)15941582e2eSPaul Kocialkowski void get_board_serial(struct tag_serialnr *serialnr)
16041582e2eSPaul Kocialkowski {
16141582e2eSPaul Kocialkowski 	omap_die_id_get_board_serial(serialnr);
16241582e2eSPaul Kocialkowski }
16341582e2eSPaul Kocialkowski 
reset_misc(void)16441582e2eSPaul Kocialkowski void reset_misc(void)
16541582e2eSPaul Kocialkowski {
16641582e2eSPaul Kocialkowski 	char reboot_mode[2] = { 0 };
16741582e2eSPaul Kocialkowski 
16841582e2eSPaul Kocialkowski 	/*
16941582e2eSPaul Kocialkowski 	 * Valid resets must contain the reboot mode magic, but we must not
17041582e2eSPaul Kocialkowski 	 * override it when set previously (e.g. reboot to bootloader).
17141582e2eSPaul Kocialkowski 	 */
17241582e2eSPaul Kocialkowski 
17341582e2eSPaul Kocialkowski 	omap_reboot_mode(reboot_mode, sizeof(reboot_mode));
17441582e2eSPaul Kocialkowski 	omap_reboot_mode_store(reboot_mode);
17541582e2eSPaul Kocialkowski }
17641582e2eSPaul Kocialkowski 
fb_set_reboot_flag(void)17741582e2eSPaul Kocialkowski int fb_set_reboot_flag(void)
17841582e2eSPaul Kocialkowski {
17941582e2eSPaul Kocialkowski 	return omap_reboot_mode_store("b");
18041582e2eSPaul Kocialkowski }
18141582e2eSPaul Kocialkowski 
board_mmc_init(bd_t * bis)18241582e2eSPaul Kocialkowski int board_mmc_init(bd_t *bis)
18341582e2eSPaul Kocialkowski {
18441582e2eSPaul Kocialkowski 	return omap_mmc_init(1, 0, 0, -1, -1);
18541582e2eSPaul Kocialkowski }
18641582e2eSPaul Kocialkowski 
board_mmc_power_init(void)18741582e2eSPaul Kocialkowski void board_mmc_power_init(void)
18841582e2eSPaul Kocialkowski {
18941582e2eSPaul Kocialkowski 	twl4030_power_mmc_init(1);
19041582e2eSPaul Kocialkowski }
191