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