xref: /rk3399_ARM-atf/lib/coreboot/coreboot_table.c (revision 72e8f2456af54b75a0a1d92aadfce0b4bcde6ba1)
13429c77aSJulius Werner /*
2*4c700c15SGovindraj Raja  * Copyright (c) 2017-2020, Arm Limited and Contributors. All rights reserved.
33429c77aSJulius Werner  *
43429c77aSJulius Werner  * SPDX-License-Identifier: BSD-3-Clause
53429c77aSJulius Werner  */
63429c77aSJulius Werner 
71c5f5031SJulius Werner #include <assert.h>
83429c77aSJulius Werner #include <string.h>
909d40e0eSAntonio Nino Diaz 
1009d40e0eSAntonio Nino Diaz #include <drivers/coreboot/cbmem_console.h>
1109d40e0eSAntonio Nino Diaz #include <common/debug.h>
1209d40e0eSAntonio Nino Diaz #include <lib/coreboot.h>
1309d40e0eSAntonio Nino Diaz #include <lib/mmio.h>
1409d40e0eSAntonio Nino Diaz #include <lib/xlat_tables/xlat_tables_v2.h>
153429c77aSJulius Werner 
163429c77aSJulius Werner /*
173429c77aSJulius Werner  * Structures describing coreboot's in-memory descriptor tables. See
183429c77aSJulius Werner  * <coreboot>/src/commonlib/include/commonlib/coreboot_tables.h for
193429c77aSJulius Werner  * canonical implementation.
203429c77aSJulius Werner  */
213429c77aSJulius Werner 
223429c77aSJulius Werner typedef struct {
233429c77aSJulius Werner 	char signature[4];
243429c77aSJulius Werner 	uint32_t header_bytes;
253429c77aSJulius Werner 	uint32_t header_checksum;
263429c77aSJulius Werner 	uint32_t table_bytes;
273429c77aSJulius Werner 	uint32_t table_checksum;
283429c77aSJulius Werner 	uint32_t table_entries;
293429c77aSJulius Werner } cb_header_t;
303429c77aSJulius Werner 
313429c77aSJulius Werner typedef enum {
32579d1e90SJulius Werner 	CB_TAG_MEMORY = 0x1,
333429c77aSJulius Werner 	CB_TAG_SERIAL = 0xf,
341c5f5031SJulius Werner 	CB_TAG_CBMEM_CONSOLE = 0x17,
353429c77aSJulius Werner } cb_tag_t;
363429c77aSJulius Werner 
373429c77aSJulius Werner typedef struct {
383429c77aSJulius Werner 	uint32_t tag;
393429c77aSJulius Werner 	uint32_t size;
403429c77aSJulius Werner 	union {
41579d1e90SJulius Werner 		coreboot_memrange_t memranges[COREBOOT_MAX_MEMRANGES];
423429c77aSJulius Werner 		coreboot_serial_t serial;
431c5f5031SJulius Werner 		uint64_t uint64;
443429c77aSJulius Werner 	};
453429c77aSJulius Werner } cb_entry_t;
463429c77aSJulius Werner 
47579d1e90SJulius Werner coreboot_memrange_t coreboot_memranges[COREBOOT_MAX_MEMRANGES];
483429c77aSJulius Werner coreboot_serial_t coreboot_serial;
49f4bbf435SJeffrey Kardatzke uint64_t coreboot_table_addr;
50f4bbf435SJeffrey Kardatzke uint32_t coreboot_table_size;
513429c77aSJulius Werner 
523429c77aSJulius Werner /*
533429c77aSJulius Werner  * The coreboot table is parsed before the MMU is enabled (i.e. with strongly
543429c77aSJulius Werner  * ordered memory), so we cannot make unaligned accesses. The table entries
553429c77aSJulius Werner  * immediately follow one another without padding, so nothing after the header
563429c77aSJulius Werner  * is guaranteed to be naturally aligned. Therefore, we need to define safety
573429c77aSJulius Werner  * functions that can read unaligned integers.
583429c77aSJulius Werner  */
read_le32(uint32_t * p)593429c77aSJulius Werner static uint32_t read_le32(uint32_t *p)
603429c77aSJulius Werner {
613429c77aSJulius Werner 	uintptr_t addr = (uintptr_t)p;
623429c77aSJulius Werner 	return mmio_read_8(addr)		|
633429c77aSJulius Werner 	       mmio_read_8(addr + 1) << 8	|
643429c77aSJulius Werner 	       mmio_read_8(addr + 2) << 16	|
653429c77aSJulius Werner 	       mmio_read_8(addr + 3) << 24;
663429c77aSJulius Werner }
read_le64(uint64_t * p)671c5f5031SJulius Werner static uint64_t read_le64(uint64_t *p)
681c5f5031SJulius Werner {
691c5f5031SJulius Werner 	return read_le32((void *)p) | (uint64_t)read_le32((void *)p + 4) << 32;
701c5f5031SJulius Werner }
711c5f5031SJulius Werner 
expand_and_mmap(uintptr_t baseaddr,size_t size)721c5f5031SJulius Werner static void expand_and_mmap(uintptr_t baseaddr, size_t size)
731c5f5031SJulius Werner {
741c5f5031SJulius Werner 	uintptr_t pageaddr = round_down(baseaddr, PAGE_SIZE);
751c5f5031SJulius Werner 	size_t expanded = round_up(baseaddr - pageaddr + size, PAGE_SIZE);
761c5f5031SJulius Werner 	mmap_add_region(pageaddr, pageaddr, expanded,
771c5f5031SJulius Werner 			MT_MEMORY | MT_RW | MT_NS | MT_EXECUTE_NEVER);
781c5f5031SJulius Werner }
791c5f5031SJulius Werner 
setup_cbmem_console(uintptr_t baseaddr)801c5f5031SJulius Werner static void setup_cbmem_console(uintptr_t baseaddr)
811c5f5031SJulius Werner {
821c5f5031SJulius Werner 	static console_cbmc_t console;
83e21a788eSAndre Przywara 	assert(!console.console.base);	/* should only have one CBMEM console */
841c5f5031SJulius Werner 
851c5f5031SJulius Werner 	/* CBMEM console structure stores its size in first header field. */
861c5f5031SJulius Werner 	uint32_t size = *(uint32_t *)baseaddr;
871c5f5031SJulius Werner 	expand_and_mmap(baseaddr, size);
881c5f5031SJulius Werner 	console_cbmc_register(baseaddr, &console);
891c5f5031SJulius Werner 	console_set_scope(&console.console, CONSOLE_FLAG_BOOT |
901c5f5031SJulius Werner 					    CONSOLE_FLAG_RUNTIME |
911c5f5031SJulius Werner 					    CONSOLE_FLAG_CRASH);
921c5f5031SJulius Werner }
933429c77aSJulius Werner 
coreboot_get_memory_type(uintptr_t start,size_t size)94e0caf8f5SSaurabh Gorecha coreboot_memory_t coreboot_get_memory_type(uintptr_t start, size_t size)
95579d1e90SJulius Werner {
96579d1e90SJulius Werner 	int i;
97579d1e90SJulius Werner 
98579d1e90SJulius Werner 	for (i = 0; i < COREBOOT_MAX_MEMRANGES; i++) {
99579d1e90SJulius Werner 		coreboot_memrange_t *range = &coreboot_memranges[i];
100579d1e90SJulius Werner 
101579d1e90SJulius Werner 		if (range->type == CB_MEM_NONE)
102579d1e90SJulius Werner 			break;	/* end of table reached */
103e0caf8f5SSaurabh Gorecha 		if ((start >= range->start) &&
104e0caf8f5SSaurabh Gorecha 		    (start - range->start < range->size) &&
105e0caf8f5SSaurabh Gorecha 		    (size <= range->size - (start - range->start))) {
106579d1e90SJulius Werner 			return range->type;
107579d1e90SJulius Werner 		}
108e0caf8f5SSaurabh Gorecha 	}
109579d1e90SJulius Werner 
110579d1e90SJulius Werner 	return CB_MEM_NONE;
111579d1e90SJulius Werner }
112579d1e90SJulius Werner 
coreboot_get_table_location(uint64_t * address,uint32_t * size)113f4bbf435SJeffrey Kardatzke void coreboot_get_table_location(uint64_t *address, uint32_t *size)
114f4bbf435SJeffrey Kardatzke {
115f4bbf435SJeffrey Kardatzke 	*address = coreboot_table_addr;
116f4bbf435SJeffrey Kardatzke 	*size = coreboot_table_size;
117f4bbf435SJeffrey Kardatzke }
118f4bbf435SJeffrey Kardatzke 
coreboot_table_setup(void * base)1193429c77aSJulius Werner void coreboot_table_setup(void *base)
1203429c77aSJulius Werner {
1213429c77aSJulius Werner 	cb_header_t *header = base;
1223429c77aSJulius Werner 	void *ptr;
1233429c77aSJulius Werner 	int i;
1243429c77aSJulius Werner 
1253429c77aSJulius Werner 	if (strncmp(header->signature, "LBIO", 4)) {
1263429c77aSJulius Werner 		ERROR("coreboot table signature corrupt!\n");
1273429c77aSJulius Werner 		return;
1283429c77aSJulius Werner 	}
129f4bbf435SJeffrey Kardatzke 	coreboot_table_addr = (uint64_t) base;
130f4bbf435SJeffrey Kardatzke 	coreboot_table_size = header->header_bytes + header->table_bytes;
1313429c77aSJulius Werner 
1323429c77aSJulius Werner 	ptr = base + header->header_bytes;
1333429c77aSJulius Werner 	for (i = 0; i < header->table_entries; i++) {
134579d1e90SJulius Werner 		size_t size;
1353429c77aSJulius Werner 		cb_entry_t *entry = ptr;
1363429c77aSJulius Werner 
1373429c77aSJulius Werner 		if (ptr - base >= header->header_bytes + header->table_bytes) {
1383429c77aSJulius Werner 			ERROR("coreboot table exceeds its bounds!\n");
1393429c77aSJulius Werner 			break;
1403429c77aSJulius Werner 		}
1413429c77aSJulius Werner 
1423429c77aSJulius Werner 		switch (read_le32(&entry->tag)) {
143579d1e90SJulius Werner 		case CB_TAG_MEMORY:
144579d1e90SJulius Werner 			size = read_le32(&entry->size) -
145579d1e90SJulius Werner 			       offsetof(cb_entry_t, memranges);
146579d1e90SJulius Werner 			if (size > sizeof(coreboot_memranges)) {
147579d1e90SJulius Werner 				ERROR("Need to truncate coreboot memranges!\n");
148579d1e90SJulius Werner 				size = sizeof(coreboot_memranges);
149579d1e90SJulius Werner 			}
150579d1e90SJulius Werner 			memcpy(&coreboot_memranges, &entry->memranges, size);
151579d1e90SJulius Werner 			break;
1523429c77aSJulius Werner 		case CB_TAG_SERIAL:
1533429c77aSJulius Werner 			memcpy(&coreboot_serial, &entry->serial,
1543429c77aSJulius Werner 			       sizeof(coreboot_serial));
1553429c77aSJulius Werner 			break;
1561c5f5031SJulius Werner 		case CB_TAG_CBMEM_CONSOLE:
1571c5f5031SJulius Werner 			setup_cbmem_console(read_le64(&entry->uint64));
1581c5f5031SJulius Werner 			break;
1593429c77aSJulius Werner 		default:
1603429c77aSJulius Werner 			/* There are many tags TF doesn't need to care about. */
1613429c77aSJulius Werner 			break;
1623429c77aSJulius Werner 		}
1633429c77aSJulius Werner 
1643429c77aSJulius Werner 		ptr += read_le32(&entry->size);
1653429c77aSJulius Werner 	}
1663429c77aSJulius Werner }
167