| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Flattened Image Tree loader. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2016 Imagination Technologies | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							| 
									
										
										
										
											2020-10-23 12:26:33 +00:00
										 |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  | #include "qapi/error.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-25 09:41:57 -03:00
										 |  |  | #include "qemu/units.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | #include "exec/memory.h"
 | 
					
						
							|  |  |  | #include "hw/loader.h"
 | 
					
						
							|  |  |  | #include "hw/loader-fit.h"
 | 
					
						
							|  |  |  | #include "qemu/cutils.h"
 | 
					
						
							|  |  |  | #include "qemu/error-report.h"
 | 
					
						
							|  |  |  | #include "sysemu/device_tree.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <libfdt.h>
 | 
					
						
							|  |  |  | #include <zlib.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define FIT_LOADER_MAX_PATH (128)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const void *fit_load_image_alloc(const void *itb, const char *name, | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |                                         int *poff, size_t *psz, Error **errp) | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const void *data; | 
					
						
							|  |  |  |     const char *comp; | 
					
						
							|  |  |  |     void *uncomp_data; | 
					
						
							|  |  |  |     char path[FIT_LOADER_MAX_PATH]; | 
					
						
							|  |  |  |     int off, sz; | 
					
						
							|  |  |  |     ssize_t uncomp_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     snprintf(path, sizeof(path), "/images/%s", name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     off = fdt_path_offset(itb, path); | 
					
						
							|  |  |  |     if (off < 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_setg(errp, "can't find node %s", path); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (poff) { | 
					
						
							|  |  |  |         *poff = off; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data = fdt_getprop(itb, off, "data", &sz); | 
					
						
							|  |  |  |     if (!data) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_setg(errp, "can't get %s/data", path); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp = fdt_getprop(itb, off, "compression", NULL); | 
					
						
							|  |  |  |     if (!comp || !strcmp(comp, "none")) { | 
					
						
							|  |  |  |         if (psz) { | 
					
						
							|  |  |  |             *psz = sz; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         uncomp_data = g_malloc(sz); | 
					
						
							|  |  |  |         memmove(uncomp_data, data, sz); | 
					
						
							|  |  |  |         return uncomp_data; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!strcmp(comp, "gzip")) { | 
					
						
							|  |  |  |         uncomp_len = UBOOT_MAX_GUNZIP_BYTES; | 
					
						
							|  |  |  |         uncomp_data = g_malloc(uncomp_len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         uncomp_len = gunzip(uncomp_data, uncomp_len, (void *) data, sz); | 
					
						
							|  |  |  |         if (uncomp_len < 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |             error_setg(errp, "unable to decompress %s image", name); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |             g_free(uncomp_data); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         data = g_realloc(uncomp_data, uncomp_len); | 
					
						
							|  |  |  |         if (psz) { | 
					
						
							|  |  |  |             *psz = uncomp_len; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return data; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     error_setg(errp, "unknown compression '%s'", comp); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fit_image_addr(const void *itb, int img, const char *name, | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |                           hwaddr *addr, Error **errp) | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const void *prop; | 
					
						
							|  |  |  |     int len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     prop = fdt_getprop(itb, img, name, &len); | 
					
						
							|  |  |  |     if (!prop) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_setg(errp, "can't find %s address", name); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return -ENOENT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (len) { | 
					
						
							|  |  |  |     case 4: | 
					
						
							|  |  |  |         *addr = fdt32_to_cpu(*(fdt32_t *)prop); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     case 8: | 
					
						
							|  |  |  |         *addr = fdt64_to_cpu(*(fdt64_t *)prop); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_setg(errp, "invalid %s address length %d", name, len); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fit_load_kernel(const struct fit_loader *ldr, const void *itb, | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |                            int cfg, void *opaque, hwaddr *pend, | 
					
						
							|  |  |  |                            Error **errp) | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												hw/core/loader-fit: Fix missing ERRP_GUARD() for error_prepend()
As the comment in qapi/error, passing @errp to error_prepend() requires
ERRP_GUARD():
* = Why, when and how to use ERRP_GUARD() =
*
* Without ERRP_GUARD(), use of the @errp parameter is restricted:
...
* - It should not be passed to error_prepend(), error_vprepend() or
*   error_append_hint(), because that doesn't work with &error_fatal.
* ERRP_GUARD() lifts these restrictions.
*
* To use ERRP_GUARD(), add it right at the beginning of the function.
* @errp can then be used without worrying about the argument being
* NULL or &error_fatal.
ERRP_GUARD() could avoid the case when @errp is &error_fatal, the user
can't see this additional information, because exit() happens in
error_setg earlier than information is added [1].
In hw/core/loader-fit.c, there are 2 functions passing @errp to
error_prepend() without ERRP_GUARD():
 - fit_load_kernel()
 - fit_load_fdt()
Their @errp parameters are both the pointers of the local @err virable
in load_fit().
Though they don't cause the issue like [1] said, to follow the
requirement of @errp, add missing ERRP_GUARD() at their beginning.
[1]: Issue description in the commit message of commit ae7c80a7bd73
     ("error: New macro ERRP_GUARD()").
Cc: Paul Burton <paulburton@kernel.org>
Cc: Aleksandar Rikalo <aleksandar.rikalo@syrmia.com>
Signed-off-by: Zhao Liu <zhao1.liu@intel.com>
Message-ID: <20240311033822.3142585-15-zhao1.liu@linux.intel.com>
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
											
										 
											2024-03-11 11:38:07 +08:00
										 |  |  |     ERRP_GUARD(); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     const char *name; | 
					
						
							|  |  |  |     const void *data; | 
					
						
							|  |  |  |     const void *load_data; | 
					
						
							|  |  |  |     hwaddr load_addr, entry_addr; | 
					
						
							|  |  |  |     int img_off, err; | 
					
						
							|  |  |  |     size_t sz; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     name = fdt_getprop(itb, cfg, "kernel", NULL); | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_setg(errp, "no kernel specified by FIT configuration"); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     load_data = data = fit_load_image_alloc(itb, name, &img_off, &sz, errp); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     if (!data) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_prepend(errp, "unable to load kernel image from FIT: "); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     err = fit_image_addr(itb, img_off, "load", &load_addr, errp); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     if (err) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_prepend(errp, "unable to read kernel load address from FIT: "); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         ret = err; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     err = fit_image_addr(itb, img_off, "entry", &entry_addr, errp); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     if (err) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_prepend(errp, "unable to read kernel entry address from FIT: "); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         ret = err; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ldr->kernel_filter) { | 
					
						
							|  |  |  |         load_data = ldr->kernel_filter(opaque, data, &load_addr, &entry_addr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pend) { | 
					
						
							|  |  |  |         *pend = load_addr + sz; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     load_addr = ldr->addr_to_phys(opaque, load_addr); | 
					
						
							|  |  |  |     rom_add_blob_fixed(name, load_data, sz, load_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = 0; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  |     g_free((void *) data); | 
					
						
							|  |  |  |     if (data != load_data) { | 
					
						
							|  |  |  |         g_free((void *) load_data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fit_load_fdt(const struct fit_loader *ldr, const void *itb, | 
					
						
							|  |  |  |                         int cfg, void *opaque, const void *match_data, | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |                         hwaddr kernel_end, Error **errp) | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												hw/core/loader-fit: Fix missing ERRP_GUARD() for error_prepend()
As the comment in qapi/error, passing @errp to error_prepend() requires
ERRP_GUARD():
* = Why, when and how to use ERRP_GUARD() =
*
* Without ERRP_GUARD(), use of the @errp parameter is restricted:
...
* - It should not be passed to error_prepend(), error_vprepend() or
*   error_append_hint(), because that doesn't work with &error_fatal.
* ERRP_GUARD() lifts these restrictions.
*
* To use ERRP_GUARD(), add it right at the beginning of the function.
* @errp can then be used without worrying about the argument being
* NULL or &error_fatal.
ERRP_GUARD() could avoid the case when @errp is &error_fatal, the user
can't see this additional information, because exit() happens in
error_setg earlier than information is added [1].
In hw/core/loader-fit.c, there are 2 functions passing @errp to
error_prepend() without ERRP_GUARD():
 - fit_load_kernel()
 - fit_load_fdt()
Their @errp parameters are both the pointers of the local @err virable
in load_fit().
Though they don't cause the issue like [1] said, to follow the
requirement of @errp, add missing ERRP_GUARD() at their beginning.
[1]: Issue description in the commit message of commit ae7c80a7bd73
     ("error: New macro ERRP_GUARD()").
Cc: Paul Burton <paulburton@kernel.org>
Cc: Aleksandar Rikalo <aleksandar.rikalo@syrmia.com>
Signed-off-by: Zhao Liu <zhao1.liu@intel.com>
Message-ID: <20240311033822.3142585-15-zhao1.liu@linux.intel.com>
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
											
										 
											2024-03-11 11:38:07 +08:00
										 |  |  |     ERRP_GUARD(); | 
					
						
							| 
									
										
										
										
											2019-12-04 10:36:14 +01:00
										 |  |  |     Error *err = NULL; | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     const char *name; | 
					
						
							|  |  |  |     const void *data; | 
					
						
							|  |  |  |     const void *load_data; | 
					
						
							|  |  |  |     hwaddr load_addr; | 
					
						
							| 
									
										
										
										
											2019-12-04 10:36:14 +01:00
										 |  |  |     int img_off; | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     size_t sz; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     name = fdt_getprop(itb, cfg, "fdt", NULL); | 
					
						
							|  |  |  |     if (!name) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     load_data = data = fit_load_image_alloc(itb, name, &img_off, &sz, errp); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     if (!data) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_prepend(errp, "unable to load FDT image from FIT: "); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-04 10:36:14 +01:00
										 |  |  |     ret = fit_image_addr(itb, img_off, "load", &load_addr, &err); | 
					
						
							|  |  |  |     if (ret == -ENOENT) { | 
					
						
							| 
									
										
										
										
											2018-06-25 09:41:57 -03:00
										 |  |  |         load_addr = ROUND_UP(kernel_end, 64 * KiB) + (10 * MiB); | 
					
						
							| 
									
										
										
										
											2019-12-04 10:36:14 +01:00
										 |  |  |         error_free(err); | 
					
						
							|  |  |  |     } else if (ret) { | 
					
						
							|  |  |  |         error_propagate_prepend(errp, err, | 
					
						
							|  |  |  |                                 "unable to read FDT load address from FIT: "); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ldr->fdt_filter) { | 
					
						
							|  |  |  |         load_data = ldr->fdt_filter(opaque, data, match_data, &load_addr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     load_addr = ldr->addr_to_phys(opaque, load_addr); | 
					
						
							|  |  |  |     sz = fdt_totalsize(load_data); | 
					
						
							|  |  |  |     rom_add_blob_fixed(name, load_data, sz, load_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = 0; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  |     g_free((void *) data); | 
					
						
							|  |  |  |     if (data != load_data) { | 
					
						
							|  |  |  |         g_free((void *) load_data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool fit_cfg_compatible(const void *itb, int cfg, const char *compat) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const void *fdt; | 
					
						
							|  |  |  |     const char *fdt_name; | 
					
						
							|  |  |  |     bool ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fdt_name = fdt_getprop(itb, cfg, "fdt", NULL); | 
					
						
							|  |  |  |     if (!fdt_name) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     fdt = fit_load_image_alloc(itb, fdt_name, NULL, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     if (!fdt) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fdt_check_header(fdt)) { | 
					
						
							|  |  |  |         ret = false; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fdt_node_check_compatible(fdt, 0, compat)) { | 
					
						
							|  |  |  |         ret = false; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = true; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  |     g_free((void *) fdt); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int load_fit(const struct fit_loader *ldr, const char *filename, void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     Error *err = NULL; | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     const struct fit_loader_match *match; | 
					
						
							|  |  |  |     const void *itb, *match_data = NULL; | 
					
						
							|  |  |  |     const char *def_cfg_name; | 
					
						
							|  |  |  |     char path[FIT_LOADER_MAX_PATH]; | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     int itb_size, configs, cfg_off, off; | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |     hwaddr kernel_end; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     itb = load_device_tree(filename, &itb_size); | 
					
						
							|  |  |  |     if (!itb) { | 
					
						
							|  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     configs = fdt_path_offset(itb, "/configurations"); | 
					
						
							|  |  |  |     if (configs < 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_report("can't find node /configurations"); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         ret = configs; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cfg_off = -FDT_ERR_NOTFOUND; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ldr->matches) { | 
					
						
							|  |  |  |         for (match = ldr->matches; match->compatible; match++) { | 
					
						
							|  |  |  |             off = fdt_first_subnode(itb, configs); | 
					
						
							|  |  |  |             while (off >= 0) { | 
					
						
							|  |  |  |                 if (fit_cfg_compatible(itb, off, match->compatible)) { | 
					
						
							|  |  |  |                     cfg_off = off; | 
					
						
							|  |  |  |                     match_data = match->data; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 off = fdt_next_subnode(itb, off); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (cfg_off >= 0) { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cfg_off < 0) { | 
					
						
							|  |  |  |         def_cfg_name = fdt_getprop(itb, configs, "default", NULL); | 
					
						
							|  |  |  |         if (def_cfg_name) { | 
					
						
							|  |  |  |             snprintf(path, sizeof(path), "/configurations/%s", def_cfg_name); | 
					
						
							|  |  |  |             cfg_off = fdt_path_offset(itb, path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cfg_off < 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |         error_report("can't find configuration"); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         ret = cfg_off; | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     ret = fit_load_kernel(ldr, itb, cfg_off, opaque, &kernel_end, &err); | 
					
						
							|  |  |  |     if (ret) { | 
					
						
							|  |  |  |         error_report_err(err); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 21:06:29 +02:00
										 |  |  |     ret = fit_load_fdt(ldr, itb, cfg_off, opaque, match_data, kernel_end, | 
					
						
							|  |  |  |                        &err); | 
					
						
							|  |  |  |     if (ret) { | 
					
						
							|  |  |  |         error_report_err(err); | 
					
						
							| 
									
										
										
										
											2016-09-08 15:51:56 +01:00
										 |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = 0; | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  |     g_free((void *) itb); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } |