Skip to content

Commit

Permalink
Merge branch 'libbpf: btf typed data dumping fixes (__int128 usage, e…
Browse files Browse the repository at this point in the history
…rror propagation)'

Alan Maguire says:

====================

This series aims to resolve further issues with the BTF typed data
dumping interfaces in libbpf.

Compilation failures with use of __int128 on 32-bit platforms were
reported [1].  As a result, the use of __int128 in libbpf typed data
dumping is replaced with __u64 usage for bitfield manipulations.
In the case of 128-bit integer values, they are simply split into
two 64-bit hex values for display (patch 1).

Tests are added for __int128 display in patch 2, using conditional
compilation to avoid problems with a lack of __int128 support.

Patch 3 resolves an issue Andrii noted about error propagation
when handling enum data display.

More followup work is required to ensure multi-dimensional char array
display works correctly.

[1] https://lore.kernel.org/bpf/1626362126-27775-1-git-send-email-alan.maguire@oracle.com/T/#mc2cb023acfd6c3cd0b661e385787b76bb757430d

Changes since v1:

 - added error handling for bitfield size > 64 bits by changing function
   signature for bitfield retrieval to return an int error value and to set
   bitfield value via a __u64 * argument (Andrii)
====================

Signed-off-by: Andrii Nakryiko <andrii@kernel.org>
  • Loading branch information
Andrii Nakryiko committed Jul 20, 2021
2 parents 875fc31 + 720c29f commit 807b8f0
Show file tree
Hide file tree
Showing 2 changed files with 85 additions and 35 deletions.
103 changes: 68 additions & 35 deletions tools/lib/bpf/btf_dump.c
Original file line number Diff line number Diff line change
Expand Up @@ -1552,31 +1552,26 @@ static int btf_dump_unsupported_data(struct btf_dump *d,
return -ENOTSUP;
}

static void btf_dump_int128(struct btf_dump *d,
const struct btf_type *t,
const void *data)
{
__int128 num = *(__int128 *)data;

if ((num >> 64) == 0)
btf_dump_type_values(d, "0x%llx", (long long)num);
else
btf_dump_type_values(d, "0x%llx%016llx", (long long)num >> 32,
(long long)num);
}

