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