14c425570SMasahiro Yamada /*
2f6e7f07cSMasahiro Yamada  * Copyright (C) 2014-2015 Masahiro Yamada <yamada.masahiro@socionext.com>
34c425570SMasahiro Yamada  *
44c425570SMasahiro Yamada  * SPDX-License-Identifier:	GPL-2.0+
54c425570SMasahiro Yamada  */
64c425570SMasahiro Yamada 
74c425570SMasahiro Yamada #include <common.h>
84c425570SMasahiro Yamada #include <spl.h>
98ea4f49aSMasahiro Yamada #include <libfdt.h>
104c425570SMasahiro Yamada #include <nand.h>
11f6e7f07cSMasahiro Yamada #include <linux/io.h>
124c425570SMasahiro Yamada #include <../drivers/mtd/nand/denali.h>
134c425570SMasahiro Yamada 
14fec48163SMasahiro Yamada #include "boot-mode/boot-device.h"
15fec48163SMasahiro Yamada 
164c425570SMasahiro Yamada static void nand_denali_wp_disable(void)
174c425570SMasahiro Yamada {
184c425570SMasahiro Yamada #ifdef CONFIG_NAND_DENALI
194c425570SMasahiro Yamada 	/*
204c425570SMasahiro Yamada 	 * Since the boot rom enables the write protection for NAND boot mode,
214c425570SMasahiro Yamada 	 * it must be disabled somewhere for "nand write", "nand erase", etc.
224c425570SMasahiro Yamada 	 * The workaround is here to not disturb the Denali NAND controller
234c425570SMasahiro Yamada 	 * driver just for a really SoC-specific thing.
244c425570SMasahiro Yamada 	 */
254c425570SMasahiro Yamada 	void __iomem *denali_reg = (void __iomem *)CONFIG_SYS_NAND_REGS_BASE;
264c425570SMasahiro Yamada 
274c425570SMasahiro Yamada 	writel(WRITE_PROTECT__FLAG, denali_reg + WRITE_PROTECT);
284c425570SMasahiro Yamada #endif
294c425570SMasahiro Yamada }
304c425570SMasahiro Yamada 
31881aa5a7SMasahiro Yamada #define VENDOR_PREFIX		"socionext,"
32881aa5a7SMasahiro Yamada #define DTB_FILE_PREFIX		"uniphier-"
338ea4f49aSMasahiro Yamada 
34881aa5a7SMasahiro Yamada static int uniphier_set_fdt_file(void)
358ea4f49aSMasahiro Yamada {
368ea4f49aSMasahiro Yamada 	DECLARE_GLOBAL_DATA_PTR;
37881aa5a7SMasahiro Yamada 	const char *compat;
38881aa5a7SMasahiro Yamada 	char dtb_name[256];
39881aa5a7SMasahiro Yamada 	int buf_len = 256;
40881aa5a7SMasahiro Yamada 	int ret;
418ea4f49aSMasahiro Yamada 
42*4565a74dSMasahiro Yamada 	if (getenv("fdt_file"))
43*4565a74dSMasahiro Yamada 		return 0;	/* do nothing if it is already set */
44*4565a74dSMasahiro Yamada 
45881aa5a7SMasahiro Yamada 	ret = fdt_get_string(gd->fdt_blob, 0, "compatible", &compat);
46881aa5a7SMasahiro Yamada 	if (ret)
47881aa5a7SMasahiro Yamada 		return -EINVAL;
48881aa5a7SMasahiro Yamada 
49881aa5a7SMasahiro Yamada 	if (strncmp(compat, VENDOR_PREFIX, strlen(VENDOR_PREFIX)))
50881aa5a7SMasahiro Yamada 		return -EINVAL;
51881aa5a7SMasahiro Yamada 
52881aa5a7SMasahiro Yamada 	compat += strlen(VENDOR_PREFIX);
53881aa5a7SMasahiro Yamada 
54881aa5a7SMasahiro Yamada 	strncat(dtb_name, DTB_FILE_PREFIX, buf_len);
55881aa5a7SMasahiro Yamada 	buf_len -= strlen(DTB_FILE_PREFIX);
56881aa5a7SMasahiro Yamada 
57881aa5a7SMasahiro Yamada 	strncat(dtb_name, compat, buf_len);
58881aa5a7SMasahiro Yamada 	buf_len -= strlen(compat);
59881aa5a7SMasahiro Yamada 
60881aa5a7SMasahiro Yamada 	strncat(dtb_name, ".dtb", buf_len);
61881aa5a7SMasahiro Yamada 
6280630dadSMasahiro Yamada 	return setenv("fdt_file", dtb_name);
638ea4f49aSMasahiro Yamada }
648ea4f49aSMasahiro Yamada 
654c425570SMasahiro Yamada int board_late_init(void)
664c425570SMasahiro Yamada {
674c425570SMasahiro Yamada 	puts("MODE:  ");
684c425570SMasahiro Yamada 
69fec48163SMasahiro Yamada 	switch (spl_boot_device_raw()) {
704c425570SMasahiro Yamada 	case BOOT_DEVICE_MMC1:
714c425570SMasahiro Yamada 		printf("eMMC Boot\n");
724c425570SMasahiro Yamada 		setenv("bootmode", "emmcboot");
734c425570SMasahiro Yamada 		break;
744c425570SMasahiro Yamada 	case BOOT_DEVICE_NAND:
754c425570SMasahiro Yamada 		printf("NAND Boot\n");
764c425570SMasahiro Yamada 		setenv("bootmode", "nandboot");
774c425570SMasahiro Yamada 		nand_denali_wp_disable();
784c425570SMasahiro Yamada 		break;
794c425570SMasahiro Yamada 	case BOOT_DEVICE_NOR:
804c425570SMasahiro Yamada 		printf("NOR Boot\n");
814c425570SMasahiro Yamada 		setenv("bootmode", "norboot");
824c425570SMasahiro Yamada 		break;
83fec48163SMasahiro Yamada 	case BOOT_DEVICE_USB:
84fec48163SMasahiro Yamada 		printf("USB Boot\n");
85fec48163SMasahiro Yamada 		setenv("bootmode", "usbboot");
86fec48163SMasahiro Yamada 		break;
874c425570SMasahiro Yamada 	default:
88d90b9745SMasahiro Yamada 		printf("Unknown\n");
89d90b9745SMasahiro Yamada 		break;
904c425570SMasahiro Yamada 	}
914c425570SMasahiro Yamada 
92881aa5a7SMasahiro Yamada 	if (uniphier_set_fdt_file())
93881aa5a7SMasahiro Yamada 		printf("fdt_file environment was not set correctly\n");
948ea4f49aSMasahiro Yamada 
954c425570SMasahiro Yamada 	return 0;
964c425570SMasahiro Yamada }
97