blob: 751538ec1d066170070ecd521f1eeae0c72c5c27 [file] [log] [blame]
aurel32f652e6a2008-12-16 10:43:48 +00001/*
2 * Functions to help device tree manipulation using libfdt.
3 * It also provides functions to read entries from device tree proc
4 * interface.
5 *
6 * Copyright 2008 IBM Corporation.
7 * Authors: Jerone Young <jyoung5@us.ibm.com>
8 * Hollis Blanchard <hollisb@us.ibm.com>
9 *
10 * This work is licensed under the GNU GPL license version 2 or later.
11 *
12 */
13
14#include <stdio.h>
15#include <sys/types.h>
16#include <sys/stat.h>
17#include <fcntl.h>
18#include <unistd.h>
19#include <stdlib.h>
20
21#include "config.h"
22#include "qemu-common.h"
aurel32f652e6a2008-12-16 10:43:48 +000023#include "device_tree.h"
Blue Swirl39b7f202009-09-22 17:51:36 +000024#include "hw/loader.h"
aurel32f652e6a2008-12-16 10:43:48 +000025
26#include <libfdt.h>
27
pbrook7ec632b2009-04-10 16:23:59 +000028void *load_device_tree(const char *filename_path, int *sizep)
aurel32f652e6a2008-12-16 10:43:48 +000029{
pbrook7ec632b2009-04-10 16:23:59 +000030 int dt_size;
aurel32f652e6a2008-12-16 10:43:48 +000031 int dt_file_load_size;
aurel32f652e6a2008-12-16 10:43:48 +000032 int ret;
pbrook7ec632b2009-04-10 16:23:59 +000033 void *fdt = NULL;
aurel32f652e6a2008-12-16 10:43:48 +000034
pbrook7ec632b2009-04-10 16:23:59 +000035 *sizep = 0;
36 dt_size = get_image_size(filename_path);
37 if (dt_size < 0) {
aurel32f652e6a2008-12-16 10:43:48 +000038 printf("Unable to get size of device tree file '%s'\n",
39 filename_path);
40 goto fail;
41 }
42
pbrook7ec632b2009-04-10 16:23:59 +000043 /* Expand to 2x size to give enough room for manipulation. */
44 dt_size *= 2;
aurel32f652e6a2008-12-16 10:43:48 +000045 /* First allocate space in qemu for device tree */
Anthony Liguori7267c092011-08-20 22:09:37 -050046 fdt = g_malloc0(dt_size);
aurel32f652e6a2008-12-16 10:43:48 +000047
pbrook7ec632b2009-04-10 16:23:59 +000048 dt_file_load_size = load_image(filename_path, fdt);
49 if (dt_file_load_size < 0) {
50 printf("Unable to open device tree file '%s'\n",
51 filename_path);
52 goto fail;
53 }
aurel32f652e6a2008-12-16 10:43:48 +000054
pbrook7ec632b2009-04-10 16:23:59 +000055 ret = fdt_open_into(fdt, fdt, dt_size);
aurel32f652e6a2008-12-16 10:43:48 +000056 if (ret) {
57 printf("Unable to copy device tree in memory\n");
58 goto fail;
59 }
60
61 /* Check sanity of device tree */
62 if (fdt_check_header(fdt)) {
63 printf ("Device tree file loaded into memory is invalid: %s\n",
64 filename_path);
65 goto fail;
66 }
pbrook7ec632b2009-04-10 16:23:59 +000067 *sizep = dt_size;
aurel32f652e6a2008-12-16 10:43:48 +000068 return fdt;
69
70fail:
Anthony Liguori7267c092011-08-20 22:09:37 -050071 g_free(fdt);
aurel32f652e6a2008-12-16 10:43:48 +000072 return NULL;
73}
74
Alexander Grafccbcfed2011-07-23 10:52:00 +020075static int findnode_nofail(void *fdt, const char *node_path)
aurel32f652e6a2008-12-16 10:43:48 +000076{
77 int offset;
78
79 offset = fdt_path_offset(fdt, node_path);
Alexander Grafccbcfed2011-07-23 10:52:00 +020080 if (offset < 0) {
81 fprintf(stderr, "%s Couldn't find node %s: %s\n", __func__, node_path,
82 fdt_strerror(offset));
83 exit(1);
84 }
aurel32f652e6a2008-12-16 10:43:48 +000085
Alexander Grafccbcfed2011-07-23 10:52:00 +020086 return offset;
87}
88
89int qemu_devtree_setprop(void *fdt, const char *node_path,
90 const char *property, void *val_array, int size)
91{
92 int r;
93
94 r = fdt_setprop(fdt, findnode_nofail(fdt, node_path), property, val_array, size);
95 if (r < 0) {
96 fprintf(stderr, "%s: Couldn't set %s/%s: %s\n", __func__, node_path,
97 property, fdt_strerror(r));
98 exit(1);
99 }
100
101 return r;
aurel32f652e6a2008-12-16 10:43:48 +0000102}
103
104int qemu_devtree_setprop_cell(void *fdt, const char *node_path,
105 const char *property, uint32_t val)
106{
Alexander Grafccbcfed2011-07-23 10:52:00 +0200107 int r;
aurel32f652e6a2008-12-16 10:43:48 +0000108
Alexander Grafccbcfed2011-07-23 10:52:00 +0200109 r = fdt_setprop_cell(fdt, findnode_nofail(fdt, node_path), property, val);
110 if (r < 0) {
111 fprintf(stderr, "%s: Couldn't set %s/%s = %#08x: %s\n", __func__,
112 node_path, property, val, fdt_strerror(r));
113 exit(1);
114 }
aurel32f652e6a2008-12-16 10:43:48 +0000115
Alexander Grafccbcfed2011-07-23 10:52:00 +0200116 return r;
aurel32f652e6a2008-12-16 10:43:48 +0000117}
118
119int qemu_devtree_setprop_string(void *fdt, const char *node_path,
120 const char *property, const char *string)
121{
Alexander Grafccbcfed2011-07-23 10:52:00 +0200122 int r;
aurel32f652e6a2008-12-16 10:43:48 +0000123
Alexander Grafccbcfed2011-07-23 10:52:00 +0200124 r = fdt_setprop_string(fdt, findnode_nofail(fdt, node_path), property, string);
125 if (r < 0) {
126 fprintf(stderr, "%s: Couldn't set %s/%s = %s: %s\n", __func__,
127 node_path, property, string, fdt_strerror(r));
128 exit(1);
129 }
aurel32f652e6a2008-12-16 10:43:48 +0000130
Alexander Grafccbcfed2011-07-23 10:52:00 +0200131 return r;
aurel32f652e6a2008-12-16 10:43:48 +0000132}
Alexander Grafd69a8e62011-07-21 01:52:57 +0200133
134int qemu_devtree_nop_node(void *fdt, const char *node_path)
135{
Alexander Grafccbcfed2011-07-23 10:52:00 +0200136 int r;
Alexander Grafd69a8e62011-07-21 01:52:57 +0200137
Alexander Grafccbcfed2011-07-23 10:52:00 +0200138 r = fdt_nop_node(fdt, findnode_nofail(fdt, node_path));
139 if (r < 0) {
140 fprintf(stderr, "%s: Couldn't nop node %s: %s\n", __func__, node_path,
141 fdt_strerror(r));
142 exit(1);
143 }
Alexander Grafd69a8e62011-07-21 01:52:57 +0200144
Alexander Grafccbcfed2011-07-23 10:52:00 +0200145 return r;
Alexander Grafd69a8e62011-07-21 01:52:57 +0200146}
Alexander Graf80ad7812011-07-22 13:55:37 +0200147
148int qemu_devtree_add_subnode(void *fdt, const char *name)
149{
Alexander Graf80ad7812011-07-22 13:55:37 +0200150 char *dupname = g_strdup(name);
151 char *basename = strrchr(dupname, '/');
152 int retval;
153
154 if (!basename) {
155 return -1;
156 }
157
158 basename[0] = '\0';
159 basename++;
160
Alexander Grafccbcfed2011-07-23 10:52:00 +0200161 retval = fdt_add_subnode(fdt, findnode_nofail(fdt, dupname), basename);
162 if (retval < 0) {
163 fprintf(stderr, "FDT: Failed to create subnode %s: %s\n", name,
164 fdt_strerror(retval));
165 exit(1);
Alexander Graf80ad7812011-07-22 13:55:37 +0200166 }
167
Alexander Graf80ad7812011-07-22 13:55:37 +0200168 g_free(dupname);
169 return retval;
170}