mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-04 05:50:17 +00:00 
			
		
		
		
	Cover the "ARM local MMIO" range as well in the default MMU mapping in order to allow future code to access the GIC-400 without crashing. For now the GIC is not touched in u-boot, thus this change is a noop. See [1](BCM2711 ARM Peripherals) for reference. TEST: Enabled CONFIG_GICV2 and accessed the GIC in C code without crash. 1: https://datasheets.raspberrypi.com/bcm2711/bcm2711-peripherals.pdf Signed-off-by: Patrick Rudolph <patrick.rudolph@9elements.com> Reviewed-by: Simon Glass <sjg@chromium.org> Acked-by: Matthias Brugger <mbrugger@suse.com> Cc: Matthias Brugger <mbrugger@suse.com> Cc: Peter Robinson <pbrobinson@gmail.com>
		
			
				
	
	
		
			243 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			243 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0
 | 
						|
/*
 | 
						|
 * (C) Copyright 2012 Stephen Warren
 | 
						|
 *
 | 
						|
 * See file CREDITS for list of people who contributed to this
 | 
						|
 * project.
 | 
						|
 */
 | 
						|
 | 
						|
#include <cpu_func.h>
 | 
						|
#include <init.h>
 | 
						|
#include <dm/device.h>
 | 
						|
#include <fdt_support.h>
 | 
						|
#include <asm/global_data.h>
 | 
						|
 | 
						|
#define BCM2711_RPI4_PCIE_XHCI_MMIO_PHYS	0x600000000UL
 | 
						|
#define BCM2711_RPI4_PCIE_XHCI_MMIO_SIZE	0x400000UL
 | 
						|
 | 
						|
#ifdef CONFIG_ARM64
 | 
						|
#include <asm/armv8/mmu.h>
 | 
						|
 | 
						|
#define MEM_MAP_MAX_ENTRIES (4)
 | 
						|
 | 
						|
static struct mm_region bcm283x_mem_map[MEM_MAP_MAX_ENTRIES] = {
 | 
						|
	{
 | 
						|
		.virt = 0x00000000UL,
 | 
						|
		.phys = 0x00000000UL,
 | 
						|
		.size = 0x3f000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_NORMAL) |
 | 
						|
			 PTE_BLOCK_INNER_SHARE
 | 
						|
	}, {
 | 
						|
		.virt = 0x3f000000UL,
 | 
						|
		.phys = 0x3f000000UL,
 | 
						|
		.size = 0x01000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
 | 
						|
			 PTE_BLOCK_NON_SHARE |
 | 
						|
			 PTE_BLOCK_PXN | PTE_BLOCK_UXN
 | 
						|
	}, {
 | 
						|
		/* List terminator */
 | 
						|
		0,
 | 
						|
	}
 | 
						|
};
 | 
						|
 | 
						|
static struct mm_region bcm2711_mem_map[MEM_MAP_MAX_ENTRIES] = {
 | 
						|
	{
 | 
						|
		.virt = 0x00000000UL,
 | 
						|
		.phys = 0x00000000UL,
 | 
						|
		.size = 0xfc000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_NORMAL) |
 | 
						|
			 PTE_BLOCK_INNER_SHARE
 | 
						|
	}, {
 | 
						|
		.virt = 0xfc000000UL,
 | 
						|
		.phys = 0xfc000000UL,
 | 
						|
		.size = 0x04000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
 | 
						|
			 PTE_BLOCK_NON_SHARE |
 | 
						|
			 PTE_BLOCK_PXN | PTE_BLOCK_UXN
 | 
						|
	}, {
 | 
						|
		.virt = BCM2711_RPI4_PCIE_XHCI_MMIO_PHYS,
 | 
						|
		.phys = BCM2711_RPI4_PCIE_XHCI_MMIO_PHYS,
 | 
						|
		.size = BCM2711_RPI4_PCIE_XHCI_MMIO_SIZE,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
 | 
						|
			 PTE_BLOCK_NON_SHARE |
 | 
						|
			 PTE_BLOCK_PXN | PTE_BLOCK_UXN
 | 
						|
	}, {
 | 
						|
		/* List terminator */
 | 
						|
		0,
 | 
						|
	}
 | 
						|
};
 | 
						|
 | 
						|
