Browse Source

libfdt: Add helpers for accessing unaligned words

This adds some helpers to load (32 or 64 bit) words from an fdt blob, even
if they're unaligned and we're on a platform that doesn't like plain
unaligned loads and stores.  We then use the helpers in a number of places.
There are two purposes for this:

1) This makes libfdt more robust against a blob loaded at an unaligned
   address.  It's usually good practice to load a blob at a 64-bit
   alignment, but it's nice to work even then.

2) Users can use these helpers to load integer values from within property
   values.  These can often be unaligned, even if the blob as a whole is
   aligned, since some property encodings have integers and strings mixed
   together without any alignment gaps.

Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
main
David Gibson 6 years ago
parent
commit
6dcb8ba408
  1. 2
      fdtget.c
  2. 18
      libfdt/fdt_ro.c
  3. 25
      libfdt/libfdt.h

2
fdtget.c

@ -76,7 +76,7 @@ static int show_cell_list(struct display_info *disp, const char *data, int len, @@ -76,7 +76,7 @@ static int show_cell_list(struct display_info *disp, const char *data, int len,
for (i = 0; i < len; i += size, p += size) {
if (i)
printf(" ");
value = size == 4 ? fdt32_to_cpu(*(const fdt32_t *)p) :
value = size == 4 ? fdt32_ld((const fdt32_t *)p) :
size == 2 ? (*p << 8) | p[1] : *p;
printf(fmt, value);
}

18
libfdt/fdt_ro.c

@ -191,8 +191,8 @@ int fdt_get_mem_rsv(const void *fdt, int n, uint64_t *address, uint64_t *size) @@ -191,8 +191,8 @@ int fdt_get_mem_rsv(const void *fdt, int n, uint64_t *address, uint64_t *size)
if (!re)
return -FDT_ERR_BADOFFSET;

*address = fdt64_to_cpu(re->address);
*size = fdt64_to_cpu(re->size);
*address = fdt64_ld(&re->address);
*size = fdt64_ld(&re->size);
return 0;
}

@ -202,7 +202,7 @@ int fdt_num_mem_rsv(const void *fdt) @@ -202,7 +202,7 @@ int fdt_num_mem_rsv(const void *fdt)
const struct fdt_reserve_entry *re;

for (i = 0; (re = fdt_mem_rsv(fdt, i)) != NULL; i++) {
if (fdt64_to_cpu(re->size) == 0)
if (fdt64_ld(&re->size) == 0)
return i;
}
return -FDT_ERR_TRUNCATED;
@ -379,7 +379,7 @@ static const struct fdt_property *fdt_get_property_by_offset_(const void *fdt, @@ -379,7 +379,7 @@ static const struct fdt_property *fdt_get_property_by_offset_(const void *fdt,
prop = fdt_offset_ptr_(fdt, offset);

if (lenp)
*lenp = fdt32_to_cpu(prop->len);
*lenp = fdt32_ld(&prop->len);

return prop;
}
@ -416,7 +416,7 @@ static const struct fdt_property *fdt_get_property_namelen_(const void *fdt, @@ -416,7 +416,7 @@ static const struct fdt_property *fdt_get_property_namelen_(const void *fdt,
offset = -FDT_ERR_INTERNAL;
break;
}
if (fdt_string_eq_(fdt, fdt32_to_cpu(prop->nameoff),
if (fdt_string_eq_(fdt, fdt32_ld(&prop->nameoff),
name, namelen)) {
if (poffset)
*poffset = offset;
@ -469,7 +469,7 @@ const void *fdt_getprop_namelen(const void *fdt, int nodeoffset, @@ -469,7 +469,7 @@ const void *fdt_getprop_namelen(const void *fdt, int nodeoffset,

/* Handle realignment */
if (fdt_version(fdt) < 0x10 && (poffset + sizeof(*prop)) % 8 &&
fdt32_to_cpu(prop->len) >= 8)
fdt32_ld(&prop->len) >= 8)
return prop->data + 4;
return prop->data;
}
@ -485,7 +485,7 @@ const void *fdt_getprop_by_offset(const void *fdt, int offset, @@ -485,7 +485,7 @@ const void *fdt_getprop_by_offset(const void *fdt, int offset,
if (namep) {
const char *name;
int namelen;
name = fdt_get_string(fdt, fdt32_to_cpu(prop->nameoff),
name = fdt_get_string(fdt, fdt32_ld(&prop->nameoff),
&namelen);
if (!name) {
if (lenp)
@ -497,7 +497,7 @@ const void *fdt_getprop_by_offset(const void *fdt, int offset, @@ -497,7 +497,7 @@ const void *fdt_getprop_by_offset(const void *fdt, int offset,

/* Handle realignment */
if (fdt_version(fdt) < 0x10 && (offset + sizeof(*prop)) % 8 &&
fdt32_to_cpu(prop->len) >= 8)
fdt32_ld(&prop->len) >= 8)
return prop->data + 4;
return prop->data;
}
@ -522,7 +522,7 @@ uint32_t fdt_get_phandle(const void *fdt, int nodeoffset) @@ -522,7 +522,7 @@ uint32_t fdt_get_phandle(const void *fdt, int nodeoffset)
return 0;
}

return fdt32_to_cpu(*php);
return fdt32_ld(php);
}

const char *fdt_get_alias_namelen(const void *fdt,

25
libfdt/libfdt.h

@ -154,6 +154,29 @@ static inline void *fdt_offset_ptr_w(void *fdt, int offset, int checklen) @@ -154,6 +154,29 @@ static inline void *fdt_offset_ptr_w(void *fdt, int offset, int checklen)

uint32_t fdt_next_tag(const void *fdt, int offset, int *nextoffset);

/*
* Alignment helpers:
* These helpers access words from a device tree blob. They're
* built to work even with unaligned pointers on platforms (ike
* ARM) that don't like unaligned loads and stores
*/

static inline uint32_t fdt32_ld(const fdt32_t *p)
{
fdt32_t v;

memcpy(&v, p, sizeof(v));
return fdt32_to_cpu(v);
}

static inline uint64_t fdt64_ld(const fdt64_t *p)
{
fdt64_t v;

memcpy(&v, p, sizeof(v));
return fdt64_to_cpu(v);
}

/**********************************************************************/
/* Traversal functions */
/**********************************************************************/
@ -214,7 +237,7 @@ int fdt_next_subnode(const void *fdt, int offset); @@ -214,7 +237,7 @@ int fdt_next_subnode(const void *fdt, int offset);
/* General functions */
/**********************************************************************/
#define fdt_get_header(fdt, field) \
(fdt32_to_cpu(((const struct fdt_header *)(fdt))->field))
(fdt32_ld(&((const struct fdt_header *)(fdt))->field))
#define fdt_magic(fdt) (fdt_get_header(fdt, magic))
#define fdt_totalsize(fdt) (fdt_get_header(fdt, totalsize))
#define fdt_off_dt_struct(fdt) (fdt_get_header(fdt, off_dt_struct))

Loading…
Cancel
Save