| /* | 
 |  * Copyright (C) 2016 Imagination Technologies | 
 |  * Author: Paul Burton <paul.burton@mips.com> | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify it | 
 |  * under the terms of the GNU General Public License as published by the | 
 |  * Free Software Foundation;  either version 2 of the  License, or (at your | 
 |  * option) any later version. | 
 |  */ | 
 |  | 
 | #define pr_fmt(fmt) "sead3: " fmt | 
 |  | 
 | #include <linux/errno.h> | 
 | #include <linux/libfdt.h> | 
 | #include <linux/printk.h> | 
 | #include <linux/sizes.h> | 
 |  | 
 | #include <asm/fw/fw.h> | 
 | #include <asm/io.h> | 
 | #include <asm/machine.h> | 
 | #include <asm/yamon-dt.h> | 
 |  | 
 | #define SEAD_CONFIG			CKSEG1ADDR(0x1b100110) | 
 | #define SEAD_CONFIG_GIC_PRESENT		BIT(1) | 
 |  | 
 | #define MIPS_REVISION			CKSEG1ADDR(0x1fc00010) | 
 | #define MIPS_REVISION_MACHINE		(0xf << 4) | 
 | #define MIPS_REVISION_MACHINE_SEAD3	(0x4 << 4) | 
 |  | 
 | /* | 
 |  * Maximum 384MB RAM at physical address 0, preceding any I/O. | 
 |  */ | 
 | static struct yamon_mem_region mem_regions[] __initdata = { | 
 | 	/* start	size */ | 
 | 	{ 0,		SZ_256M + SZ_128M }, | 
 | 	{} | 
 | }; | 
 |  | 
 | static __init bool sead3_detect(void) | 
 | { | 
 | 	uint32_t rev; | 
 |  | 
 | 	rev = __raw_readl((void *)MIPS_REVISION); | 
 | 	return (rev & MIPS_REVISION_MACHINE) == MIPS_REVISION_MACHINE_SEAD3; | 
 | } | 
 |  | 
 | static __init int append_memory(void *fdt) | 
 | { | 
 | 	return yamon_dt_append_memory(fdt, mem_regions); | 
 | } | 
 |  | 
 | static __init int remove_gic(void *fdt) | 
 | { | 
 | 	const unsigned int cpu_ehci_int = 2; | 
 | 	const unsigned int cpu_uart_int = 4; | 
 | 	const unsigned int cpu_eth_int = 6; | 
 | 	int gic_off, cpu_off, uart_off, eth_off, ehci_off, err; | 
 | 	uint32_t cfg, cpu_phandle; | 
 |  | 
 | 	/* leave the GIC node intact if a GIC is present */ | 
 | 	cfg = __raw_readl((uint32_t *)SEAD_CONFIG); | 
 | 	if (cfg & SEAD_CONFIG_GIC_PRESENT) | 
 | 		return 0; | 
 |  | 
 | 	gic_off = fdt_node_offset_by_compatible(fdt, -1, "mti,gic"); | 
 | 	if (gic_off < 0) { | 
 | 		pr_err("unable to find DT GIC node: %d\n", gic_off); | 
 | 		return gic_off; | 
 | 	} | 
 |  | 
 | 	err = fdt_nop_node(fdt, gic_off); | 
 | 	if (err) { | 
 | 		pr_err("unable to nop GIC node\n"); | 
 | 		return err; | 
 | 	} | 
 |  | 
 | 	cpu_off = fdt_node_offset_by_compatible(fdt, -1, | 
 | 			"mti,cpu-interrupt-controller"); | 
 | 	if (cpu_off < 0) { | 
 | 		pr_err("unable to find CPU intc node: %d\n", cpu_off); | 
 | 		return cpu_off; | 
 | 	} | 
 |  | 
 | 	cpu_phandle = fdt_get_phandle(fdt, cpu_off); | 
 | 	if (!cpu_phandle) { | 
 | 		pr_err("unable to get CPU intc phandle\n"); | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	uart_off = fdt_node_offset_by_compatible(fdt, -1, "ns16550a"); | 
 | 	while (uart_off >= 0) { | 
 | 		err = fdt_setprop_u32(fdt, uart_off, "interrupt-parent", | 
 | 				      cpu_phandle); | 
 | 		if (err) { | 
 | 			pr_warn("unable to set UART interrupt-parent: %d\n", | 
 | 				err); | 
 | 			return err; | 
 | 		} | 
 |  | 
 | 		err = fdt_setprop_u32(fdt, uart_off, "interrupts", | 
 | 				      cpu_uart_int); | 
 | 		if (err) { | 
 | 			pr_err("unable to set UART interrupts property: %d\n", | 
 | 			       err); | 
 | 			return err; | 
 | 		} | 
 |  | 
 | 		uart_off = fdt_node_offset_by_compatible(fdt, uart_off, | 
 | 							 "ns16550a"); | 
 | 	} | 
 | 	if (uart_off != -FDT_ERR_NOTFOUND) { | 
 | 		pr_err("error searching for UART DT node: %d\n", uart_off); | 
 | 		return uart_off; | 
 | 	} | 
 |  | 
 | 	eth_off = fdt_node_offset_by_compatible(fdt, -1, "smsc,lan9115"); | 
 | 	if (eth_off < 0) { | 
 | 		pr_err("unable to find ethernet DT node: %d\n", eth_off); | 
 | 		return eth_off; | 
 | 	} | 
 |  | 
 | 	err = fdt_setprop_u32(fdt, eth_off, "interrupt-parent", cpu_phandle); | 
 | 	if (err) { | 
 | 		pr_err("unable to set ethernet interrupt-parent: %d\n", err); | 
 | 		return err; | 
 | 	} | 
 |  | 
 | 	err = fdt_setprop_u32(fdt, eth_off, "interrupts", cpu_eth_int); | 
 | 	if (err) { | 
 | 		pr_err("unable to set ethernet interrupts property: %d\n", err); | 
 | 		return err; | 
 | 	} | 
 |  | 
 | 	ehci_off = fdt_node_offset_by_compatible(fdt, -1, "generic-ehci"); | 
 | 	if (ehci_off < 0) { | 
 | 		pr_err("unable to find EHCI DT node: %d\n", ehci_off); | 
 | 		return ehci_off; | 
 | 	} | 
 |  | 
 | 	err = fdt_setprop_u32(fdt, ehci_off, "interrupt-parent", cpu_phandle); | 
 | 	if (err) { | 
 | 		pr_err("unable to set EHCI interrupt-parent: %d\n", err); | 
 | 		return err; | 
 | 	} | 
 |  | 
 | 	err = fdt_setprop_u32(fdt, ehci_off, "interrupts", cpu_ehci_int); | 
 | 	if (err) { | 
 | 		pr_err("unable to set EHCI interrupts property: %d\n", err); | 
 | 		return err; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct mips_fdt_fixup sead3_fdt_fixups[] __initconst = { | 
 | 	{ yamon_dt_append_cmdline, "append command line" }, | 
 | 	{ append_memory, "append memory" }, | 
 | 	{ remove_gic, "remove GIC when not present" }, | 
 | 	{ yamon_dt_serial_config, "append serial configuration" }, | 
 | 	{ }, | 
 | }; | 
 |  | 
 | static __init const void *sead3_fixup_fdt(const void *fdt, | 
 | 					  const void *match_data) | 
 | { | 
 | 	static unsigned char fdt_buf[16 << 10] __initdata; | 
 | 	int err; | 
 |  | 
 | 	if (fdt_check_header(fdt)) | 
 | 		panic("Corrupt DT"); | 
 |  | 
 | 	/* if this isn't SEAD3, something went wrong */ | 
 | 	BUG_ON(fdt_node_check_compatible(fdt, 0, "mti,sead-3")); | 
 |  | 
 | 	fw_init_cmdline(); | 
 |  | 
 | 	err = apply_mips_fdt_fixups(fdt_buf, sizeof(fdt_buf), | 
 | 				    fdt, sead3_fdt_fixups); | 
 | 	if (err) | 
 | 		panic("Unable to fixup FDT: %d", err); | 
 |  | 
 | 	return fdt_buf; | 
 | } | 
 |  | 
 | static __init unsigned int sead3_measure_hpt_freq(void) | 
 | { | 
 | 	void __iomem *status_reg = (void __iomem *)0xbf000410; | 
 | 	unsigned int freq, orig, tick = 0; | 
 | 	unsigned long flags; | 
 |  | 
 | 	local_irq_save(flags); | 
 |  | 
 | 	orig = readl(status_reg) & 0x2;		      /* get original sample */ | 
 | 	/* wait for transition */ | 
 | 	while ((readl(status_reg) & 0x2) == orig) | 
 | 		; | 
 | 	orig = orig ^ 0x2;			      /* flip the bit */ | 
 |  | 
 | 	write_c0_count(0); | 
 |  | 
 | 	/* wait 1 second (the sampling clock transitions every 10ms) */ | 
 | 	while (tick < 100) { | 
 | 		/* wait for transition */ | 
 | 		while ((readl(status_reg) & 0x2) == orig) | 
 | 			; | 
 | 		orig = orig ^ 0x2;			      /* flip the bit */ | 
 | 		tick++; | 
 | 	} | 
 |  | 
 | 	freq = read_c0_count(); | 
 |  | 
 | 	local_irq_restore(flags); | 
 |  | 
 | 	return freq; | 
 | } | 
 |  | 
 | extern char __dtb_sead3_begin[]; | 
 |  | 
 | MIPS_MACHINE(sead3) = { | 
 | 	.fdt = __dtb_sead3_begin, | 
 | 	.detect = sead3_detect, | 
 | 	.fixup_fdt = sead3_fixup_fdt, | 
 | 	.measure_hpt_freq = sead3_measure_hpt_freq, | 
 | }; |