static unsigned __int128 btf_dump_bitfield_get_data(struct btf_dump *d,
const struct btf_type *t,
const void *data,
__u8 bits_offset,
__u8 bit_sz)
static int btf_dump_get_bitfield_value(struct btf_dump *d,
const struct btf_type *t,
const void *data,
__u8 bits_offset,
__u8 bit_sz,
__u64 *value)
{
__u16 left_shift_bits, right_shift_bits;
__u8 nr_copy_bits, nr_copy_bytes;
unsigned __int128 num = 0, ret;
const __u8 *bytes = data;
int sz = t->size;
__u64 num = 0;
int i;

/* Maximum supported bitfield size is 64 bits */
if (sz > 8) {
pr_warn("unexpected bitfield size %d\n", sz);
return -EINVAL;
}

/* Bitfield value retrieval is done in two steps; first relevant bytes are
* stored in num, then we left/right shift num to eliminate irrelevant bits.
*/
Expand All @@ -1591,12 +1586,12 @@ static unsigned __int128 btf_dump_bitfield_get_data(struct btf_dump *d,
#else
# error "Unrecognized __BYTE_ORDER__"
#endif
left_shift_bits = 128 - nr_copy_bits;
right_shift_bits = 128 - bit_sz;
left_shift_bits = 64 - nr_copy_bits;
right_shift_bits = 64 - bit_sz;

ret = (num << left_shift_bits) >> right_shift_bits;
*value = (num << left_shift_bits) >> right_shift_bits;

return ret;
return 0;
}

static int btf_dump_bitfield_check_zero(struct btf_dump *d,
Expand All @@ -1605,9 +1600,12 @@ static int btf_dump_bitfield_check_zero(struct btf_dump *d,
__u8 bits_offset,
__u8 bit_sz)
{
__int128 check_num;
__u64 check_num;
int err;

check_num = btf_dump_bitfield_get_data(d, t, data, bits_offset, bit_sz);
err = btf_dump_get_bitfield_value(d, t, data, bits_offset, bit_sz, &check_num);
if (err)
return err;
if (check_num == 0)
return -ENODATA;
return 0;
Expand All @@ -1619,10 +1617,14 @@ static int btf_dump_bitfield_data(struct btf_dump *d,
__u8 bits_offset,
__u8 bit_sz)
{
unsigned __int128 print_num;
__u64 print_num;
int err;

print_num = btf_dump_bitfield_get_data(d, t, data, bits_offset, bit_sz);
btf_dump_int128(d, t, &print_num);
err = btf_dump_get_bitfield_value(d, t, data, bits_offset, bit_sz, &print_num);
if (err)
return err;

btf_dump_type_values(d, "0x%llx", (unsigned long long)print_num);

return 0;
}
Expand Down Expand Up @@ -1681,9 +1683,29 @@ static int btf_dump_int_data(struct btf_dump *d,
return btf_dump_bitfield_data(d, t, data, 0, 0);

switch (sz) {
case 16:
btf_dump_int128(d, t, data);
case 16: {
const __u64 *ints = data;
__u64 lsi, msi;

/* avoid use of __int128 as some 32-bit platforms do not
* support it.
*/
#if __BYTE_ORDER == __LITTLE_ENDIAN
lsi = ints[0];
msi = ints[1];
#elif __BYTE_ORDER == __BIG_ENDIAN
lsi = ints[1];
msi = ints[0];
#else
# error "Unrecognized __BYTE_ORDER__"
#endif
if (msi == 0)
btf_dump_type_values(d, "0x%llx", (unsigned long long)lsi);
else
btf_dump_type_values(d, "0x%llx%016llx", (unsigned long long)msi,
(unsigned long long)lsi);
break;
}
case 8:
if (sign)
btf_dump_type_values(d, "%lld", *(long long *)data);
Expand Down Expand Up @@ -1931,9 +1953,16 @@ static int btf_dump_get_enum_value(struct btf_dump *d,

/* handle unaligned enum value */
if (!ptr_is_aligned(data, sz)) {
*value = (__s64)btf_dump_bitfield_get_data(d, t, data, 0, 0);
__u64 val;
int err;

err = btf_dump_get_bitfield_value(d, t, data, 0, 0, &val);
if (err)
return err;
*value = (__s64)val;
return 0;
}

switch (t->size) {
case 8:
*value = *(__s64 *)data;
Expand Down Expand Up @@ -2137,8 +2166,9 @@ static int btf_dump_type_data_check_zero(struct btf_dump *d,
return -ENODATA;
}
case BTF_KIND_ENUM:
if (btf_dump_get_enum_value(d, t, data, id, &value))
return 0;
err = btf_dump_get_enum_value(d, t, data, id, &value);
if (err)
return err;
if (value == 0)
return -ENODATA;
return 0;
Expand Down Expand Up @@ -2209,10 +2239,13 @@ static int btf_dump_dump_type_data(struct btf_dump *d,
case BTF_KIND_ENUM:
/* handle bitfield and int enum values */
if (bit_sz) {
unsigned __int128 print_num;
__u64 print_num;
__s64 enum_val;

print_num = btf_dump_bitfield_get_data(d, t, data, bits_offset, bit_sz);
err = btf_dump_get_bitfield_value(d, t, data, bits_offset, bit_sz,
&print_num);
if (err)
break;
enum_val = (__s64)print_num;
err = btf_dump_enum_data(d, t, id, &enum_val);
} else
Expand Down
17 changes: 17 additions & 0 deletions tools/testing/selftests/bpf/prog_tests/btf_dump.c
Original file line number Diff line number Diff line change
Expand Up @@ -327,6 +327,14 @@ static int btf_dump_data(struct btf *btf, struct btf_dump *d,
static void test_btf_dump_int_data(struct btf *btf, struct btf_dump *d,
char *str)
{
#ifdef __SIZEOF_INT128__
__int128 i = 0xffffffffffffffff;

/* this dance is required because we cannot directly initialize
* a 128-bit value to anything larger than a 64-bit value.
*/
i = (i << 64) | (i - 1);
#endif
/* simple int */
TEST_BTF_DUMP_DATA_C(btf, d, NULL, str, int, BTF_F_COMPACT, 1234);
TEST_BTF_DUMP_DATA(btf, d, NULL, str, int, BTF_F_COMPACT | BTF_F_NONAME,
Expand All @@ -348,6 +356,15 @@ static void test_btf_dump_int_data(struct btf *btf, struct btf_dump *d,
TEST_BTF_DUMP_DATA(btf, d, NULL, str, int, 0, "(int)-4567", -4567);

TEST_BTF_DUMP_DATA_OVER(btf, d, NULL, str, int, sizeof(int)-1, "", 1);

#ifdef __SIZEOF_INT128__
TEST_BTF_DUMP_DATA(btf, d, NULL, str, __int128, BTF_F_COMPACT,
"(__int128)0xffffffffffffffff",
0xffffffffffffffff);
ASSERT_OK(btf_dump_data(btf, d, "__int128", NULL, 0, &i, 16, str,
"(__int128)0xfffffffffffffffffffffffffffffffe"),
"dump __int128");
#endif
}

static void test_btf_dump_float_data(struct btf *btf, struct btf_dump *d,
Expand Down

0 comments on commit 807b8f0

Please sign in to comment.