static struct mm_region bcm2712_mem_map[MEM_MAP_MAX_ENTRIES] = {
 | 
						|
	{
 | 
						|
		/* First 1GB of DRAM */
 | 
						|
		.virt = 0x00000000UL,
 | 
						|
		.phys = 0x00000000UL,
 | 
						|
		.size = 0x40000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_NORMAL) |
 | 
						|
			 PTE_BLOCK_INNER_SHARE
 | 
						|
	}, {
 | 
						|
		/* Beginning of AXI bus where uSD controller lives */
 | 
						|
		.virt = 0x1000000000UL,
 | 
						|
		.phys = 0x1000000000UL,
 | 
						|
		.size = 0x0002000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
 | 
						|
			 PTE_BLOCK_NON_SHARE |
 | 
						|
			 PTE_BLOCK_PXN | PTE_BLOCK_UXN
 | 
						|
	}, {
 | 
						|
		/* SoC bus */
 | 
						|
		.virt = 0x107c000000UL,
 | 
						|
		.phys = 0x107c000000UL,
 | 
						|
		.size = 0x0004000000UL,
 | 
						|
		.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
 | 
						|
			 PTE_BLOCK_NON_SHARE |
 | 
						|
			 PTE_BLOCK_PXN | PTE_BLOCK_UXN
 | 
						|
	}, {
 | 
						|
		/* List terminator */
 | 
						|
		0,
 | 
						|
	}
 | 
						|
};
 | 
						|
 | 
						|
struct mm_region *mem_map = bcm283x_mem_map;
 | 
						|
 | 
						|
/*
 | 
						|
 * I/O address space varies on different chip versions.
 | 
						|
 * We set the base address by inspecting the DTB.
 | 
						|
 */
 | 
						|
static const struct udevice_id board_ids[] = {
 | 
						|
	{ .compatible = "brcm,bcm2837", .data = (ulong)&bcm283x_mem_map},
 | 
						|
	{ .compatible = "brcm,bcm2838", .data = (ulong)&bcm2711_mem_map},
 | 
						|
	{ .compatible = "brcm,bcm2711", .data = (ulong)&bcm2711_mem_map},
 | 
						|
	{ .compatible = "brcm,bcm2712", .data = (ulong)&bcm2712_mem_map},
 | 
						|
	{ },
 | 
						|
};
 | 
						|
 | 
						|
