blob: 9047f5275f3250ae1106b859f311a16e0c34cbf6 [file] [log] [blame]
Simon Glassdd7fb9b2019-12-06 21:41:34 -07001// SPDX-License-Identifier: Intel
2/*
3 * Access to binman information at runtime
4 *
5 * Copyright 2019 Google LLC
6 * Written by Simon Glass <sjg@chromium.org>
7 */
8
Simon Glassdd7fb9b2019-12-06 21:41:34 -07009#include <binman.h>
10#include <dm.h>
Simon Glass0f2af882020-05-10 11:40:05 -060011#include <log.h>
Simon Glass9bc15642020-02-03 07:36:16 -070012#include <malloc.h>
Simon Glassc4d9faf2020-07-07 21:32:04 -060013#include <mapmem.h>
Simon Glassdd7fb9b2019-12-06 21:41:34 -070014
Simon Glassc0ef7892020-07-07 21:32:02 -060015/**
16 * struct binman_info - Information needed by the binman library
17 *
18 * @image: Node describing the image we are running from
19 * @rom_offset: Offset from an image_pos to the memory-mapped address, or
20 * ROM_OFFSET_NONE if the ROM is not memory-mapped. Can be positive or
21 * negative
22 */
Simon Glassdd7fb9b2019-12-06 21:41:34 -070023struct binman_info {
24 ofnode image;
Simon Glassc0ef7892020-07-07 21:32:02 -060025 int rom_offset;
Simon Glassdd7fb9b2019-12-06 21:41:34 -070026};
27
Simon Glassc0ef7892020-07-07 21:32:02 -060028#define ROM_OFFSET_NONE (-1)
29
Simon Glassdd7fb9b2019-12-06 21:41:34 -070030static struct binman_info *binman;
31
Simon Glasscb31d762021-01-13 20:29:56 -070032/**
33 * find_image_node() - Find the top-level binman node
34 *
35 * Finds the binman node which can be used to load entries. The correct node
36 * depends on whether multiple-images is in use.
37 *
38 * @nodep: Returns the node found, on success
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +010039 * Return: 0 if OK, , -EINVAL if there is no /binman node, -ECHILD if multiple
Simon Glasscb31d762021-01-13 20:29:56 -070040 * images are being used but the first image is not available
41 */
42static int find_image_node(ofnode *nodep)
43{
44 ofnode node;
45
46 node = ofnode_path("/binman");
47 if (!ofnode_valid(node))
48 return log_msg_ret("binman node", -EINVAL);
49 if (ofnode_read_bool(node, "multiple-images")) {
50 node = ofnode_first_subnode(node);
51
52 if (!ofnode_valid(node))
53 return log_msg_ret("first image", -ECHILD);
54 }
55 *nodep = node;
56
57 return 0;
58}
59
Simon Glassc7380c62020-07-07 21:32:03 -060060static int binman_entry_find_internal(ofnode node, const char *name,
61 struct binman_entry *entry)
Simon Glassdd7fb9b2019-12-06 21:41:34 -070062{
Simon Glassdd7fb9b2019-12-06 21:41:34 -070063 int ret;
64
Simon Glassdd7fb9b2019-12-06 21:41:34 -070065 if (!ofnode_valid(node))
Simon Glassc7380c62020-07-07 21:32:03 -060066 node = binman->image;
67 node = ofnode_find_subnode(node, name);
68 if (!ofnode_valid(node))
69 return log_msg_ret("node", -ENOENT);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070070
71 ret = ofnode_read_u32(node, "image-pos", &entry->image_pos);
72 if (ret)
Simon Glassdff2ea12020-09-27 18:46:19 -060073 return log_msg_ret("image-pos", ret);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070074 ret = ofnode_read_u32(node, "size", &entry->size);
75 if (ret)
Simon Glassc7380c62020-07-07 21:32:03 -060076 return log_msg_ret("size", ret);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070077
78 return 0;
79}
80
Simon Glassc7380c62020-07-07 21:32:03 -060081int binman_entry_find(const char *name, struct binman_entry *entry)
82{
83 return binman_entry_find_internal(binman->image, name, entry);
84}
85
Simon Glassc4d9faf2020-07-07 21:32:04 -060086int binman_entry_map(ofnode parent, const char *name, void **bufp, int *sizep)
87{
88 struct binman_entry entry;
89 int ret;
90
91 if (binman->rom_offset == ROM_OFFSET_NONE)
92 return -EPERM;
93 ret = binman_entry_find_internal(parent, name, &entry);
94 if (ret)
95 return log_msg_ret("entry", ret);
96 if (sizep)
97 *sizep = entry.size;
98 *bufp = map_sysmem(entry.image_pos + binman->rom_offset, entry.size);
99
100 return 0;
101}
102
103ofnode binman_section_find_node(const char *name)
104{
105 return ofnode_find_subnode(binman->image, name);
106}
107
Simon Glassc0ef7892020-07-07 21:32:02 -0600108void binman_set_rom_offset(int rom_offset)
109{
110 binman->rom_offset = rom_offset;
111}
112
Simon Glassdff2ea12020-09-27 18:46:19 -0600113int binman_get_rom_offset(void)
114{
115 return binman->rom_offset;
116}
117
Simon Glass23c0ea52021-01-13 20:29:57 -0700118int binman_select_subnode(const char *name)
119{
120 ofnode node;
121 int ret;
122
123 ret = find_image_node(&node);
124 if (ret)
125 return log_msg_ret("main", -ENOENT);
126 node = ofnode_find_subnode(node, name);
127 if (!ofnode_valid(node))
128 return log_msg_ret("node", -ENOENT);
129 binman->image = node;
Simon Glass5dc338f2021-03-15 18:11:14 +1300130 log_info("binman: Selected image subnode '%s'\n",
131 ofnode_get_name(binman->image));
Simon Glass23c0ea52021-01-13 20:29:57 -0700132
133 return 0;
134}
135
Simon Glassdd7fb9b2019-12-06 21:41:34 -0700136int binman_init(void)
137{
Simon Glasscb31d762021-01-13 20:29:56 -0700138 int ret;
139
Simon Glassdd7fb9b2019-12-06 21:41:34 -0700140 binman = malloc(sizeof(struct binman_info));
141 if (!binman)
142 return log_msg_ret("space for binman", -ENOMEM);
Simon Glasscb31d762021-01-13 20:29:56 -0700143 ret = find_image_node(&binman->image);
144 if (ret)
145 return log_msg_ret("node", -ENOENT);
Michal Simekf3d93c62020-08-26 15:13:14 +0200146 binman_set_rom_offset(ROM_OFFSET_NONE);
Simon Glassf91f23c2021-01-21 13:57:16 -0700147 log_debug("binman: Selected image node '%s'\n",
148 ofnode_get_name(binman->image));
Michal Simekb6f4b9f2020-11-23 09:08:19 +0100149
Simon Glassdd7fb9b2019-12-06 21:41:34 -0700150 return 0;
151}