14c425570SMasahiro Yamada /*
240749d5aSMasahiro Yamada  * Copyright (C) 2014      Panasonic Corporation
340749d5aSMasahiro Yamada  * Copyright (C) 2015-2016 Socionext Inc.
440749d5aSMasahiro Yamada  *   Author: Masahiro Yamada <yamada.masahiro@socionext.com>
54c425570SMasahiro Yamada  *
64c425570SMasahiro Yamada  * SPDX-License-Identifier:	GPL-2.0+
74c425570SMasahiro Yamada  */
84c425570SMasahiro Yamada 
94c425570SMasahiro Yamada #include <common.h>
104c425570SMasahiro Yamada #include <spl.h>
118ea4f49aSMasahiro Yamada #include <libfdt.h>
124c425570SMasahiro Yamada #include <nand.h>
13f6e7f07cSMasahiro Yamada #include <linux/io.h>
144c425570SMasahiro Yamada #include <../drivers/mtd/nand/denali.h>
154c425570SMasahiro Yamada 
16784548efSMasahiro Yamada #include "init.h"
17fec48163SMasahiro Yamada 
184c425570SMasahiro Yamada static void nand_denali_wp_disable(void)
194c425570SMasahiro Yamada {
204c425570SMasahiro Yamada #ifdef CONFIG_NAND_DENALI
214c425570SMasahiro Yamada 	/*
224c425570SMasahiro Yamada 	 * Since the boot rom enables the write protection for NAND boot mode,
234c425570SMasahiro Yamada 	 * it must be disabled somewhere for "nand write", "nand erase", etc.
244c425570SMasahiro Yamada 	 * The workaround is here to not disturb the Denali NAND controller
254c425570SMasahiro Yamada 	 * driver just for a really SoC-specific thing.
264c425570SMasahiro Yamada 	 */
274c425570SMasahiro Yamada 	void __iomem *denali_reg = (void __iomem *)CONFIG_SYS_NAND_REGS_BASE;
284c425570SMasahiro Yamada 
294c425570SMasahiro Yamada 	writel(WRITE_PROTECT__FLAG, denali_reg + WRITE_PROTECT);
304c425570SMasahiro Yamada #endif
314c425570SMasahiro Yamada }
324c425570SMasahiro Yamada 
33881aa5a7SMasahiro Yamada static int uniphier_set_fdt_file(void)
348ea4f49aSMasahiro Yamada {
358ea4f49aSMasahiro Yamada 	DECLARE_GLOBAL_DATA_PTR;
36881aa5a7SMasahiro Yamada 	const char *compat;
37881aa5a7SMasahiro Yamada 	char dtb_name[256];
3840749d5aSMasahiro Yamada 	int buf_len = sizeof(dtb_name);
398ea4f49aSMasahiro Yamada 
404565a74dSMasahiro Yamada 	if (getenv("fdt_file"))
414565a74dSMasahiro Yamada 		return 0;	/* do nothing if it is already set */
424565a74dSMasahiro Yamada 
43b02e4044SSimon Glass 	compat = fdt_stringlist_get(gd->fdt_blob, 0, "compatible", 0, NULL);
44b02e4044SSimon Glass 	if (!compat)
45881aa5a7SMasahiro Yamada 		return -EINVAL;
46881aa5a7SMasahiro Yamada 
4740749d5aSMasahiro Yamada 	/* rip off the vendor prefix "socionext,"  */
4840749d5aSMasahiro Yamada 	compat = strchr(compat, ',');
4940749d5aSMasahiro Yamada 	if (!compat)
50881aa5a7SMasahiro Yamada 		return -EINVAL;
5140749d5aSMasahiro Yamada 	compat++;
52881aa5a7SMasahiro Yamada 
5340749d5aSMasahiro Yamada 	strncpy(dtb_name, compat, buf_len);
54881aa5a7SMasahiro Yamada 	buf_len -= strlen(compat);
55881aa5a7SMasahiro Yamada 
56881aa5a7SMasahiro Yamada 	strncat(dtb_name, ".dtb", buf_len);
57881aa5a7SMasahiro Yamada 
5880630dadSMasahiro Yamada 	return setenv("fdt_file", dtb_name);
598ea4f49aSMasahiro Yamada }
608ea4f49aSMasahiro Yamada 
614c425570SMasahiro Yamada int board_late_init(void)
624c425570SMasahiro Yamada {
634c425570SMasahiro Yamada 	puts("MODE:  ");
644c425570SMasahiro Yamada 
65784548efSMasahiro Yamada 	switch (uniphier_boot_device_raw()) {
664c425570SMasahiro Yamada 	case BOOT_DEVICE_MMC1:
67*63754842SMasahiro Yamada 		printf("eMMC Boot");
684c425570SMasahiro Yamada 		setenv("bootmode", "emmcboot");
694c425570SMasahiro Yamada 		break;
704c425570SMasahiro Yamada 	case BOOT_DEVICE_NAND:
71*63754842SMasahiro Yamada 		printf("NAND Boot");
724c425570SMasahiro Yamada 		setenv("bootmode", "nandboot");
734c425570SMasahiro Yamada 		nand_denali_wp_disable();
744c425570SMasahiro Yamada 		break;
754c425570SMasahiro Yamada 	case BOOT_DEVICE_NOR:
76*63754842SMasahiro Yamada 		printf("NOR Boot");
774c425570SMasahiro Yamada 		setenv("bootmode", "norboot");
784c425570SMasahiro Yamada 		break;
79fec48163SMasahiro Yamada 	case BOOT_DEVICE_USB:
80*63754842SMasahiro Yamada 		printf("USB Boot");
81fec48163SMasahiro Yamada 		setenv("bootmode", "usbboot");
82fec48163SMasahiro Yamada 		break;
834c425570SMasahiro Yamada 	default:
84*63754842SMasahiro Yamada 		printf("Unknown");
85d90b9745SMasahiro Yamada 		break;
864c425570SMasahiro Yamada 	}
874c425570SMasahiro Yamada 
88*63754842SMasahiro Yamada 	if (uniphier_have_internal_stm())
89*63754842SMasahiro Yamada 		printf(" (STM: %s)",
90*63754842SMasahiro Yamada 		       uniphier_boot_from_backend() ? "OFF" : "ON");
91*63754842SMasahiro Yamada 
92*63754842SMasahiro Yamada 	printf("\n");
93*63754842SMasahiro Yamada 
94881aa5a7SMasahiro Yamada 	if (uniphier_set_fdt_file())
95881aa5a7SMasahiro Yamada 		printf("fdt_file environment was not set correctly\n");
968ea4f49aSMasahiro Yamada 
974c425570SMasahiro Yamada 	return 0;
984c425570SMasahiro Yamada }
99