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