2019-06-20 23:19:41 +02:00
|
|
|
// SPDX-License-Identifier: LGPL-2.1-or-later
|
2006-11-27 06:21:28 +01:00
|
|
|
/*
|
|
|
|
* libfdt - Flat Device Tree manipulation
|
|
|
|
* Testcase for fdt_subnode_offset()
|
|
|
|
* Copyright (C) 2006 David Gibson, IBM Corporation.
|
|
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2007-03-23 05:16:54 +01:00
|
|
|
#include <stdint.h>
|
2006-11-27 06:21:28 +01:00
|
|
|
|
|
|
|
#include <libfdt.h>
|
|
|
|
|
|
|
|
#include "tests.h"
|
|
|
|
#include "testdata.h"
|
|
|
|
|
2008-08-04 07:30:13 +02:00
|
|
|
static int check_subnode(struct fdt_header *fdt, int parent, const char *name)
|
2006-11-27 06:21:28 +01:00
|
|
|
{
|
|
|
|
int offset;
|
2007-11-19 07:26:22 +01:00
|
|
|
const struct fdt_node_header *nh;
|
2006-11-27 06:21:28 +01:00
|
|
|
uint32_t tag;
|
|
|
|
|
|
|
|
verbose_printf("Checking subnode \"%s\" of %d...", name, parent);
|
|
|
|
offset = fdt_subnode_offset(fdt, parent, name);
|
|
|
|
verbose_printf("offset %d...", offset);
|
2006-12-15 05:12:51 +01:00
|
|
|
if (offset < 0)
|
|
|
|
FAIL("fdt_subnode_offset(\"%s\"): %s", name, fdt_strerror(offset));
|
2007-11-19 07:26:22 +01:00
|
|
|
nh = fdt_offset_ptr(fdt, offset, sizeof(*nh));
|
2006-11-27 06:21:28 +01:00
|
|
|
verbose_printf("pointer %p\n", nh);
|
|
|
|
if (! nh)
|
|
|
|
FAIL("NULL retrieving subnode \"%s\"", name);
|
|
|
|
|
|
|
|
tag = fdt32_to_cpu(nh->tag);
|
|
|
|
|
|
|
|
if (tag != FDT_BEGIN_NODE)
|
|
|
|
FAIL("Incorrect tag 0x%08x on property \"%s\"", tag, name);
|
2007-09-28 07:51:04 +02:00
|
|
|
if (!nodename_eq(nh->name, name))
|
2006-11-27 06:21:28 +01:00
|
|
|
FAIL("Subnode name mismatch \"%s\" instead of \"%s\"",
|
|
|
|
nh->name, name);
|
|
|
|
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2006-12-15 05:12:47 +01:00
|
|
|
void *fdt;
|
2006-11-27 06:21:28 +01:00
|
|
|
int subnode1_offset, subnode2_offset;
|
2007-09-28 07:51:04 +02:00
|
|
|
int subsubnode1_offset, subsubnode2_offset, subsubnode2_offset2;
|
2011-09-12 03:18:43 +02:00
|
|
|
int ss12_off, ss21_off;
|
2006-11-27 06:21:28 +01:00
|
|
|
|
|
|
|
test_init(argc, argv);
|
2006-11-28 07:20:01 +01:00
|
|
|
fdt = load_blob_arg(argc, argv);
|
2006-11-27 06:21:28 +01:00
|
|
|
|
2007-09-28 07:51:04 +02:00
|
|
|
subnode1_offset = check_subnode(fdt, 0, "subnode@1");
|
|
|
|
subnode2_offset = check_subnode(fdt, 0, "subnode@2");
|
2006-11-27 06:21:28 +01:00
|
|
|
|
|
|
|
if (subnode1_offset == subnode2_offset)
|
|
|
|
FAIL("Different subnodes have same offset");
|
|
|
|
|
libfdt: Abolish _typed() variants, add _cell() variants
In a number of places through libfdt and its tests, we have *_typed()
macro variants on functions which use gcc's typeof and statement
expression extensions to allow passing literals where the underlying
function takes a buffer and size.
These seemed like a good idea at the time, but in fact they have some
problems. They use typeof and statement expressions, extensions I'd
prefer to avoid for portability. Plus, they have potential gotchas -
although they'll deal with the size of the thing passed, they won't
deal with other representation issues (like endianness) and results
could be very strange if the type of the expression passed isn't what
you think it is.
In fact, the only users of these _typed() macros were when the value
passed is a single cell (32-bit integer). Therefore, this patch
removes all these _typed() macros and replaces them with explicit
_cell() variants which handle a single 32-bit integer, and which also
perform endian convesions as appropriate.
With this in place, it now becomes easy to use standardized big-endian
representation for integer valued properties in the testcases,
regardless of the platform we're running on. We therefore do that,
which has the additional advantage that all the example trees created
during a test run are now byte-for-byte identical regardless of
platform.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2007-11-20 03:35:46 +01:00
|
|
|
check_property_cell(fdt, subnode1_offset, "prop-int", TEST_VALUE_1);
|
|
|
|
check_property_cell(fdt, subnode2_offset, "prop-int", TEST_VALUE_2);
|
2006-11-27 06:21:28 +01:00
|
|
|
|
|
|
|
subsubnode1_offset = check_subnode(fdt, subnode1_offset, "subsubnode");
|
2007-09-28 07:51:04 +02:00
|
|
|
subsubnode2_offset = check_subnode(fdt, subnode2_offset, "subsubnode@0");
|
|
|
|
subsubnode2_offset2 = check_subnode(fdt, subnode2_offset, "subsubnode");
|
2006-11-27 06:21:28 +01:00
|
|
|
|
libfdt: Abolish _typed() variants, add _cell() variants
In a number of places through libfdt and its tests, we have *_typed()
macro variants on functions which use gcc's typeof and statement
expression extensions to allow passing literals where the underlying
function takes a buffer and size.
These seemed like a good idea at the time, but in fact they have some
problems. They use typeof and statement expressions, extensions I'd
prefer to avoid for portability. Plus, they have potential gotchas -
although they'll deal with the size of the thing passed, they won't
deal with other representation issues (like endianness) and results
could be very strange if the type of the expression passed isn't what
you think it is.
In fact, the only users of these _typed() macros were when the value
passed is a single cell (32-bit integer). Therefore, this patch
removes all these _typed() macros and replaces them with explicit
_cell() variants which handle a single 32-bit integer, and which also
perform endian convesions as appropriate.
With this in place, it now becomes easy to use standardized big-endian
representation for integer valued properties in the testcases,
regardless of the platform we're running on. We therefore do that,
which has the additional advantage that all the example trees created
during a test run are now byte-for-byte identical regardless of
platform.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2007-11-20 03:35:46 +01:00
|
|
|
check_property_cell(fdt, subsubnode1_offset, "prop-int", TEST_VALUE_1);
|
|
|
|
check_property_cell(fdt, subsubnode2_offset, "prop-int", TEST_VALUE_2);
|
|
|
|
check_property_cell(fdt, subsubnode2_offset2, "prop-int", TEST_VALUE_2);
|
2007-09-28 07:51:04 +02:00
|
|
|
|
|
|
|
if (subsubnode2_offset != subsubnode2_offset2)
|
|
|
|
FAIL("Different offsets with and without unit address");
|
2006-11-27 06:21:28 +01:00
|
|
|
|
2011-09-12 03:18:43 +02:00
|
|
|
check_subnode(fdt, subnode1_offset, "ss1");
|
2008-10-30 03:41:08 +01:00
|
|
|
ss21_off = fdt_subnode_offset(fdt, subnode2_offset, "ss1");
|
|
|
|
if (ss21_off != -FDT_ERR_NOTFOUND)
|
|
|
|
FAIL("Incorrectly found ss1 in subnode2");
|
|
|
|
|
|
|
|
ss12_off = fdt_subnode_offset(fdt, subnode1_offset, "ss2");
|
|
|
|
if (ss12_off != -FDT_ERR_NOTFOUND)
|
|
|
|
FAIL("Incorrectly found ss2 in subnode1");
|
2011-09-12 03:18:43 +02:00
|
|
|
check_subnode(fdt, subnode2_offset, "ss2");
|
2008-10-30 03:41:08 +01:00
|
|
|
|
2006-11-27 06:21:28 +01:00
|
|
|
PASS();
|
|
|
|
}
|