Subversion Repositories HelenOS

Rev

Rev 1899 | Rev 1972 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
1894 jermar 1
/*
2
 * Copyright (C) 2006 Jakub Jermar
3
 * All rights reserved.
4
 *
5
 * Redistribution and use in source and binary forms, with or without
6
 * modification, are permitted provided that the following conditions
7
 * are met:
8
 *
9
 * - Redistributions of source code must retain the above copyright
10
 *   notice, this list of conditions and the following disclaimer.
11
 * - Redistributions in binary form must reproduce the above copyright
12
 *   notice, this list of conditions and the following disclaimer in the
13
 *   documentation and/or other materials provided with the distribution.
14
 * - The name of the author may not be used to endorse or promote products
15
 *   derived from this software without specific prior written permission.
16
 *
17
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
 */
28
 
29
/** @addtogroup ofw
30
 * @{
31
 */
32
/**
33
 * @file
34
 * @brief   OpenFirmware device tree navigation.
35
 *
36
 */
37
 
1895 jermar 38
#include <genarch/ofw/ofw_tree.h>
39
#include <arch/memstr.h>
40
#include <func.h>
41
#include <print.h>
42
#include <panic.h>
43
 
44
#define PATH_MAX_LEN    80
45
#define NAME_BUF_LEN    50
46
 
47
static ofw_tree_node_t *ofw_root;
48
 
49
void ofw_tree_init(ofw_tree_node_t *root)
50
{
51
    ofw_root = root;
52
}
53
 
1896 jermar 54
/** Get OpenFirmware node property.
55
 *
56
 * @param node Node in which to lookup the property.
57
 * @param name Name of the property.
58
 *
59
 * @return Pointer to the property structure or NULL if no such property.
60
 */
61
ofw_tree_property_t *ofw_tree_getprop(const ofw_tree_node_t *node, const char *name)
62
{
63
    int i;
64
 
65
    for (i = 0; i < node->properties; i++) {
66
        if (strcmp(node->property[i].name, name) == 0)
67
            return &node->property[i];
68
    }
69
 
70
    return NULL;
71
}
72
 
1895 jermar 73
/** Return value of the 'name' property.
74
 *
75
 * @param node Node of interest.
76
 *
77
 * @return Value of the 'name' property belonging to the node.
78
 */
79
const char *ofw_tree_node_name(const ofw_tree_node_t *node)
80
{
1896 jermar 81
    ofw_tree_property_t *prop;
1895 jermar 82
 
1896 jermar 83
    prop = ofw_tree_getprop(node, "name");
84
    if (!prop)
85
        panic("Node without name property.\n");
86
 
87
    if (prop->size < 2)
88
        panic("Invalid name property.\n");
1895 jermar 89
 
1896 jermar 90
    return prop->value;
1895 jermar 91
}
92
 
93
/** Lookup child of given name.
94
 *
95
 * @param node Node whose child is being looked up.
1896 jermar 96
 * @param name Name of the child being looked up.
1895 jermar 97
 *
98
 * @return NULL if there is no such child or pointer to the matching child node.
99
 */
1899 jermar 100
ofw_tree_node_t *ofw_tree_find_child(ofw_tree_node_t *node, const char *name)
1895 jermar 101
{
102
    ofw_tree_node_t *cur;
103
 
1896 jermar 104
    /*
105
     * Try to find the disambigued name.
106
     */
1895 jermar 107
    for (cur = node->child; cur; cur = cur->peer) {
1896 jermar 108
        if (strcmp(cur->da_name, name) == 0)
1895 jermar 109
            return cur;
110
    }
111
 
1896 jermar 112
    /*
113
     * Disambigued name not found.
114
     * Lets try our luck with possibly ambiguous "name" property.
115
     *
116
     * We need to do this because paths stored in "/aliases"
117
     * are not always fully-qualified.
118
     */
119
    for (cur = node->child; cur; cur = cur->peer) {
120
        if (strcmp(ofw_tree_node_name(cur), name) == 0)
121
            return cur;
122
    }
123
 
1895 jermar 124
    return NULL;
125
}
126
 
1899 jermar 127
/** Lookup first child of given device type.
128
 *
129
 * @param node Node whose child is being looked up.
130
 * @param name Device type of the child being looked up.
131
 *
132
 * @return NULL if there is no such child or pointer to the matching child node.
133
 */
