mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-10-24 22:11:26 +02:00 
			
		
		
		
	This function can fail if the device tree runs out of space. Rather than silently booting with an incomplete device tree, allow the failure to be detected. Unfortunately this involves changing a lot of places in the code. I have not changed behvaiour to return an error where one is not currently returned, to avoid unexpected breakage. Eventually it would be nice to allow boards to register functions to be called to update the device tree. This would avoid all the many functions to do this. However it's not clear yet if this should be done using driver model or with a linker list. This work is left for later. Signed-off-by: Simon Glass <sjg@chromium.org> Acked-by: Anatolij Gustschin <agust@denx.de>
		
			
				
	
	
		
			77 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			77 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright 2004 Freescale Semiconductor.
 | |
|  *
 | |
|  * SPDX-License-Identifier:	GPL-2.0+
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <libfdt.h>
 | |
| #include <fdt_support.h>
 | |
| #include "cadmus.h"
 | |
| 
 | |
| #if defined(CONFIG_OF_BOARD_SETUP)
 | |
| static void cds_pci_fixup(void *blob)
 | |
| {
 | |
| 	int node;
 | |
| 	const char *path;
 | |
| 	int len, slot, i;
 | |
| 	u32 *map = NULL, *piccells = NULL;
 | |
| 	int off, cells;
 | |
| 
 | |
| 	node = fdt_path_offset(blob, "/aliases");
 | |
| 	if (node >= 0) {
 | |
| 		path = fdt_getprop(blob, node, "pci0", NULL);
 | |
| 		if (path) {
 | |
| 			node = fdt_path_offset(blob, path);
 | |
| 			if (node >= 0) {
 | |
| 				map = fdt_getprop_w(blob, node, "interrupt-map", &len);
 | |
| 			}
 | |
| 			/* Each item in "interrupt-map" property is translated with
 | |
| 			 * following cells:
 | |
| 			 * PCI #address-cells, PCI #interrupt-cells,
 | |
| 			 * PIC address, PIC #address-cells, PIC #interrupt-cells.
 | |
| 			 */
 | |
| 			cells = fdt_getprop_u32_default(blob, path, "#address-cells", 1);
 | |
| 			cells += fdt_getprop_u32_default(blob, path, "#interrupt-cells", 1);
 | |
| 			off = fdt_node_offset_by_phandle(blob, fdt32_to_cpu(*(map+cells)));
 | |
| 			if (off <= 0)
 | |
| 				return;
 | |
| 			cells += 1;
 | |
| 			piccells = (u32 *)fdt_getprop(blob, off, "#address-cells", NULL);
 | |
| 			if (piccells == NULL)
 | |
| 				return;
 | |
| 			cells += *piccells;
 | |
| 			piccells = (u32 *)fdt_getprop(blob, off, "#interrupt-cells", NULL);
 | |
| 			if (piccells == NULL)
 | |
| 				return;
 | |
| 			cells += *piccells;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (map) {
 | |
| 		len /= sizeof(u32);
 | |
| 
 | |
| 		slot = get_pci_slot();
 | |
| 
 | |
| 		for (i=0;i<len;i+=cells) {
 | |
| 			/* We rotate the interrupt pins so that the mapping
 | |
| 			 * changes depending on the slot the carrier card is in.
 | |
| 			 */
 | |
| 			map[3] = ((map[3] + slot - 2) % 4) + 1;
 | |
| 			map+=cells;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int ft_board_setup(void *blob, bd_t *bd)
 | |
| {
 | |
| 	ft_cpu_setup(blob, bd);
 | |
| #ifdef CONFIG_PCI
 | |
| 	ft_pci_setup(blob, bd);
 | |
| 	cds_pci_fixup(blob);
 | |
| #endif
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| #endif
 |