aboutsummaryrefslogtreecommitdiff
path: root/libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c
diff options
context:
space:
mode:
Diffstat (limited to 'libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c')
-rw-r--r--libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c137
1 files changed, 137 insertions, 0 deletions
diff --git a/libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c b/libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c
new file mode 100644
index 0000000..1906ad4
--- /dev/null
+++ b/libctf/testsuite/libctf-writable/libctf-bitfield-name-lookup.c
@@ -0,0 +1,137 @@
+/* Verify that name lookup of basic types including old-style bitfield types
+ yields the non-bitfield. */
+
+#include <ctf-api.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+
+int bitfieldery (int count, int up, int pos)
+{
+ unsigned char *ctf_written;
+ size_t size;
+ ctf_dict_t *dict;
+ const char *err = "opening";
+ int open_err;
+ ctf_encoding_t en;
+ ctf_encoding_t basic;
+ ctf_id_t type;
+ size_t i;
+
+ /* This is rendered annoying by two factors: old-style bitfields are not
+ generated by current compilers, so we need to build a suitable dict by
+ hand; and this is an open-time bug, so we need to serialize it and then
+ load it back in again. */
+
+ if ((dict = ctf_create (&open_err)) == NULL)
+ goto open_err;
+
+ /* Populate with a pile of bitfields of increasing/decreasing size, with a
+ single basic type dropped in at position POS. Oscillate the offset
+ between 0 and 1. */
+
+ basic.cte_bits = count;
+ basic.cte_offset = 0;
+ basic.cte_format = CTF_INT_SIGNED;
+
+ en.cte_bits = up ? 0 : count - 1;
+ en.cte_offset = 0;
+ en.cte_format = CTF_INT_SIGNED;
+
+ for (i = 0; i < count; i++)
+ {
+ if (i == pos)
+ {
+ err = "populating with basic type";
+ if (ctf_add_integer (dict, CTF_ADD_ROOT, "int", &basic) < 0)
+ goto err;
+ }
+
+ err = "populating";
+ if (ctf_add_integer (dict, CTF_ADD_ROOT, "int", &en) < 0)
+ goto err;
+
+ en.cte_bits += up ? 1 : -1;
+ if (en.cte_offset == 0)
+ en.cte_offset = 1;
+ else
+ en.cte_offset = 0;
+ }
+
+ /* Possibly populate with at-end basic type. */
+ if (i == pos)
+ {
+ err = "populating with basic type";
+ if (ctf_add_integer (dict, CTF_ADD_ROOT, "int", &basic) < 0)
+ goto err;
+ }
+
+ err = "writing";
+ if ((ctf_written = ctf_write_mem (dict, &size, 4096)) == NULL)
+ goto err;
+ ctf_dict_close (dict);
+
+ err = "opening";
+ if ((dict = ctf_simple_open ((char *) ctf_written, size, NULL, 0,
+ 0, NULL, 0, &open_err)) == NULL)
+ goto open_err;
+
+ err = "looking up";
+ if ((type = ctf_lookup_by_name (dict, "int")) == CTF_ERR)
+ goto err;
+
+ err = "encoding check";
+ if (ctf_type_encoding (dict, type, &en) < 0)
+ goto err;
+
+ if (en.cte_bits < count || en.cte_offset != 0)
+ {
+ fprintf (stderr, "Name lookup with count %i, pos %i, counting %s "
+ "gave bitfield ID %lx with bits %i, offset %i\n", count, pos,
+ up ? "up" : "down", type, en.cte_bits, en.cte_offset);
+ return 1;
+ }
+ ctf_dict_close (dict);
+ free (ctf_written);
+
+ return 0;
+
+ open_err:
+ fprintf (stdout, "Error %s: %s\n", err, ctf_errmsg (open_err));
+ return 1;
+
+ err:
+ fprintf (stdout, "Error %s: %s\n", err, ctf_errmsg (ctf_errno (dict)));
+ return 1;
+}
+
+/* Do a bunch of tests with a type of a given size: up and down, basic type
+ at and near the start and end, and in the middle. */
+
+void mass_bitfieldery (long size)
+{
+ size *= 8;
+ bitfieldery (size, 1, 0);
+ bitfieldery (size, 0, 0);
+ bitfieldery (size, 1, 1);
+ bitfieldery (size, 0, 1);
+ bitfieldery (size, 1, size / 2);
+ bitfieldery (size, 0, size / 2);
+ bitfieldery (size, 1, size - 1);
+ bitfieldery (size, 0, size - 1);
+ bitfieldery (size, 1, size);
+ bitfieldery (size, 0, size);
+}
+
+int main (void)
+{
+ mass_bitfieldery (sizeof (char));
+ mass_bitfieldery (sizeof (short));
+ mass_bitfieldery (sizeof (int));
+ mass_bitfieldery (sizeof (long));
+ mass_bitfieldery (sizeof (uint64_t));
+
+ printf ("All done.\n");
+
+ return 0;
+}