134
ofw_tree_node_t *ofw_tree_find_child_by_device_type(ofw_tree_node_t *node, const char *name)
135
{
136
    ofw_tree_node_t *cur;
137
    ofw_tree_property_t *prop;
138
 
139
    for (cur = node->child; cur; cur = cur->peer) {
140
        prop = ofw_tree_getprop(cur, "device_type");
141
        if (!prop || !prop->value)
142
            continue;
143
        if (strcmp(prop->value, name) == 0)
144
            return cur;
145
    }
146
 
147
    return NULL;
148
}
149
 
1908 jermar 150
/** Lookup node with matching node_handle.
151
 *
152
 * @param root Root of the searched subtree.
153
 * @param handle OpenFirmware handle.
154
 *
155
 * @return NULL if there is no such node or pointer to the matching node.
156
 */
157
ofw_tree_node_t *ofw_tree_find_node_by_handle(ofw_tree_node_t *root, uint32_t handle)
158
{
159
    ofw_tree_node_t *node;
160
 
161
    if (root->node_handle == handle)
162
        return root;
163
 
164
    if (root->peer) {
165
        node = ofw_tree_find_node_by_handle(root->peer, handle);
166
        if (node)
167
            return node;
168
    }
169
 
170
    if (root->child) {
171
        node = ofw_tree_find_node_by_handle(root->child, handle);
172
        if (node)
173
            return node;
174
    }
175
 
176
    return NULL;   
177
}
178
 
1899 jermar 179
/** Lookup first peer of given device type.
180
 *
181
 * @param node Node whose peer is being looked up.
182
 * @param name Device type of the child being looked up.
183
 *
184
 * @return NULL if there is no such child or pointer to the matching child node.
185
 */
186
ofw_tree_node_t *ofw_tree_find_peer_by_device_type(ofw_tree_node_t *node, const char *name)
187
{
188
    ofw_tree_node_t *cur;
189
    ofw_tree_property_t *prop;
190
 
191
    for (cur = node->peer; cur; cur = cur->peer) {
192
        prop = ofw_tree_getprop(cur, "device_type");
193
        if (!prop || !prop->value)
194
            continue;
195
        if (strcmp(prop->value, name) == 0)
196
            return cur;
197
    }
198
 
199
    return NULL;
200
}
201
 
202
 
1895 jermar 203
/** Lookup OpenFirmware node by its path.
204
 *
205
 * @param path Path to the node.
206
 *
207
 * @return NULL if there is no such node or pointer to the leaf node.
208
 */
209
ofw_tree_node_t *ofw_tree_lookup(const char *path)
210
{
211
    char buf[NAME_BUF_LEN+1];
212
    ofw_tree_node_t *node = ofw_root;
213
    index_t i, j;
214
 
215
    if (path[0] != '/')
216
        return NULL;
217
 
218
    for (i = 1; i < strlen(path) && node; i = j + 1) {
219
        for (j = i; j < strlen(path) && path[j] != '/'; j++)
220
            ;
221
        if (i == j) /* skip extra slashes */
222
            continue;
223
 
224
        memcpy(buf, &path[i], j - i);
225
        buf[j - i] = '\0';
226
        node = ofw_tree_find_child(node, buf);
227
    }
228
 
229
    return node;
230
}
231
 
232
/** Recursively print subtree rooted in a node.
233
 *
234
 * @param node Root of the subtree.
235
 * @param path Current path, NULL for the very root of the entire tree.
236
 */
237
static void ofw_tree_node_print(const ofw_tree_node_t *node, const char *path)
238
{
239
    char p[PATH_MAX_LEN];
240
 
241
    if (node->parent) {
242
        snprintf(p, PATH_MAX_LEN, "%s/%s", path, node->da_name);
243
        printf("%s\n", p);
244
    } else {
245
        snprintf(p, PATH_MAX_LEN, "%s", node->da_name);
246
        printf("/\n");
247
    }
248
 
249
    if (node->child)
250
        ofw_tree_node_print(node->child, p);
251
 
252
    if (node->peer)
253
        ofw_tree_node_print(node->peer, path);
254
}
255
 
256
/** Print the structure of the OpenFirmware device tree. */
257
void ofw_tree_print(void)
258
{
259
    ofw_tree_node_print(ofw_root, NULL);
260
}
261
 
1894 jermar 262
/** @}
263
 */