blob: e71c1b9e9920c9687df70c960a46b8b0834888f4 [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
9#include <common.h>
10#include <binman.h>
11#include <dm.h>
Simon Glass0f2af882020-05-10 11:40:05 -060012#include <log.h>
Simon Glass9bc15642020-02-03 07:36:16 -070013#include <malloc.h>
Simon Glassc4d9faf2020-07-07 21:32:04 -060014#include <mapmem.h>
Simon Glassdd7fb9b2019-12-06 21:41:34 -070015
Simon Glassc0ef7892020-07-07 21:32:02 -060016/**
17 * struct binman_info - Information needed by the binman library
18 *
19 * @image: Node describing the image we are running from
20 * @rom_offset: Offset from an image_pos to the memory-mapped address, or
21 * ROM_OFFSET_NONE if the ROM is not memory-mapped. Can be positive or
22 * negative
23 */
Simon Glassdd7fb9b2019-12-06 21:41:34 -070024struct binman_info {
25 ofnode image;
Simon Glassc0ef7892020-07-07 21:32:02 -060026 int rom_offset;
Simon Glassdd7fb9b2019-12-06 21:41:34 -070027};
28
Simon Glassc0ef7892020-07-07 21:32:02 -060029#define ROM_OFFSET_NONE (-1)
30
Simon Glassdd7fb9b2019-12-06 21:41:34 -070031static struct binman_info *binman;
32
Simon Glassc7380c62020-07-07 21:32:03 -060033static int binman_entry_find_internal(ofnode node, const char *name,
34 struct binman_entry *entry)
Simon Glassdd7fb9b2019-12-06 21:41:34 -070035{
Simon Glassdd7fb9b2019-12-06 21:41:34 -070036 int ret;
37
Simon Glassdd7fb9b2019-12-06 21:41:34 -070038 if (!ofnode_valid(node))
Simon Glassc7380c62020-07-07 21:32:03 -060039 node = binman->image;
40 node = ofnode_find_subnode(node, name);
41 if (!ofnode_valid(node))
42 return log_msg_ret("node", -ENOENT);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070043
44 ret = ofnode_read_u32(node, "image-pos", &entry->image_pos);
45 if (ret)
Simon Glassdff2ea12020-09-27 18:46:19 -060046 return log_msg_ret("image-pos", ret);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070047 ret = ofnode_read_u32(node, "size", &entry->size);
48 if (ret)
Simon Glassc7380c62020-07-07 21:32:03 -060049 return log_msg_ret("size", ret);
Simon Glassdd7fb9b2019-12-06 21:41:34 -070050
51 return 0;
52}
53
Simon Glassc7380c62020-07-07 21:32:03 -060054int binman_entry_find(const char *name, struct binman_entry *entry)
55{
56 return binman_entry_find_internal(binman->image, name, entry);
57}
58
Simon Glassc4d9faf2020-07-07 21:32:04 -060059int binman_entry_map(ofnode parent, const char *name, void **bufp, int *sizep)
60{
61 struct binman_entry entry;
62 int ret;
63
64 if (binman->rom_offset == ROM_OFFSET_NONE)
65 return -EPERM;
66 ret = binman_entry_find_internal(parent, name, &entry);
67 if (ret)
68 return log_msg_ret("entry", ret);
69 if (sizep)
70 *sizep = entry.size;
71 *bufp = map_sysmem(entry.image_pos + binman->rom_offset, entry.size);
72
73 return 0;
74}
75
76ofnode binman_section_find_node(const char *name)
77{
78 return ofnode_find_subnode(binman->image, name);
79}
80
Simon Glassc0ef7892020-07-07 21:32:02 -060081void binman_set_rom_offset(int rom_offset)
82{
83 binman->rom_offset = rom_offset;
84}
85
Simon Glassdff2ea12020-09-27 18:46:19 -060086int binman_get_rom_offset(void)
87{
88 return binman->rom_offset;
89}
90
Simon Glassdd7fb9b2019-12-06 21:41:34 -070091int binman_init(void)
92{
93 binman = malloc(sizeof(struct binman_info));
94 if (!binman)
95 return log_msg_ret("space for binman", -ENOMEM);
96 binman->image = ofnode_path("/binman");
97 if (!ofnode_valid(binman->image))
98 return log_msg_ret("binman node", -EINVAL);
Simon Glass4583ec32020-09-27 18:46:20 -060099 if (ofnode_read_bool(binman->image, "multiple-images")) {
100 ofnode node = ofnode_first_subnode(binman->image);
101
102 if (!ofnode_valid(node))
103 return log_msg_ret("first image", -ENOENT);
104 binman->image = node;
105 }
Simon Glassc0ef7892020-07-07 21:32:02 -0600106 binman->rom_offset = ROM_OFFSET_NONE;
Simon Glassdd7fb9b2019-12-06 21:41:34 -0700107
108 return 0;
109}