static void _rpi_update_mem_map(struct mm_region *pd)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < MEM_MAP_MAX_ENTRIES; i++) {
 | 
						|
		mem_map[i].virt = pd[i].virt;
 | 
						|
		mem_map[i].phys = pd[i].phys;
 | 
						|
		mem_map[i].size = pd[i].size;
 | 
						|
		mem_map[i].attrs = pd[i].attrs;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void rpi_update_mem_map(void)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	struct mm_region *mm;
 | 
						|
	const struct udevice_id *of_match = board_ids;
 | 
						|
 | 
						|
	while (of_match->compatible) {
 | 
						|
		ret = fdt_node_check_compatible(gd->fdt_blob, 0,
 | 
						|
						of_match->compatible);
 | 
						|
		if (!ret) {
 | 
						|
			mm = (struct mm_region *)of_match->data;
 | 
						|
			_rpi_update_mem_map(mm);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		of_match++;
 | 
						|
	}
 | 
						|
}
 | 
						|
#else
 | 
						|
static void rpi_update_mem_map(void) {}
 | 
						|
#endif
 | 
						|
 | 
						|
/* Default bcm283x devices addresses */
 | 
						|
unsigned long rpi_mbox_base  = 0x3f00b880;
 | 
						|
unsigned long rpi_sdhci_base = 0x3f300000;
 | 
						|
unsigned long rpi_wdog_base  = 0x3f100000;
 | 
						|
unsigned long rpi_timer_base = 0x3f003000;
 | 
						|
 | 
						|
int arch_cpu_init(void)
 | 
						|
{
 | 
						|
	icache_enable();
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int mach_cpu_init(void)
 | 
						|
{
 | 
						|
	int ret, soc, offset;
 | 
						|
	u64 io_base, size;
 | 
						|
 | 
						|
	rpi_update_mem_map();
 | 
						|
 | 
						|
	/* Get IO base from device tree */
 | 
						|
	soc = fdt_path_offset(gd->fdt_blob, "/soc");
 | 
						|
	if (soc < 0)
 | 
						|
		return soc;
 | 
						|
 | 
						|
	ret = fdt_read_range((void *)gd->fdt_blob, soc, 0, NULL,
 | 
						|
			     &io_base, &size);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	rpi_mbox_base  = io_base + 0x00b880;
 | 
						|
	rpi_sdhci_base = io_base + 0x300000;
 | 
						|
	rpi_wdog_base  = io_base + 0x100000;
 | 
						|
	rpi_timer_base = io_base + 0x003000;
 | 
						|
 | 
						|
	offset = fdt_node_offset_by_compatible(gd->fdt_blob, soc,
 | 
						|
					       "brcm,bcm2835-mbox");
 | 
						|
	if (offset > soc)
 | 
						|
		rpi_mbox_base = fdt_get_base_address(gd->fdt_blob, offset);
 | 
						|
 | 
						|
	offset = fdt_node_offset_by_compatible(gd->fdt_blob, soc,
 | 
						|
					       "brcm,bcm2835-sdhci");
 | 
						|
	if (offset > soc)
 | 
						|
		rpi_sdhci_base = fdt_get_base_address(gd->fdt_blob, offset);
 | 
						|
 | 
						|
	offset = fdt_node_offset_by_compatible(gd->fdt_blob, soc,
 | 
						|
					       "brcm,bcm2835-system-timer");
 | 
						|
	if (offset > soc)
 | 
						|
		rpi_timer_base = fdt_get_base_address(gd->fdt_blob, offset);
 | 
						|
 | 
						|
	offset = fdt_node_offset_by_compatible(gd->fdt_blob, soc,
 | 
						|
					       "brcm,bcm2712-pm");
 | 
						|
	if (offset > soc)
 | 
						|
		rpi_wdog_base = fdt_get_base_address(gd->fdt_blob, offset);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
#if defined(CONFIG_DISPLAY_CPUINFO)
 | 
						|
int print_cpuinfo(void)
 | 
						|
{
 | 
						|
	printf("CPU: BCM283x\n");
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef CONFIG_ARMV7_LPAE
 | 
						|
#ifdef CONFIG_TARGET_RPI_4_32B
 | 
						|
#define BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT	0xffc00000UL
 | 
						|
#include <addr_map.h>
 | 
						|
#include <asm/system.h>
 | 
						|
 | 
						|
int init_addr_map(void)
 | 
						|
{
 | 
						|
	mmu_set_region_dcache_behaviour_phys(BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT,
 | 
						|
					     BCM2711_RPI4_PCIE_XHCI_MMIO_PHYS,
 | 
						|
					     BCM2711_RPI4_PCIE_XHCI_MMIO_SIZE,
 | 
						|
					     DCACHE_OFF);
 | 
						|
 | 
						|
	/* identity mapping for 0..BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT */
 | 
						|
	addrmap_set_entry(0, 0, BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT, 0);
 | 
						|
	/* XHCI MMIO on PCIe at BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT */
 | 
						|
	addrmap_set_entry(BCM2711_RPI4_PCIE_XHCI_MMIO_VIRT,
 | 
						|
			  BCM2711_RPI4_PCIE_XHCI_MMIO_PHYS,
 | 
						|
			  BCM2711_RPI4_PCIE_XHCI_MMIO_SIZE, 1);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
void enable_caches(void)
 | 
						|
{
 | 
						|
	dcache_enable();
 | 
						|
}
 | 
						|
#endif
 |