#pragma ident "%Z%%M% %I% %E% SMI"
#include <sys/promif.h>
#include <sys/promimpl.h>
pnode_t
prom_nextnode(pnode_t nodeid)
{
cell_t ci[5];
ci[0] = p1275_ptr2cell("peer");
ci[1] = (cell_t)1;
ci[2] = (cell_t)1;
ci[3] = p1275_dnode2cell(nodeid);
ci[4] = p1275_dnode2cell(OBP_NONODE);
promif_preprom();
(void) p1275_cif_handler(&ci);
promif_postprom();
return (p1275_cell2dnode(ci[4]));
}
pnode_t
prom_childnode(pnode_t nodeid)
{
cell_t ci[5];
ci[0] = p1275_ptr2cell("child");
ci[1] = (cell_t)1;
ci[2] = (cell_t)1;
ci[3] = p1275_dnode2cell(nodeid);
ci[4] = p1275_dnode2cell(OBP_NONODE);
promif_preprom();
(void) p1275_cif_handler(&ci);
promif_postprom();
return (p1275_cell2dnode(ci[4]));
}
void
prom_walk_devs(pnode_t node, int (*cb)(pnode_t, void *, void *), void *arg,
void *result)
{
pnode_t stack[OBP_STACKDEPTH];
int stackidx = 0;
if (node == OBP_NONODE || node == OBP_BADNODE) {
prom_panic("Invalid node specified as root of prom tree walk");
}
stack[0] = node;
for (;;) {
pnode_t curnode = stack[stackidx];
pnode_t child;
if (curnode == OBP_NONODE || curnode == OBP_BADNODE) {
stackidx--;
if (stackidx < 0)
return;
stack[stackidx] = prom_nextnode(stack[stackidx]);
continue;
}
switch ((*cb)(curnode, arg, result)) {
case PROM_WALK_TERMINATE:
return;
case PROM_WALK_CONTINUE:
child = prom_childnode(curnode);
if (child != OBP_NONODE && child != OBP_BADNODE) {
stackidx++;
stack[stackidx] = child;
} else {
stack[stackidx] =
prom_nextnode(stack[stackidx]);
}
break;
default:
prom_panic("unrecognized walk directive");
}
}
}
static int
bytype_cb(pnode_t node, void *arg, void *result)
{
if (prom_devicetype(node, (char *)arg)) {
*((pnode_t *)result) = node;
return (PROM_WALK_TERMINATE);
}
return (PROM_WALK_CONTINUE);
}
pnode_t
prom_findnode_bydevtype(pnode_t node, char *devtype)
{
pnode_t result = OBP_NONODE;
prom_walk_devs(node, bytype_cb, devtype, &result);
return (result);
}
static int
byname_cb(pnode_t node, void *arg, void *result)
{
if (prom_getnode_byname(node, (char *)arg)) {
*((pnode_t *)result) = node;
return (PROM_WALK_TERMINATE);
}
return (PROM_WALK_CONTINUE);
}
pnode_t
prom_findnode_byname(pnode_t node, char *name)
{
pnode_t result = OBP_NONODE;
prom_walk_devs(node, byname_cb, name, &result);
return (result);
}
pnode_t
prom_rootnode(void)
{
static pnode_t rootnode;
return (rootnode ? rootnode : (rootnode = prom_nextnode(OBP_NONODE)));
}
pnode_t
prom_parentnode(pnode_t nodeid)
{
cell_t ci[5];
ci[0] = p1275_ptr2cell("parent");
ci[1] = (cell_t)1;
ci[2] = (cell_t)1;
ci[3] = p1275_dnode2cell(nodeid);
ci[4] = p1275_dnode2cell(OBP_NONODE);
promif_preprom();
(void) p1275_cif_handler(&ci);
promif_postprom();
return (p1275_cell2dnode(ci[4]));
}
pnode_t
prom_finddevice(char *path)
{
cell_t ci[5];
#ifdef PROM_32BIT_ADDRS
char *opath = NULL;
size_t len;
if ((uintptr_t)path > (uint32_t)-1) {
opath = path;
len = prom_strlen(opath) + 1;
path = promplat_alloc(len);
if (path == NULL) {
return (OBP_BADNODE);
}
(void) prom_strcpy(path, opath);
}
#endif
promif_preprom();
ci[0] = p1275_ptr2cell("finddevice");
ci[1] = (cell_t)1;
ci[2] = (cell_t)1;
ci[3] = p1275_ptr2cell(path);
ci[4] = p1275_dnode2cell(OBP_BADNODE);
(void) p1275_cif_handler(&ci);
promif_postprom();
#ifdef PROM_32BIT_ADDRS
if (opath != NULL)
promplat_free(path, len);
#endif
return ((pnode_t)p1275_cell2dnode(ci[4]));
}
pnode_t
prom_chosennode(void)
{
static pnode_t chosen;
pnode_t node;
if (chosen)
return (chosen);
node = prom_finddevice("/chosen");
if (node != OBP_BADNODE)
return (chosen = node);
prom_fatal_error("prom_chosennode: Can't find </chosen>\n");
return ((pnode_t)0);
}
pnode_t
prom_alias_node(void)
{
static pnode_t node;
if (node == 0)
node = prom_finddevice("/aliases");
return (node);
}
pnode_t
prom_optionsnode(void)
{
static pnode_t node;
if (node == 0)
node = prom_finddevice("/options");
return (node);
}