FDT: Add fixup support for multiple banks of memory
Add fdt_fixup_memory_banks and reimplement fdt_fixup_memory using it. Tested on OMAP3 beagle board with two banks of memory. Signed-off-by: John Rigby <john.rigby@linaro.org> CC: Jerry Van Baren <vanbaren@cideas.com> Acked-by: Gerald Van Baren <vanbaren@cideas.com>
This commit is contained in:
		
							parent
							
								
									d1263fced2
								
							
						
					
					
						commit
						a6bd9e83ab
					
				| 
						 | 
					@ -362,10 +362,40 @@ void do_fixup_by_compat_u32(void *fdt, const char *compat,
 | 
				
			||||||
	do_fixup_by_compat(fdt, compat, prop, &val, 4, create);
 | 
						do_fixup_by_compat(fdt, compat, prop, &val, 4, create);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int fdt_fixup_memory(void *blob, u64 start, u64 size)
 | 
					/*
 | 
				
			||||||
 | 
					 * Get cells len in bytes
 | 
				
			||||||
 | 
					 *     if #NNNN-cells property is 2 then len is 8
 | 
				
			||||||
 | 
					 *     otherwise len is 4
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static int get_cells_len(void *blob, char *nr_cells_name)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int err, nodeoffset, len = 0;
 | 
						const u32 *cell;
 | 
				
			||||||
	u8 tmp[16];
 | 
					
 | 
				
			||||||
 | 
						cell = fdt_getprop(blob, 0, nr_cells_name, NULL);
 | 
				
			||||||
 | 
						if (cell && *cell == 2)
 | 
				
			||||||
 | 
							return 8;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 4;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Write a 4 or 8 byte big endian cell
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static void write_cell(u8 *addr, u64 val, int size)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int shift = (size - 1) * 8;
 | 
				
			||||||
 | 
						while (size-- > 0) {
 | 
				
			||||||
 | 
							*addr++ = (val >> shift) & 0xff;
 | 
				
			||||||
 | 
							shift -= 8;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int fdt_fixup_memory_banks(void *blob, u64 start[], u64 size[], int banks)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int err, nodeoffset;
 | 
				
			||||||
 | 
						int addr_cell_len, size_cell_len, len;
 | 
				
			||||||
 | 
						u8 tmp[banks * 8];
 | 
				
			||||||
 | 
						int bank;
 | 
				
			||||||
	const u32 *addrcell, *sizecell;
 | 
						const u32 *addrcell, *sizecell;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = fdt_check_header(blob);
 | 
						err = fdt_check_header(blob);
 | 
				
			||||||
| 
						 | 
					@ -391,44 +421,15 @@ int fdt_fixup_memory(void *blob, u64 start, u64 size)
 | 
				
			||||||
		return err;
 | 
							return err;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	addrcell = fdt_getprop(blob, 0, "#address-cells", NULL);
 | 
						addr_cell_len = get_cells_len(blob, "#address-cells");
 | 
				
			||||||
	/* use shifts and mask to ensure endianness */
 | 
						size_cell_len = get_cells_len(blob, "#size-cells");
 | 
				
			||||||
	if ((addrcell) && (*addrcell == 2)) {
 | 
					 | 
				
			||||||
		tmp[0] = (start >> 56) & 0xff;
 | 
					 | 
				
			||||||
		tmp[1] = (start >> 48) & 0xff;
 | 
					 | 
				
			||||||
		tmp[2] = (start >> 40) & 0xff;
 | 
					 | 
				
			||||||
		tmp[3] = (start >> 32) & 0xff;
 | 
					 | 
				
			||||||
		tmp[4] = (start >> 24) & 0xff;
 | 
					 | 
				
			||||||
		tmp[5] = (start >> 16) & 0xff;
 | 
					 | 
				
			||||||
		tmp[6] = (start >>  8) & 0xff;
 | 
					 | 
				
			||||||
		tmp[7] = (start      ) & 0xff;
 | 
					 | 
				
			||||||
		len = 8;
 | 
					 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		tmp[0] = (start >> 24) & 0xff;
 | 
					 | 
				
			||||||
		tmp[1] = (start >> 16) & 0xff;
 | 
					 | 
				
			||||||
		tmp[2] = (start >>  8) & 0xff;
 | 
					 | 
				
			||||||
		tmp[3] = (start      ) & 0xff;
 | 
					 | 
				
			||||||
		len = 4;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	sizecell = fdt_getprop(blob, 0, "#size-cells", NULL);
 | 
						for (bank = 0, len = 0; bank < banks; bank++) {
 | 
				
			||||||
	/* use shifts and mask to ensure endianness */
 | 
							write_cell(tmp + len, start[bank], addr_cell_len);
 | 
				
			||||||
	if ((sizecell) && (*sizecell == 2)) {
 | 
							len += addr_cell_len;
 | 
				
			||||||
		tmp[0+len] = (size >> 56) & 0xff;
 | 
					
 | 
				
			||||||
		tmp[1+len] = (size >> 48) & 0xff;
 | 
							write_cell(tmp + len, size[bank], size_cell_len);
 | 
				
			||||||
		tmp[2+len] = (size >> 40) & 0xff;
 | 
							len += size_cell_len;
 | 
				
			||||||
		tmp[3+len] = (size >> 32) & 0xff;
 | 
					 | 
				
			||||||
		tmp[4+len] = (size >> 24) & 0xff;
 | 
					 | 
				
			||||||
		tmp[5+len] = (size >> 16) & 0xff;
 | 
					 | 
				
			||||||
		tmp[6+len] = (size >>  8) & 0xff;
 | 
					 | 
				
			||||||
		tmp[7+len] = (size      ) & 0xff;
 | 
					 | 
				
			||||||
		len += 8;
 | 
					 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		tmp[0+len] = (size >> 24) & 0xff;
 | 
					 | 
				
			||||||
		tmp[1+len] = (size >> 16) & 0xff;
 | 
					 | 
				
			||||||
		tmp[2+len] = (size >>  8) & 0xff;
 | 
					 | 
				
			||||||
		tmp[3+len] = (size      ) & 0xff;
 | 
					 | 
				
			||||||
		len += 4;
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err = fdt_setprop(blob, nodeoffset, "reg", tmp, len);
 | 
						err = fdt_setprop(blob, nodeoffset, "reg", tmp, len);
 | 
				
			||||||
| 
						 | 
					@ -440,6 +441,11 @@ int fdt_fixup_memory(void *blob, u64 start, u64 size)
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int fdt_fixup_memory(void *blob, u64 start, u64 size)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						return fdt_fixup_memory_banks(blob, &start, &size, 1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void fdt_fixup_ethernet(void *fdt)
 | 
					void fdt_fixup_ethernet(void *fdt)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int node, i, j;
 | 
						int node, i, j;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -48,6 +48,7 @@ void do_fixup_by_compat(void *fdt, const char *compat,
 | 
				
			||||||
void do_fixup_by_compat_u32(void *fdt, const char *compat,
 | 
					void do_fixup_by_compat_u32(void *fdt, const char *compat,
 | 
				
			||||||
			    const char *prop, u32 val, int create);
 | 
								    const char *prop, u32 val, int create);
 | 
				
			||||||
int fdt_fixup_memory(void *blob, u64 start, u64 size);
 | 
					int fdt_fixup_memory(void *blob, u64 start, u64 size);
 | 
				
			||||||
 | 
					int fdt_fixup_memory_banks(void *blob, u64 start[], u64 size[], int banks);
 | 
				
			||||||
void fdt_fixup_ethernet(void *fdt);
 | 
					void fdt_fixup_ethernet(void *fdt);
 | 
				
			||||||
int fdt_find_and_setprop(void *fdt, const char *node, const char *prop,
 | 
					int fdt_find_and_setprop(void *fdt, const char *node, const char *prop,
 | 
				
			||||||
			 const void *val, int len, int create);
 | 
								 const void *val, int len, int create);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue