blob: f7201a17140ba90e7e5e1c70a7a1f247f7718614 [file] [log] [blame]
/* { 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* } } } */