xref: /rk3399_rockchip-uboot/drivers/core/fdtaddr.c (revision 8f7de5145da2de88e169e58343cceeee233362d4)
1d6ffb00aSSimon Glass /*
2d6ffb00aSSimon Glass  * Device addresses
3d6ffb00aSSimon Glass  *
4d6ffb00aSSimon Glass  * Copyright (c) 2017 Google, Inc
5d6ffb00aSSimon Glass  *
6d6ffb00aSSimon Glass  * (C) Copyright 2012
7d6ffb00aSSimon Glass  * Pavel Herrmann <morpheus.ibis@gmail.com>
8d6ffb00aSSimon Glass  *
9d6ffb00aSSimon Glass  * SPDX-License-Identifier:	GPL-2.0+
10d6ffb00aSSimon Glass  */
11d6ffb00aSSimon Glass 
12d6ffb00aSSimon Glass #include <common.h>
13d6ffb00aSSimon Glass #include <dm.h>
14d6ffb00aSSimon Glass #include <fdt_support.h>
15d6ffb00aSSimon Glass #include <asm/io.h>
16d6ffb00aSSimon Glass #include <dm/device-internal.h>
17d6ffb00aSSimon Glass 
18d6ffb00aSSimon Glass DECLARE_GLOBAL_DATA_PTR;
19d6ffb00aSSimon Glass 
devfdt_get_addr_index(struct udevice * dev,int index)20a821c4afSSimon Glass fdt_addr_t devfdt_get_addr_index(struct udevice *dev, int index)
21d6ffb00aSSimon Glass {
22d6ffb00aSSimon Glass #if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA)
23d6ffb00aSSimon Glass 	fdt_addr_t addr;
24d6ffb00aSSimon Glass 
25d6ffb00aSSimon Glass 	if (CONFIG_IS_ENABLED(OF_TRANSLATE)) {
26d6ffb00aSSimon Glass 		const fdt32_t *reg;
27d6ffb00aSSimon Glass 		int len = 0;
28d6ffb00aSSimon Glass 		int na, ns;
29d6ffb00aSSimon Glass 
30d6ffb00aSSimon Glass 		na = fdt_address_cells(gd->fdt_blob,
31d6ffb00aSSimon Glass 				       dev_of_offset(dev->parent));
32d6ffb00aSSimon Glass 		if (na < 1) {
33d6ffb00aSSimon Glass 			debug("bad #address-cells\n");
34d6ffb00aSSimon Glass 			return FDT_ADDR_T_NONE;
35d6ffb00aSSimon Glass 		}
36d6ffb00aSSimon Glass 
37d6ffb00aSSimon Glass 		ns = fdt_size_cells(gd->fdt_blob, dev_of_offset(dev->parent));
38d6ffb00aSSimon Glass 		if (ns < 0) {
39d6ffb00aSSimon Glass 			debug("bad #size-cells\n");
40d6ffb00aSSimon Glass 			return FDT_ADDR_T_NONE;
41d6ffb00aSSimon Glass 		}
42d6ffb00aSSimon Glass 
43d6ffb00aSSimon Glass 		reg = fdt_getprop(gd->fdt_blob, dev_of_offset(dev), "reg",
44d6ffb00aSSimon Glass 				  &len);
45d6ffb00aSSimon Glass 		if (!reg || (len <= (index * sizeof(fdt32_t) * (na + ns)))) {
46d6ffb00aSSimon Glass 			debug("Req index out of range\n");
47d6ffb00aSSimon Glass 			return FDT_ADDR_T_NONE;
48d6ffb00aSSimon Glass 		}
49d6ffb00aSSimon Glass 
50d6ffb00aSSimon Glass 		reg += index * (na + ns);
51d6ffb00aSSimon Glass 
52d6ffb00aSSimon Glass 		/*
53d6ffb00aSSimon Glass 		 * Use the full-fledged translate function for complex
54d6ffb00aSSimon Glass 		 * bus setups.
55d6ffb00aSSimon Glass 		 */
56d6ffb00aSSimon Glass 		addr = fdt_translate_address((void *)gd->fdt_blob,
57d6ffb00aSSimon Glass 					     dev_of_offset(dev), reg);
58d6ffb00aSSimon Glass 	} else {
59d6ffb00aSSimon Glass 		/*
60d6ffb00aSSimon Glass 		 * Use the "simple" translate function for less complex
61d6ffb00aSSimon Glass 		 * bus setups.
62d6ffb00aSSimon Glass 		 */
63d6ffb00aSSimon Glass 		addr = fdtdec_get_addr_size_auto_parent(gd->fdt_blob,
64d6ffb00aSSimon Glass 				dev_of_offset(dev->parent), dev_of_offset(dev),
65d6ffb00aSSimon Glass 				"reg", index, NULL, false);
66d6ffb00aSSimon Glass 		if (CONFIG_IS_ENABLED(SIMPLE_BUS) && addr != FDT_ADDR_T_NONE) {
67d6ffb00aSSimon Glass 			if (device_get_uclass_id(dev->parent) ==
68d6ffb00aSSimon Glass 			    UCLASS_SIMPLE_BUS)
69d6ffb00aSSimon Glass 				addr = simple_bus_translate(dev->parent, addr);
70d6ffb00aSSimon Glass 		}
71d6ffb00aSSimon Glass 	}
72d6ffb00aSSimon Glass 
73d6ffb00aSSimon Glass 	/*
74d6ffb00aSSimon Glass 	 * Some platforms need a special address translation. Those
75d6ffb00aSSimon Glass 	 * platforms (e.g. mvebu in SPL) can configure a translation
76d6ffb00aSSimon Glass 	 * offset in the DM by calling dm_set_translation_offset() that
77a821c4afSSimon Glass 	 * will get added to all addresses returned by devfdt_get_addr().
78d6ffb00aSSimon Glass 	 */
79d6ffb00aSSimon Glass 	addr += dm_get_translation_offset();
80d6ffb00aSSimon Glass 
81d6ffb00aSSimon Glass 	return addr;
82d6ffb00aSSimon Glass #else
83d6ffb00aSSimon Glass 	return FDT_ADDR_T_NONE;
84d6ffb00aSSimon Glass #endif
85d6ffb00aSSimon Glass }
86d6ffb00aSSimon Glass 
devfdt_get_addr_size_index(struct udevice * dev,int index,fdt_size_t * size)87a821c4afSSimon Glass fdt_addr_t devfdt_get_addr_size_index(struct udevice *dev, int index,
88d6ffb00aSSimon Glass 				   fdt_size_t *size)
89d6ffb00aSSimon Glass {
90d6ffb00aSSimon Glass #if CONFIG_IS_ENABLED(OF_CONTROL)
91d6ffb00aSSimon Glass 	/*
92d6ffb00aSSimon Glass 	 * Only get the size in this first call. We'll get the addr in the
93d6ffb00aSSimon Glass 	 * next call to the exisiting dev_get_xxx function which handles
94d6ffb00aSSimon Glass 	 * all config options.
95d6ffb00aSSimon Glass 	 */
96d6ffb00aSSimon Glass 	fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, dev_of_offset(dev),
97d6ffb00aSSimon Glass 					   "reg", index, size, false);
98d6ffb00aSSimon Glass 
99d6ffb00aSSimon Glass 	/*
100d6ffb00aSSimon Glass 	 * Get the base address via the existing function which handles
101d6ffb00aSSimon Glass 	 * all Kconfig cases
102d6ffb00aSSimon Glass 	 */
103a821c4afSSimon Glass 	return devfdt_get_addr_index(dev, index);
104d6ffb00aSSimon Glass #else
105d6ffb00aSSimon Glass 	return FDT_ADDR_T_NONE;
106d6ffb00aSSimon Glass #endif
107d6ffb00aSSimon Glass }
108d6ffb00aSSimon Glass 
devfdt_get_addr_name(struct udevice * dev,const char * name)109a821c4afSSimon Glass fdt_addr_t devfdt_get_addr_name(struct udevice *dev, const char *name)
110d6ffb00aSSimon Glass {
111d6ffb00aSSimon Glass #if CONFIG_IS_ENABLED(OF_CONTROL)
112d6ffb00aSSimon Glass 	int index;
113d6ffb00aSSimon Glass 
114d6ffb00aSSimon Glass 	index = fdt_stringlist_search(gd->fdt_blob, dev_of_offset(dev),
115d6ffb00aSSimon Glass 				      "reg-names", name);
116d6ffb00aSSimon Glass 	if (index < 0)
117d6ffb00aSSimon Glass 		return index;
118d6ffb00aSSimon Glass 
119a821c4afSSimon Glass 	return devfdt_get_addr_index(dev, index);
120d6ffb00aSSimon Glass #else
121d6ffb00aSSimon Glass 	return FDT_ADDR_T_NONE;
122d6ffb00aSSimon Glass #endif
123d6ffb00aSSimon Glass }
124d6ffb00aSSimon Glass 
devfdt_get_addr(struct udevice * dev)125a821c4afSSimon Glass fdt_addr_t devfdt_get_addr(struct udevice *dev)
126d6ffb00aSSimon Glass {
127a821c4afSSimon Glass 	return devfdt_get_addr_index(dev, 0);
128d6ffb00aSSimon Glass }
129d6ffb00aSSimon Glass 
devfdt_remap_addr_index(struct udevice * dev,int index)130*8f7de514SShawn Lin void *devfdt_remap_addr_index(struct udevice *dev, int index)
131*8f7de514SShawn Lin {
132*8f7de514SShawn Lin        fdt_addr_t addr = devfdt_get_addr_index(dev, index);
133*8f7de514SShawn Lin 
134*8f7de514SShawn Lin         if (addr == FDT_ADDR_T_NONE)
135*8f7de514SShawn Lin                 return NULL;
136*8f7de514SShawn Lin 
137*8f7de514SShawn Lin         return map_physmem(addr, 0, MAP_NOCACHE);
138*8f7de514SShawn Lin }
139*8f7de514SShawn Lin 
devfdt_get_addr_ptr(struct udevice * dev)140a821c4afSSimon Glass void *devfdt_get_addr_ptr(struct udevice *dev)
141d6ffb00aSSimon Glass {
142a821c4afSSimon Glass 	return (void *)(uintptr_t)devfdt_get_addr_index(dev, 0);
143d6ffb00aSSimon Glass }
144d6ffb00aSSimon Glass 
devfdt_map_physmem(struct udevice * dev,unsigned long size)145a821c4afSSimon Glass void *devfdt_map_physmem(struct udevice *dev, unsigned long size)
146d6ffb00aSSimon Glass {
147a821c4afSSimon Glass 	fdt_addr_t addr = devfdt_get_addr(dev);
148d6ffb00aSSimon Glass 
149d6ffb00aSSimon Glass 	if (addr == FDT_ADDR_T_NONE)
150d6ffb00aSSimon Glass 		return NULL;
151d6ffb00aSSimon Glass 
152d6ffb00aSSimon Glass 	return map_physmem(addr, size, MAP_NOCACHE);
153d6ffb00aSSimon Glass }
154