This patch adds yet missing C++ API entry points to create union types and to set the fields of structs previously defined as opaque. A missing convenience method returning NULL pointer values is also added and used in one of the two provided tests.
gcc/jit * docs/cp/topics/expressions.rst: Update doc. * docs/cp/topics/types.rst: Update doc. * libgccjit++.h (context::new_union_type): New method. (context::null): New method. (type::null): New method. (struct_::set_fields): New method. gcc/testsuite * jit.dg/test-accessing-union.cc: New test. * jit.dg/test-linked-list.cc: New test. --- gcc/jit/docs/cp/topics/expressions.rst | 10 ++ gcc/jit/docs/cp/topics/types.rst | 27 +++- gcc/jit/libgccjit++.h | 61 ++++++++- gcc/testsuite/jit.dg/test-accessing-union.cc | 86 +++++++++++++ gcc/testsuite/jit.dg/test-linked-list.cc | 128 +++++++++++++++++++ 5 files changed, 304 insertions(+), 8 deletions(-) create mode 100644 gcc/testsuite/jit.dg/test-accessing-union.cc create mode 100644 gcc/testsuite/jit.dg/test-linked-list.cc diff --git a/gcc/jit/docs/cp/topics/expressions.rst b/gcc/jit/docs/cp/topics/expressions.rst index 239e004371e..3e9534790a3 100644 --- a/gcc/jit/docs/cp/topics/expressions.rst +++ b/gcc/jit/docs/cp/topics/expressions.rst @@ -86,6 +86,16 @@ Simple expressions ctxt.new_rvalue (numeric_type, 1) +.. function:: gccjit::rvalue \ + gccjit::context::null (gccjit::type pointer_type) const + + Given a pointer type, get the rvalue for the null pointer. + Essentially this is just a shortcut for: + + .. code-block:: c++ + + ctxt.new_rvalue (pointer_type, (void *) 0) + .. function:: gccjit::rvalue \ gccjit::context::new_rvalue (gccjit::type numeric_type, \ double value) const diff --git a/gcc/jit/docs/cp/topics/types.rst b/gcc/jit/docs/cp/topics/types.rst index c695ceb3098..f41f504da6a 100644 --- a/gcc/jit/docs/cp/topics/types.rst +++ b/gcc/jit/docs/cp/topics/types.rst @@ -179,8 +179,6 @@ You can model C `struct` types by creating :class:`gccjit::struct_` and fields.push_back (field_next); node.set_fields (fields); -.. FIXME: the above API doesn't seem to exist yet - .. function:: gccjit::field \ gccjit::context::new_field (gccjit::type type,\ const char *name, \ @@ -196,10 +194,25 @@ You can model C `struct` types by creating :class:`gccjit::struct_` and Construct a new struct type, with the given name and fields. .. function:: gccjit::struct_ \ - gccjit::context::new_opaque_struct (const std::string &name, \ - gccjit::location loc) + gccjit::context::new_opaque_struct (const std::string &name, \ + gccjit::location loc) Construct a new struct type, with the given name, but without - specifying the fields. The fields can be omitted (in which case the - size of the struct is not known), or later specified using - :c:func:`gcc_jit_struct_set_fields`. + specifying the fields. The fields can be omitted (in which case + the size of the struct is not known), or later specified using + :func:`set_fields`. + +.. function:: void \ + gccjit::struct_::set_fields (std::vector <field> &fields, \ + gccjit::location loc) + + Populate the fields of a formerly-opaque struct type. + + This can only be called once on a given struct type. + +.. function:: gccjit::type \ + gccjit::context::new_union_type (const std::string &name, \ + std::vector<field> &fields,\ + gccjit::location loc) + + Construct a new union type, with the given name and fields. diff --git a/gcc/jit/libgccjit++.h b/gcc/jit/libgccjit++.h index 4b88e877bc9..17b10bc55c3 100644 --- a/gcc/jit/libgccjit++.h +++ b/gcc/jit/libgccjit++.h @@ -165,6 +165,10 @@ namespace gccjit struct_ new_opaque_struct_type (const std::string &name, location loc = location ()); + type new_union_type (const std::string &name, + std::vector <field> &fields, + location loc = location ()); + param new_param (type type_, const std::string &name, location loc = location ()); @@ -189,6 +193,7 @@ namespace gccjit long value) const; rvalue zero (type numeric_type) const; rvalue one (type numeric_type) const; + rvalue null (type pointer_type) const; rvalue new_rvalue (type numeric_type, double value) const; rvalue new_rvalue (type pointer_type, @@ -361,9 +366,10 @@ namespace gccjit type get_aligned (size_t alignment_in_bytes); type get_vector (size_t num_units); - // Shortcuts for getting values of numeric types: + // Shortcuts for getting values of numeric and pointer types: rvalue zero (); rvalue one (); + rvalue null (); }; class struct_ : public type @@ -373,6 +379,9 @@ namespace gccjit struct_ (gcc_jit_struct *inner); gcc_jit_struct *get_inner_struct () const; + + void set_fields (std::vector <field> &fields, + location loc = location ()); }; class function : public object @@ -864,6 +873,26 @@ context::new_opaque_struct_type (const std::string &name, name.c_str ())); } +inline type +context::new_union_type (const std::string &name, + std::vector<field> &fields, + location loc) +{ + /* Treat std::vector as an array, relying on it not being resized: */ + field *as_array_of_wrappers = &fields[0]; + + /* Treat the array as being of the underlying pointers, relying on + the wrapper type being such a pointer internally. */ + gcc_jit_field **as_array_of_ptrs = + reinterpret_cast<gcc_jit_field **> (as_array_of_wrappers); + + return type (gcc_jit_context_new_union_type (m_inner_ctxt, + loc.get_inner_location (), + name.c_str (), + fields.size (), + as_array_of_ptrs)); +} + inline param context::new_param (type type_, const std::string &name, @@ -955,6 +984,13 @@ context::one (type numeric_type) const numeric_type.get_inner_type ())); } +inline rvalue +context::null (type pointer_type) const +{ + return rvalue (gcc_jit_context_null (m_inner_ctxt, + pointer_type.get_inner_type ())); +} + inline rvalue context::new_rvalue (type numeric_type, double value) const @@ -1442,6 +1478,12 @@ type::one () return get_context ().new_rvalue (*this, 1); } +inline rvalue +type::null () +{ + return get_context ().null (*this); +} + // class struct_ inline struct_::struct_ () : type (NULL) {} inline struct_::struct_ (gcc_jit_struct *inner) : @@ -1456,6 +1498,23 @@ struct_::get_inner_struct () const return reinterpret_cast<gcc_jit_struct *> (get_inner_object ()); } +inline void +struct_::set_fields (std::vector <field> &fields, location loc) +{ + /* Treat std::vector as an array, relying on it not being resized: */ + field *as_array_of_wrappers = &fields[0]; + + /* Treat the array as being of the underlying pointers, relying on + the wrapper type being such a pointer internally. */ + gcc_jit_field **as_array_of_ptrs = + reinterpret_cast<gcc_jit_field **> (as_array_of_wrappers); + + gcc_jit_struct_set_fields (get_inner_struct (), + loc.get_inner_location (), + fields.size (), + as_array_of_ptrs); +} + // class function inline function::function () : object () {} inline function::function (gcc_jit_function *inner) diff --git a/gcc/testsuite/jit.dg/test-accessing-union.cc b/gcc/testsuite/jit.dg/test-accessing-union.cc new file mode 100644 index 00000000000..8fa03fd209c --- /dev/null +++ b/gcc/testsuite/jit.dg/test-accessing-union.cc @@ -0,0 +1,86 @@ +#include <cstdlib> +#include <cstdio> +#include <vector> + +#include "libgccjit++.h" + +#include "harness.h" + +union int_or_float +{ + int as_int; + float as_float; +}; + +void +create_code (gcc_jit_context* c_ctxt, void *user_data) +{ + /* Let's try to inject the equivalent of: + float + test_union (int i) + { + union int_or_float u; + u.as_int = i; + return u.as_float; + } + */ + gccjit::context ctxt (c_ctxt); + gccjit::type int_type = + ctxt.get_type (GCC_JIT_TYPE_INT); + gccjit::type float_type = + ctxt.get_type (GCC_JIT_TYPE_FLOAT); + gccjit::field as_int = + ctxt.new_field (int_type, "as_int"); + gccjit::field as_float = + ctxt.new_field (float_type, "as_float"); + std::vector <gccjit::field> fields; + fields.push_back (as_int); + fields.push_back (as_float); + gccjit::type union_type = + ctxt.new_union_type ("int_or_float", fields); + + /* Build the test function. */ + gccjit::param param_i = + ctxt.new_param (int_type, "i"); + std::vector <gccjit::param> params; + params.push_back (param_i); + gccjit::function test_fn = + ctxt.new_function (GCC_JIT_FUNCTION_EXPORTED, + float_type, + "test_union", + params, + 0); + + gccjit::lvalue u = + test_fn.new_local (union_type, "u"); + + gccjit::block block = test_fn.new_block (); + + /* u.as_int = i; */ + block.add_assignment ( + /* "u.as_int = ..." */ + u.access_field (as_int), + param_i); + + /* return u.as_float; */ + block.end_with_return (u.access_field (as_float)); +} + +void +verify_code (gcc_jit_context *ctxt, gcc_jit_result *result) +{ + typedef float (*fn_type) (int i); + CHECK_NON_NULL (result); + + fn_type test_union = + (fn_type)gcc_jit_result_get_code (result, "test_union"); + CHECK_NON_NULL (test_union); + + /* Call the JIT-generated function. */ + float f_result = test_union (42); + + union int_or_float u; + u.as_float = f_result; + + CHECK_VALUE (u.as_int, 42); +} diff --git a/gcc/testsuite/jit.dg/test-linked-list.cc b/gcc/testsuite/jit.dg/test-linked-list.cc new file mode 100644 index 00000000000..933ab706b00 --- /dev/null +++ b/gcc/testsuite/jit.dg/test-linked-list.cc @@ -0,0 +1,128 @@ +#include <cstdlib> +#include <cstdio> +#include <vector> + +#include "libgccjit++.h" + +#include "harness.h" + +/* A doubly-linked list, to ensure that the JIT API can cope with + self-referential types. */ +struct node +{ + struct node *prev; + struct node *next; + int value; +}; + +void +create_code (gcc_jit_context *c_ctxt, void *user_data) +{ + /* Let's try to inject the equivalent of: + int + test_linked_list (struct node *n) + { + int total = 0; + while (n) + { + total += n->value; + n = n->next; + } + return total; + } + */ + gccjit::context ctxt (c_ctxt); + gccjit::type t_int = + ctxt.get_type (GCC_JIT_TYPE_INT); + gccjit::struct_ t_node = + ctxt.new_opaque_struct_type ("node"); + gccjit::type t_node_ptr = + t_node.get_pointer (); + + gccjit::field f_prev = + ctxt.new_field (t_node_ptr, "prev"); + gccjit::field f_next = + ctxt.new_field (t_node_ptr, "next"); + gccjit::field f_value = + ctxt.new_field (t_int, "value"); + std::vector <gccjit::field> fields; + fields.push_back (f_prev); + fields.push_back (f_next); + fields.push_back (f_value); + t_node.set_fields (fields); + + /* Build the test function. */ + gccjit::param param_n = + ctxt.new_param (t_node_ptr, "n"); + std::vector <gccjit::param> params; + params.push_back (param_n); + gccjit::function fn = + ctxt.new_function (GCC_JIT_FUNCTION_EXPORTED, + t_int, + "test_linked_list", + params, + 0); + /* int total; */ + gccjit::lvalue total = + fn.new_local (t_int, "total"); + + gccjit::block initial = fn.new_block ("initial"); + gccjit::block loop_test = fn.new_block ("loop_test"); + gccjit::block loop_body = fn.new_block ("loop_body"); + gccjit::block final = fn.new_block ("final"); + + /* total = 0; */ + initial.add_assignment ( + total, + ctxt.zero (t_int)); + initial.end_with_jump (loop_test); + + /* while (n) */ + loop_test.end_with_conditional (param_n != t_node_ptr.null (), + loop_body, + final); + + /* total += n->value; */ + loop_body.add_assignment_op (total, + GCC_JIT_BINARY_OP_PLUS, + param_n.dereference_field (f_value)); + + /* n = n->next; */ + loop_body.add_assignment (param_n, + param_n.dereference_field (f_next)); + + loop_body.end_with_jump (loop_test); + + /* return total; */ + final.end_with_return (total); +} + +void +verify_code (gcc_jit_context *ctxt, gcc_jit_result *result) +{ + struct node a, b, c; + typedef int (*fn_type) (struct node *n); + CHECK_NON_NULL (result); + + fn_type test_linked_list = + (fn_type)gcc_jit_result_get_code (result, "test_linked_list"); + CHECK_NON_NULL (test_linked_list); + + /* Construct a simple linked-list on the stack: a->b->c: */ + a.prev = NULL; + a.next = &b; + a.value = 5; + + b.prev = &a; + b.next = &c; + b.value = 3; + + c.prev = &b; + c.next = NULL; + c.value = 7; + + CHECK_VALUE (test_linked_list (NULL), 0); + CHECK_VALUE (test_linked_list (&a), 15); + CHECK_VALUE (test_linked_list (&b), 10); + CHECK_VALUE (test_linked_list (&c), 7); +} -- 2.32.0