/* { dg-do compile { target x86_64-*-* } } */ #include <stdlib.h> #include <stdio.h> #include "libgccjit.h" #define TEST_ESCHEWS_SET_OPTIONS static void set_options (gcc_jit_context *ctxt, const char *argv0) { // Set "-O0". gcc_jit_context_set_int_option(ctxt, GCC_JIT_INT_OPTION_OPTIMIZATION_LEVEL, 0); } #define TEST_COMPILING_TO_FILE #define OUTPUT_KIND GCC_JIT_OUTPUT_KIND_ASSEMBLER #define OUTPUT_FILENAME "output-of-test-always_inline-attribute.c.s" #include "harness.h" gcc_jit_function* create_function (gcc_jit_context *ctxt, const char *func_name, gcc_jit_type *int_type, gcc_jit_type *pint_type) { /* The `a` function argument */ gcc_jit_param *a = gcc_jit_context_new_param (ctxt, NULL, pint_type, "a"); gcc_jit_function *func = gcc_jit_context_new_function (ctxt, NULL, GCC_JIT_FUNCTION_INTERNAL, int_type, func_name, 1, &a, 0); gcc_jit_block *if_cond = gcc_jit_function_new_block (func, "if_cond"); gcc_jit_block *if_body = gcc_jit_function_new_block (func, "if_body"); gcc_jit_block *after_if = gcc_jit_function_new_block (func, "after_if"); /* if (!a) */ gcc_jit_block_end_with_conditional ( if_cond, NULL, gcc_jit_context_new_comparison ( ctxt, NULL, GCC_JIT_COMPARISON_EQ, gcc_jit_param_as_rvalue (a), gcc_jit_context_null (ctxt, pint_type)), if_body, after_if); /* return -1; */ gcc_jit_block_end_with_return ( if_body, NULL, gcc_jit_context_new_rvalue_from_int (ctxt, int_type, -1)); /* return *a; */ gcc_jit_block_end_with_return ( after_if, NULL, gcc_jit_lvalue_as_rvalue ( gcc_jit_rvalue_dereference ( gcc_jit_param_as_rvalue (a), NULL))); return func; } void create_code (gcc_jit_context *ctxt, void *user_data) { /* Let's try to inject the equivalent of: __attribute__ ((always_inline)) static inline int removed (int *a) { if (!a) { return -1; } return *a; } static int not_removed (int *a) { if (!a) { return -1; } return *a; } int foo () { int x = 0; x += removed(NULL); x += not_removed(NULL); return x; } */ gcc_jit_type *int_type = gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_INT); gcc_jit_type *pint_type = gcc_jit_type_get_pointer (int_type); /* Creating the `removed` function. */ gcc_jit_function *removed_func = create_function (ctxt, "removed", int_type, pint_type); /* This one is to declare the function as "inline" */ gcc_jit_function_add_attribute(removed_func, GCC_JIT_FN_ATTRIBUTE_INLINE); /* __attribute__ ((always_inline)) */ gcc_jit_function_add_attribute(removed_func, GCC_JIT_FN_ATTRIBUTE_ALWAYS_INLINE); /* Creating the `not_removed` function. */ gcc_jit_function *not_removed_func = create_function (ctxt, "not_removed", int_type, pint_type); /* Creating the `foo` function. */ gcc_jit_function *foo_func = gcc_jit_context_new_function (ctxt, NULL, GCC_JIT_FUNCTION_EXPORTED, int_type, "foo", 0, NULL, 0); gcc_jit_block *foo_block = gcc_jit_function_new_block (foo_func, NULL); /* Build locals: */ gcc_jit_lvalue *x = gcc_jit_function_new_local (foo_func, NULL, int_type, "x"); /* int x = 0; */ gcc_jit_block_add_assignment ( foo_block, NULL, x, gcc_jit_context_new_rvalue_from_int (ctxt, int_type, 0)); /* x += removed(NULL); */ gcc_jit_rvalue *null = gcc_jit_context_null (ctxt, pint_type); gcc_jit_block_add_assignment_op ( foo_block, NULL, x, GCC_JIT_BINARY_OP_PLUS, gcc_jit_context_new_call (ctxt, NULL, removed_func, 1, &null)); /* x += not_removed(NULL); */ gcc_jit_block_add_assignment_op ( foo_block, NULL, x, GCC_JIT_BINARY_OP_PLUS, gcc_jit_context_new_call (ctxt, NULL, not_removed_func, 1, &null)); /* return x; */ gcc_jit_block_end_with_return (foo_block, NULL, gcc_jit_lvalue_as_rvalue(x)); } /* { dg-final { jit-verify-output-file-was-created "" } } */ /* Check that the "removed" function was inlined, but not the others */ /* { dg-final { jit-verify-assembler-output-not ".type\\s+removed,\\s+@function" { target { ! *-*-darwin* } } } } */ /* { dg-final { jit-verify-assembler-output ".type\\s+not_removed,\\s+@function" { target { ! *-*-darwin* } } } } */ /* { dg-final { jit-verify-assembler-output ".type\\s+foo,\\s+@function" { target { ! *-*-darwin* } } } } */ /* { dg-final { jit-verify-assembler-output-not "\\n_removed:" { target *-*-darwin* } } } */ /* { dg-final { jit-verify-assembler-output "\\n_not_removed:" { target *-*-darwin* } } } */ /* { dg-final { jit-verify-assembler-output "\\n_foo:" { target *-*-darwin* } } } */