New jit API entrypoint: gcc_jit_context_compile_to_file

gcc/jit/ChangeLog:
	* docs/cp/topics/results.rst: Rename to...
	* docs/cp/topics/compilation.rst: ...this, and add section on
	ahead-of-time compilation.
	* docs/cp/topics/index.rst: Update for renaming of results.rst
	to compilation.rst.
	* docs/examples/emit-alphabet.bf: New file, a sample "brainf"
	script.
	* docs/examples/tut05-bf.c: New file, implementing a compiler
	for "brainf".
	* docs/internals/test-hello-world.exe.log.txt: Update to reflect
	changes to logger output.
	* docs/intro/index.rst: Add tutorial05.rst
	* docs/intro/tutorial05.rst: New file.
	* docs/topics/results.rst: Rename to...
	* docs/topics/compilation.rst: ...this, and add section on
	ahead-of-time compilation.
	* docs/topics/index.rst: Update for renaming of results.rst to
	compilation.rst.
	* jit-playback.c (gcc::jit::playback::context::compile): Convert
	return type from result * to void.  Move the code to convert to
	dso and dlopen the result to a new pure virtual "postprocess"
	method.
	(gcc::jit::playback::compile_to_memory::compile_to_memory): New
	function.
	(gcc::jit::playback::compile_to_memory::postprocess): New
	function, based on playback::context::compile.
	(gcc::jit::playback::compile_to_file::compile_to_file): New
	function.
	(gcc::jit::playback::compile_to_file::postprocess): New function.
	(gcc::jit::playback::compile_to_file::copy_file): New function.
	(gcc::jit::playback::context::convert_to_dso): Move internals
	to...
	(gcc::jit::playback::context::invoke_driver): New method.  Add
	"-shared" and "-c" options to driver's argv as needed.
	* jit-playback.h: Include "timevar.h".
	(gcc::jit::playback::context::compile): Convert return type from
	result * to void.
	(gcc::jit::playback::context::postprocess): New pure virtual
	function, making this an abstract base class.
	(gcc::jit::playback::context::get_tempdir): New accessor.
	(gcc::jit::playback::context::invoke_driver): New function.
	(class gcc::jit::playback::compile_to_memory): New subclass of
	playback::context.
	(class gcc::jit::playback::compile_to_file): Likewise.
	* jit-recording.c (gcc::jit::recording::context::compile): Use a
	playback::compile_to_memory, and extract its result.
	(gcc::jit::recording::context::compile_to_file): New function.
	* jit-recording.h (gcc::jit::recording::context::compile_to_file):
	New function.
	* libgccjit++.h (gccjit::context::compile_to_file): New method.
	* libgccjit.c (gcc_jit_context_compile): Update log message to
	clarify that this is an in-memory compile.
	(gcc_jit_context_compile_to_file): New function.
	* libgccjit.h (gcc_jit_context): Clarify that you can compile
	a context more than once, and that you can compile to a file
	as well as to memory.
	(gcc_jit_result): Clarify that this is the result of an
	in-memory compilation.
	(gcc_jit_context_compile): Clarify that you can compile, and that
	this is an in-memory compilation.
	(enum gcc_jit_output_kind): New enum.
	(gcc_jit_context_compile_to_file): New function.
	(gcc_jit_context_enable_dump): Clarify comment to cover both forms
	of compilation.
	* libgccjit.map (gcc_jit_context_compile_to_file): New API
	entrypoint.
	* notes.txt: Update to show the playback::context::postprocess
	virtual function.

gcc/testsuite/ChangeLog:
	* jit.dg/harness.h: Include <unistd.h>.
	(CHECK_NO_ERRORS): New.
	(verify_code): Wrap prototype in #ifndef TEST_COMPILING_TO_FILE.
	(test_jit): Support new macro TEST_COMPILING_TO_FILE for exercising
	gcc_jit_context_compile_to_file.
	* jit.dg/jit.exp (fixed_host_execute): Fix the code for passing on
	args to the spawned executable.
	(jit-expand-vars): New function.
	(jit-exe-params): New variable.
	(dg-jit-set-exe-params): New function.
	(jit-dg-test): Detect testcases that use
	jit-verify-compile-to-file and call jit-setup-compile-to-file.
	Set arguments of spawned process to jit-exe-params.
	(jit-get-output-filename): New function.
	(jit-setup-compile-to-file): New function.
	(jit-verify-compile-to-file): New function.
	(jit-run-executable): New function.
	(jit-verify-executable): New function.
	* jit.dg/test-compile-to-assembler.c: New testcase.
	* jit.dg/test-compile-to-dynamic-library.c: New testcase.
	* jit.dg/test-compile-to-executable.c: New testcase.
	* jit.dg/test-compile-to-object.c: New testcase.

From-SVN: r219876
This commit is contained in:
David Malcolm 2015-01-20 01:32:48 +00:00 committed by David Malcolm
parent cb22ab4164
commit fdce7209c2
28 changed files with 3508 additions and 589 deletions

View File

@ -1,3 +1,75 @@
2015-01-19 David Malcolm <dmalcolm@redhat.com>
* docs/cp/topics/results.rst: Rename to...
* docs/cp/topics/compilation.rst: ...this, and add section on
ahead-of-time compilation.
* docs/cp/topics/index.rst: Update for renaming of results.rst
to compilation.rst.
* docs/examples/emit-alphabet.bf: New file, a sample "brainf"
script.
* docs/examples/tut05-bf.c: New file, implementing a compiler
for "brainf".
* docs/internals/test-hello-world.exe.log.txt: Update to reflect
changes to logger output.
* docs/intro/index.rst: Add tutorial05.rst
* docs/intro/tutorial05.rst: New file.
* docs/topics/results.rst: Rename to...
* docs/topics/compilation.rst: ...this, and add section on
ahead-of-time compilation.
* docs/topics/index.rst: Update for renaming of results.rst to
compilation.rst.
* docs/_build/texinfo/libgccjit.texi: Regenerate.
* jit-playback.c (gcc::jit::playback::context::compile): Convert
return type from result * to void. Move the code to convert to
dso and dlopen the result to a new pure virtual "postprocess"
method.
(gcc::jit::playback::compile_to_memory::compile_to_memory): New
function.
(gcc::jit::playback::compile_to_memory::postprocess): New
function, based on playback::context::compile.
(gcc::jit::playback::compile_to_file::compile_to_file): New
function.
(gcc::jit::playback::compile_to_file::postprocess): New function.
(gcc::jit::playback::compile_to_file::copy_file): New function.
(gcc::jit::playback::context::convert_to_dso): Move internals
to...
(gcc::jit::playback::context::invoke_driver): New method. Add
"-shared" and "-c" options to driver's argv as needed.
* jit-playback.h: Include "timevar.h".
(gcc::jit::playback::context::compile): Convert return type from
result * to void.
(gcc::jit::playback::context::postprocess): New pure virtual
function, making this an abstract base class.
(gcc::jit::playback::context::get_tempdir): New accessor.
(gcc::jit::playback::context::invoke_driver): New function.
(class gcc::jit::playback::compile_to_memory): New subclass of
playback::context.
(class gcc::jit::playback::compile_to_file): Likewise.
* jit-recording.c (gcc::jit::recording::context::compile): Use a
playback::compile_to_memory, and extract its result.
(gcc::jit::recording::context::compile_to_file): New function.
* jit-recording.h (gcc::jit::recording::context::compile_to_file):
New function.
* libgccjit++.h (gccjit::context::compile_to_file): New method.
* libgccjit.c (gcc_jit_context_compile): Update log message to
clarify that this is an in-memory compile.
(gcc_jit_context_compile_to_file): New function.
* libgccjit.h (gcc_jit_context): Clarify that you can compile
a context more than once, and that you can compile to a file
as well as to memory.
(gcc_jit_result): Clarify that this is the result of an
in-memory compilation.
(gcc_jit_context_compile): Clarify that you can compile, and that
this is an in-memory compilation.
(enum gcc_jit_output_kind): New enum.
(gcc_jit_context_compile_to_file): New function.
(gcc_jit_context_enable_dump): Clarify comment to cover both forms
of compilation.
* libgccjit.map (gcc_jit_context_compile_to_file): New API
entrypoint.
* notes.txt: Update to show the playback::context::postprocess
virtual function.
2015-01-19 David Malcolm <dmalcolm@redhat.com>
* jit-recording.c

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,58 @@
.. Copyright (C) 2014-2015 Free Software Foundation, Inc.
Originally contributed by David Malcolm <dmalcolm@redhat.com>
This is free software: you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see
<http://www.gnu.org/licenses/>.
.. default-domain:: cpp
Compiling a context
===================
Once populated, a :class:`gccjit::context` can be compiled to
machine code, either in-memory via :func:`gccjit::context::compile` or
to disk via :func:`gccjit::context::compile_to_file`.
You can compile a context multiple times (using either form of
compilation), although any errors that occur on the context will
prevent any future compilation of that context.
In-memory compilation
*********************
.. function:: gcc_jit_result *\
gccjit::context::compile ()
This calls into GCC and builds the code, returning a
`gcc_jit_result *`.
This is a thin wrapper around the
:c:func:`gcc_jit_context_compile` API entrypoint.
Ahead-of-time compilation
*************************
Although libgccjit is primarily aimed at just-in-time compilation, it
can also be used for implementing more traditional ahead-of-time
compilers, via the :func:`gccjit::context::compile_to_file` method.
.. function:: void \
gccjit::context::compile_to_file (enum gcc_jit_output_kind,\
const char *output_path)
Compile the :class:`gccjit::context` to a file of the given
kind.
This is a thin wrapper around the
:c:func:`gcc_jit_context_compile_to_file` API entrypoint.

View File

@ -27,4 +27,4 @@ Topic Reference
expressions.rst
functions.rst
locations.rst
results.rst
compilation.rst

View File

@ -1,48 +0,0 @@
.. Copyright (C) 2014-2015 Free Software Foundation, Inc.
Originally contributed by David Malcolm <dmalcolm@redhat.com>
This is free software: you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see
<http://www.gnu.org/licenses/>.
.. default-domain:: cpp
Compilation results
===================
.. type:: gcc_jit_result
A `gcc_jit_result` encapsulates the result of compiling a context.
.. function:: gcc_jit_result *\
gccjit::context::compile ()
This calls into GCC and builds the code, returning a
`gcc_jit_result *`.
.. function:: void *\
gcc_jit_result_get_code (gcc_jit_result *result,\
const char *funcname)
Locate a given function within the built machine code.
This will need to be cast to a function pointer of the
correct type before it can be called.
.. function:: void\
gcc_jit_result_release (gcc_jit_result *result)
Once we're done with the code, this unloads the built .so file.
This cleans up the result; after calling this, it's no longer
valid to use the result.

View File

@ -0,0 +1,17 @@
[
Emit the uppercase alphabet
]
cell 0 = 26
++++++++++++++++++++++++++
cell 1 = 65
>+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
while cell#0 != 0
[
>
. emit cell#1
+ increment cell@1
<- decrement cell@0
]

View File

@ -0,0 +1,446 @@
/* A compiler for the "bf" language. */
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include "libgccjit.h"
/* Make "main" function:
int
main (int argc, char **argv)
{
...
}
*/
static gcc_jit_function *
make_main (gcc_jit_context *ctxt)
{
gcc_jit_type *int_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_INT);
gcc_jit_param *param_argc =
gcc_jit_context_new_param (ctxt, NULL, int_type, "argc");
gcc_jit_type *char_ptr_ptr_type =
gcc_jit_type_get_pointer (
gcc_jit_type_get_pointer (
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CHAR)));
gcc_jit_param *param_argv =
gcc_jit_context_new_param (ctxt, NULL, char_ptr_ptr_type, "argv");
gcc_jit_param *params[2] = {param_argc, param_argv};
gcc_jit_function *func_main =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_EXPORTED,
int_type,
"main",
2, params,
0);
return func_main;
}
#define MAX_OPEN_PARENS 16
typedef struct bf_compiler
{
const char *filename;
int line;
int column;
gcc_jit_context *ctxt;
gcc_jit_type *void_type;
gcc_jit_type *int_type;
gcc_jit_type *byte_type;
gcc_jit_type *array_type;
gcc_jit_function *func_getchar;
gcc_jit_function *func_putchar;
gcc_jit_function *func;
gcc_jit_block *curblock;
gcc_jit_rvalue *int_zero;
gcc_jit_rvalue *int_one;
gcc_jit_rvalue *byte_zero;
gcc_jit_rvalue *byte_one;
gcc_jit_lvalue *data_cells;
gcc_jit_lvalue *idx;
int num_open_parens;
gcc_jit_block *paren_test[MAX_OPEN_PARENS];
gcc_jit_block *paren_body[MAX_OPEN_PARENS];
gcc_jit_block *paren_after[MAX_OPEN_PARENS];
} bf_compiler;
/* Bail out, with a message on stderr. */
static void
fatal_error (bf_compiler *bfc, const char *msg)
{
fprintf (stderr,
"%s:%i:%i: %s",
bfc->filename, bfc->line, bfc->column, msg);
abort ();
}
/* Get "data_cells[idx]" as an lvalue. */
static gcc_jit_lvalue *
bf_get_current_data (bf_compiler *bfc, gcc_jit_location *loc)
{
return gcc_jit_context_new_array_access (
bfc->ctxt,
loc,
gcc_jit_lvalue_as_rvalue (bfc->data_cells),
gcc_jit_lvalue_as_rvalue (bfc->idx));
}
/* Get "data_cells[idx] == 0" as a boolean rvalue. */
static gcc_jit_rvalue *
bf_current_data_is_zero (bf_compiler *bfc, gcc_jit_location *loc)
{
return gcc_jit_context_new_comparison (
bfc->ctxt,
loc,
GCC_JIT_COMPARISON_EQ,
gcc_jit_lvalue_as_rvalue (bf_get_current_data (bfc, loc)),
bfc->byte_zero);
}
/* Compile one bf character. */
static void
bf_compile_char (bf_compiler *bfc,
unsigned char ch)
{
gcc_jit_location *loc =
gcc_jit_context_new_location (bfc->ctxt,
bfc->filename,
bfc->line,
bfc->column);
/* Turn this on to trace execution, by injecting putchar ()
of each source char. */
if (0)
{
gcc_jit_rvalue *arg =
gcc_jit_context_new_rvalue_from_int (
bfc->ctxt,
bfc->int_type,
ch);
gcc_jit_rvalue *call =
gcc_jit_context_new_call (bfc->ctxt,
loc,
bfc->func_putchar,
1, &arg);
gcc_jit_block_add_eval (bfc->curblock,
loc,
call);
}
switch (ch)
{
case '>':
gcc_jit_block_add_comment (bfc->curblock,
loc,
"'>': idx += 1;");
gcc_jit_block_add_assignment_op (bfc->curblock,
loc,
bfc->idx,
GCC_JIT_BINARY_OP_PLUS,
bfc->int_one);
break;
case '<':
gcc_jit_block_add_comment (bfc->curblock,
loc,
"'<': idx -= 1;");
gcc_jit_block_add_assignment_op (bfc->curblock,
loc,
bfc->idx,
GCC_JIT_BINARY_OP_MINUS,
bfc->int_one);
break;
case '+':
gcc_jit_block_add_comment (bfc->curblock,
loc,
"'+': data[idx] += 1;");
gcc_jit_block_add_assignment_op (bfc->curblock,
loc,
bf_get_current_data (bfc, loc),
GCC_JIT_BINARY_OP_PLUS,
bfc->byte_one);
break;
case '-':
gcc_jit_block_add_comment (bfc->curblock,
loc,
"'-': data[idx] -= 1;");
gcc_jit_block_add_assignment_op (bfc->curblock,
loc,
bf_get_current_data (bfc, loc),
GCC_JIT_BINARY_OP_MINUS,
bfc->byte_one);
break;
case '.':
{
gcc_jit_rvalue *arg =
gcc_jit_context_new_cast (
bfc->ctxt,
loc,
gcc_jit_lvalue_as_rvalue (bf_get_current_data (bfc, loc)),
bfc->int_type);
gcc_jit_rvalue *call =
gcc_jit_context_new_call (bfc->ctxt,
loc,
bfc->func_putchar,
1, &arg);
gcc_jit_block_add_comment (bfc->curblock,
loc,
"'.': putchar ((int)data[idx]);");
gcc_jit_block_add_eval (bfc->curblock,
loc,
call);
}
break;
case ',':
{
gcc_jit_rvalue *call =
gcc_jit_context_new_call (bfc->ctxt,
loc,
bfc->func_getchar,
0, NULL);
gcc_jit_block_add_comment (
bfc->curblock,
loc,
"',': data[idx] = (unsigned char)getchar ();");
gcc_jit_block_add_assignment (bfc->curblock,
loc,
bf_get_current_data (bfc, loc),
gcc_jit_context_new_cast (
bfc->ctxt,
loc,
call,
bfc->byte_type));
}
break;
case '[':
{
gcc_jit_block *loop_test =
gcc_jit_function_new_block (bfc->func, NULL);
gcc_jit_block *on_zero =
gcc_jit_function_new_block (bfc->func, NULL);
gcc_jit_block *on_non_zero =
gcc_jit_function_new_block (bfc->func, NULL);
if (bfc->num_open_parens == MAX_OPEN_PARENS)
fatal_error (bfc, "too many open parens");
gcc_jit_block_end_with_jump (
bfc->curblock,
loc,
loop_test);
gcc_jit_block_add_comment (
loop_test,
loc,
"'['");
gcc_jit_block_end_with_conditional (
loop_test,
loc,
bf_current_data_is_zero (bfc, loc),
on_zero,
on_non_zero);
bfc->paren_test[bfc->num_open_parens] = loop_test;
bfc->paren_body[bfc->num_open_parens] = on_non_zero;
bfc->paren_after[bfc->num_open_parens] = on_zero;
bfc->num_open_parens += 1;
bfc->curblock = on_non_zero;
}
break;
case ']':
{
gcc_jit_block_add_comment (
bfc->curblock,
loc,
"']'");
if (bfc->num_open_parens == 0)
fatal_error (bfc, "mismatching parens");
bfc->num_open_parens -= 1;
gcc_jit_block_end_with_jump (
bfc->curblock,
loc,
bfc->paren_test[bfc->num_open_parens]);
bfc->curblock = bfc->paren_after[bfc->num_open_parens];
}
break;
case '\n':
bfc->line +=1;
bfc->column = 0;
break;
}
if (ch != '\n')
bfc->column += 1;
}
/* Compile the given .bf file into a gcc_jit_context, containing a
single "main" function suitable for compiling into an executable. */
gcc_jit_context *
bf_compile (const char *filename)
{
bf_compiler bfc;
FILE *f_in;
int ch;
memset (&bfc, 0, sizeof (bfc));
bfc.filename = filename;
f_in = fopen (filename, "r");
if (!f_in)
fatal_error (&bfc, "unable to open file");
bfc.line = 1;
bfc.ctxt = gcc_jit_context_acquire ();
gcc_jit_context_set_int_option (
bfc.ctxt,
GCC_JIT_INT_OPTION_OPTIMIZATION_LEVEL,
3);
gcc_jit_context_set_bool_option (
bfc.ctxt,
GCC_JIT_BOOL_OPTION_DUMP_INITIAL_GIMPLE,
0);
gcc_jit_context_set_bool_option (
bfc.ctxt,
GCC_JIT_BOOL_OPTION_DEBUGINFO,
1);
gcc_jit_context_set_bool_option (
bfc.ctxt,
GCC_JIT_BOOL_OPTION_DUMP_EVERYTHING,
0);
gcc_jit_context_set_bool_option (
bfc.ctxt,
GCC_JIT_BOOL_OPTION_KEEP_INTERMEDIATES,
0);
bfc.void_type =
gcc_jit_context_get_type (bfc.ctxt, GCC_JIT_TYPE_VOID);
bfc.int_type =
gcc_jit_context_get_type (bfc.ctxt, GCC_JIT_TYPE_INT);
bfc.byte_type =
gcc_jit_context_get_type (bfc.ctxt, GCC_JIT_TYPE_UNSIGNED_CHAR);
bfc.array_type =
gcc_jit_context_new_array_type (bfc.ctxt,
NULL,
bfc.byte_type,
30000);
bfc.func_getchar =
gcc_jit_context_new_function (bfc.ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
bfc.int_type,
"getchar",
0, NULL,
0);
gcc_jit_param *param_c =
gcc_jit_context_new_param (bfc.ctxt, NULL, bfc.int_type, "c");
bfc.func_putchar =
gcc_jit_context_new_function (bfc.ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
bfc.void_type,
"putchar",
1, &param_c,
0);
bfc.func = make_main (bfc.ctxt);
bfc.curblock =
gcc_jit_function_new_block (bfc.func, "initial");
bfc.int_zero = gcc_jit_context_zero (bfc.ctxt, bfc.int_type);
bfc.int_one = gcc_jit_context_one (bfc.ctxt, bfc.int_type);
bfc.byte_zero = gcc_jit_context_zero (bfc.ctxt, bfc.byte_type);
bfc.byte_one = gcc_jit_context_one (bfc.ctxt, bfc.byte_type);
bfc.data_cells =
gcc_jit_context_new_global (bfc.ctxt, NULL,
GCC_JIT_GLOBAL_INTERNAL,
bfc.array_type,
"data_cells");
bfc.idx =
gcc_jit_function_new_local (bfc.func, NULL,
bfc.int_type,
"idx");
gcc_jit_block_add_comment (bfc.curblock,
NULL,
"idx = 0;");
gcc_jit_block_add_assignment (bfc.curblock,
NULL,
bfc.idx,
bfc.int_zero);
bfc.num_open_parens = 0;
while ( EOF != (ch = fgetc (f_in)))
bf_compile_char (&bfc, (unsigned char)ch);
gcc_jit_block_end_with_return (bfc.curblock, NULL, bfc.int_zero);
fclose (f_in);
return bfc.ctxt;
}
/* Entrypoint to the compiler. */
int
main (int argc, char **argv)
{
const char *input_file;
const char *output_file;
gcc_jit_context *ctxt;
const char *err;
if (argc != 3)
{
fprintf (stderr, "%s: INPUT_FILE OUTPUT_FILE\n", argv[0]);
return 1;
}
input_file = argv[1];
output_file = argv[2];
ctxt = bf_compile (input_file);
gcc_jit_context_compile_to_file (ctxt,
GCC_JIT_OUTPUT_KIND_EXECUTABLE,
output_file);
err = gcc_jit_context_get_first_error (ctxt);
if (err)
{
gcc_jit_context_release (ctxt);
return 1;
}
gcc_jit_context_release (ctxt);
return 0;
}
/* Use the built compiler to compile the example to an executable:
{ dg-jit-set-exe-params SRCDIR/gcc/jit/docs/examples/emit-alphabet.bf emit-alphabet.bf.exe }
Then run the executable, and verify that it emits the alphabet:
{ dg-final { jit-run-executable emit-alphabet.bf.exe "ABCDEFGHIJKLMNOPQRSTUVWXYZ" } } */

View File

@ -38,14 +38,20 @@ JIT: entering: gcc_jit_block_add_eval
JIT: exiting: gcc_jit_block_add_eval
JIT: entering: gcc_jit_block_end_with_void_return
JIT: exiting: gcc_jit_block_end_with_void_return
JIT: entering: gcc_jit_context_dump_reproducer_to_file
JIT: entering: void gcc::jit::recording::context::dump_reproducer_to_file(const char*)
JIT: exiting: void gcc::jit::recording::context::dump_reproducer_to_file(const char*)
JIT: exiting: gcc_jit_context_dump_reproducer_to_file
JIT: entering: gcc_jit_context_compile
JIT: compiling ctxt: 0x1283e20
JIT: in-memory compile of ctxt: 0x1283e20
JIT: entering: gcc::jit::result* gcc::jit::recording::context::compile()
JIT: entering: void gcc::jit::recording::context::validate()
JIT: exiting: void gcc::jit::recording::context::validate()
JIT: entering: gcc::jit::playback::context::context(gcc::jit::recording::context*)
JIT: exiting: gcc::jit::playback::context::context(gcc::jit::recording::context*)
JIT: entering: gcc::jit::result* gcc::jit::playback::context::compile()
JIT: entering: gcc::jit::playback::compile_to_memory::compile_to_memory(gcc::jit::recording::context*)
JIT: exiting: gcc::jit::playback::compile_to_memory::compile_to_memory(gcc::jit::recording::context*)
JIT: entering: void gcc::jit::playback::context::compile()
JIT: entering: gcc::jit::tempdir::tempdir(gcc::jit::logger*, int)
JIT: exiting: gcc::jit::tempdir::tempdir(gcc::jit::logger*, int)
JIT: entering: bool gcc::jit::tempdir::create()
@ -86,29 +92,37 @@ JIT: entering: void gcc::jit::playback::function::postprocess()
JIT: exiting: void gcc::jit::playback::function::postprocess()
JIT: exiting: void gcc::jit::playback::context::replay()
JIT: entering: void jit_langhook_write_globals()
JIT: entering: void gcc::jit::playback::context::write_global_decls_1()
JIT: exiting: void gcc::jit::playback::context::write_global_decls_1()
JIT: entering: void gcc::jit::playback::context::write_global_decls_2()
JIT: exiting: void gcc::jit::playback::context::write_global_decls_2()
JIT: exiting: void jit_langhook_write_globals()
JIT: exiting: toplev::main
JIT: entering: void gcc::jit::playback::context::extract_any_requested_dumps(vec<gcc::jit::recording::requested_dump>*)
JIT: exiting: void gcc::jit::playback::context::extract_any_requested_dumps(vec<gcc::jit::recording::requested_dump>*)
JIT: entering: toplev::finalize
JIT: exiting: toplev::finalize
JIT: entering: void gcc::jit::playback::context::convert_to_dso(const char*)
JIT: argv[0]: x86_64-unknown-linux-gnu-gcc-5.0.0
JIT: argv[1]: -shared
JIT: argv[2]: /tmp/libgccjit-CKq1M9/fake.s
JIT: argv[3]: -o
JIT: argv[4]: /tmp/libgccjit-CKq1M9/fake.so
JIT: argv[5]: -fno-use-linker-plugin
JIT: argv[6]: (null)
JIT: exiting: void gcc::jit::playback::context::convert_to_dso(const char*)
JIT: entering: gcc::jit::result* gcc::jit::playback::context::dlopen_built_dso()
JIT: GCC_JIT_BOOL_OPTION_DEBUGINFO was set: handing over tempdir to jit::result
JIT: entering: gcc::jit::result::result(gcc::jit::logger*, void*, gcc::jit::tempdir*)
JIT: exiting: gcc::jit::result::result(gcc::jit::logger*, void*, gcc::jit::tempdir*)
JIT: exiting: gcc::jit::result* gcc::jit::playback::context::dlopen_built_dso()
JIT: entering: virtual void gcc::jit::playback::compile_to_memory::postprocess(const char*)
JIT: entering: void gcc::jit::playback::context::convert_to_dso(const char*)
JIT: entering: void gcc::jit::playback::context::invoke_driver(const char*, const char*, const char*, timevar_id_t, bool, bool)
JIT: argv[0]: x86_64-unknown-linux-gnu-gcc-5.0.0
JIT: argv[1]: -shared
JIT: argv[2]: /tmp/libgccjit-CKq1M9/fake.s
JIT: argv[3]: -o
JIT: argv[4]: /tmp/libgccjit-CKq1M9/fake.so
JIT: argv[5]: -fno-use-linker-plugin
JIT: argv[6]: (null)
JIT: exiting: void gcc::jit::playback::context::invoke_driver(const char*, const char*, const char*, timevar_id_t, bool, bool)
JIT: exiting: void gcc::jit::playback::context::convert_to_dso(const char*)
JIT: entering: gcc::jit::result* gcc::jit::playback::context::dlopen_built_dso()
JIT: GCC_JIT_BOOL_OPTION_DEBUGINFO was set: handing over tempdir to jit::result
JIT: entering: gcc::jit::result::result(gcc::jit::logger*, void*, gcc::jit::tempdir*)
JIT: exiting: gcc::jit::result::result(gcc::jit::logger*, void*, gcc::jit::tempdir*)
JIT: exiting: gcc::jit::result* gcc::jit::playback::context::dlopen_built_dso()
JIT: exiting: virtual void gcc::jit::playback::compile_to_memory::postprocess(const char*)
JIT: entering: void gcc::jit::playback::context::release_mutex()
JIT: exiting: void gcc::jit::playback::context::release_mutex()
JIT: exiting: gcc::jit::result* gcc::jit::playback::context::compile()
JIT: exiting: void gcc::jit::playback::context::compile()
JIT: entering: gcc::jit::playback::context::~context()
JIT: exiting: gcc::jit::playback::context::~context()
JIT: exiting: gcc::jit::result* gcc::jit::recording::context::compile()

View File

@ -25,3 +25,4 @@ Tutorial
tutorial02.rst
tutorial03.rst
tutorial04.rst
tutorial05.rst

View File

@ -0,0 +1,256 @@
.. Copyright (C) 2015 Free Software Foundation, Inc.
Originally contributed by David Malcolm <dmalcolm@redhat.com>
This is free software: you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see
<http://www.gnu.org/licenses/>.
Tutorial part 5: Implementing an Ahead-of-Time compiler
-------------------------------------------------------
If you have a pre-existing language frontend that's compatible with
libgccjit's license, it's possible to hook it up to libgccjit as a
backend. In the previous example we showed
how to do that for in-memory JIT-compilation, but libgccjit can also
compile code directly to a file, allowing you to implement a more
traditional ahead-of-time compiler ("JIT" is something of a misnomer
for this use-case).
The essential difference is to compile the context using
:c:func:`gcc_jit_context_compile_to_file` rather than
:c:func:`gcc_jit_context_compile`.
The "brainf" language
*********************
In this example we use libgccjit to construct an ahead-of-time compiler
for an esoteric programming language that we shall refer to as "brainf".
brainf scripts operate on an array of bytes, with a notional data pointer
within the array.
brainf is hard for humans to read, but it's trivial to write a parser for
it, as there is no lexing; just a stream of bytes. The operations are:
====================== =============================
Character Meaning
====================== =============================
``>`` ``idx += 1``
``<`` ``idx -= 1``
``+`` ``data[idx] += 1``
``-`` ``data[idx] -= 1``
``.`` ``output (data[idx])``
``,`` ``data[idx] = input ()``
``[`` loop until ``data[idx] == 0``
``]`` end of loop
Anything else ignored
====================== =============================
Unlike the previous example, we'll implement an ahead-of-time compiler,
which reads ``.bf`` scripts and outputs executables (though it would
be trivial to have it run them JIT-compiled in-process).
Here's what a simple ``.bf`` script looks like:
.. literalinclude:: ../examples/emit-alphabet.bf
:lines: 1-
.. note::
This example makes use of whitespace and comments for legibility, but
could have been written as::
++++++++++++++++++++++++++
>+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
[>.+<-]
It's not a particularly useful language, except for providing
compiler-writers with a test case that's easy to parse. The point
is that you can use :c:func:`gcc_jit_context_compile_to_file`
to use libgccjit as a backend for a pre-existing language frontend
(provided that the pre-existing frontend is compatible with libgccjit's
license).
Converting a brainf script to libgccjit IR
******************************************
As before we write simple code to populate a :c:type:`gcc_jit_context *`.
.. literalinclude:: ../examples/tut05-bf.c
:start-after: #define MAX_OPEN_PARENS 16
:end-before: /* Entrypoint to the compiler. */
:language: c
Compiling a context to a file
*****************************
Unlike the previous tutorial, this time we'll compile the context
directly to an executable, using :c:func:`gcc_jit_context_compile_to_file`:
.. code-block:: c
gcc_jit_context_compile_to_file (ctxt,
GCC_JIT_OUTPUT_KIND_EXECUTABLE,
output_file);
Here's the top-level of the compiler, which is what actually calls into
:c:func:`gcc_jit_context_compile_to_file`:
.. literalinclude:: ../examples/tut05-bf.c
:start-after: /* Entrypoint to the compiler. */
:end-before: /* Use the built compiler to compile the example to an executable:
:language: c
Note how once the context is populated you could trivially instead compile
it to memory using :c:func:`gcc_jit_context_compile` and run it in-process
as in the previous tutorial.
To create an executable, we need to export a ``main`` function. Here's
how to create one from the JIT API:
.. literalinclude:: ../examples/tut05-bf.c
:start-after: #include "libgccjit.h"
:end-before: #define MAX_OPEN_PARENS 16
:language: c
.. note::
The above implementation ignores ``argc`` and ``argv``, but you could
make use of them by exposing ``param_argc`` and ``param_argv`` to the
caller.
Upon compiling this C code, we obtain a bf-to-machine-code compiler;
let's call it ``bfc``:
.. code-block:: console
$ gcc \
tut05-bf.c \
-o bfc \
-lgccjit
We can now use ``bfc`` to compile .bf files into machine code executables:
.. code-block:: console
$ ./bfc \
emit-alphabet.bf \
a.out
which we can run directly:
.. code-block:: console
$ ./a.out
ABCDEFGHIJKLMNOPQRSTUVWXYZ
Success!
We can also inspect the generated executable using standard tools:
.. code-block:: console
$ objdump -d a.out |less
which shows that libgccjit has managed to optimize the function
somewhat (for example, the runs of 26 and 65 increment operations
have become integer constants 0x1a and 0x41):
.. code-block:: console
0000000000400620 <main>:
400620: 80 3d 39 0a 20 00 00 cmpb $0x0,0x200a39(%rip) # 601060 <data
400627: 74 07 je 400630 <main
400629: eb fe jmp 400629 <main+0x9>
40062b: 0f 1f 44 00 00 nopl 0x0(%rax,%rax,1)
400630: 48 83 ec 08 sub $0x8,%rsp
400634: 0f b6 05 26 0a 20 00 movzbl 0x200a26(%rip),%eax # 601061 <data_cells+0x1>
40063b: c6 05 1e 0a 20 00 1a movb $0x1a,0x200a1e(%rip) # 601060 <data_cells>
400642: 8d 78 41 lea 0x41(%rax),%edi
400645: 40 88 3d 15 0a 20 00 mov %dil,0x200a15(%rip) # 601061 <data_cells+0x1>
40064c: 0f 1f 40 00 nopl 0x0(%rax)
400650: 40 0f b6 ff movzbl %dil,%edi
400654: e8 87 fe ff ff callq 4004e0 <putchar@plt>
400659: 0f b6 05 01 0a 20 00 movzbl 0x200a01(%rip),%eax # 601061 <data_cells+0x1>
400660: 80 2d f9 09 20 00 01 subb $0x1,0x2009f9(%rip) # 601060 <data_cells>
400667: 8d 78 01 lea 0x1(%rax),%edi
40066a: 40 88 3d f0 09 20 00 mov %dil,0x2009f0(%rip) # 601061 <data_cells+0x1>
400671: 75 dd jne 400650 <main+0x30>
400673: 31 c0 xor %eax,%eax
400675: 48 83 c4 08 add $0x8,%rsp
400679: c3 retq
40067a: 66 0f 1f 44 00 00 nopw 0x0(%rax,%rax,1)
We also set up debugging information (via
:c:func:`gcc_jit_context_new_location` and
:c:macro:`GCC_JIT_BOOL_OPTION_DEBUGINFO`), so it's possible to use ``gdb``
to singlestep through the generated binary and inspect the internal
state ``idx`` and ``data_cells``:
.. code-block:: console
(gdb) break main
Breakpoint 1 at 0x400790
(gdb) run
Starting program: a.out
Breakpoint 1, 0x0000000000400790 in main (argc=1, argv=0x7fffffffe448)
(gdb) stepi
0x0000000000400797 in main (argc=1, argv=0x7fffffffe448)
(gdb) stepi
0x00000000004007a0 in main (argc=1, argv=0x7fffffffe448)
(gdb) stepi
9 >+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
(gdb) list
4
5 cell 0 = 26
6 ++++++++++++++++++++++++++
7
8 cell 1 = 65
9 >+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
10
11 while cell#0 != 0
12 [
13 >
(gdb) n
6 ++++++++++++++++++++++++++
(gdb) n
9 >+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
(gdb) p idx
$1 = 1
(gdb) p data_cells
$2 = "\032", '\000' <repeats 29998 times>
(gdb) p data_cells[0]
$3 = 26 '\032'
(gdb) p data_cells[1]
$4 = 0 '\000'
(gdb) list
4
5 cell 0 = 26
6 ++++++++++++++++++++++++++
7
8 cell 1 = 65
9 >+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++<
10
11 while cell#0 != 0
12 [
13 >
Other forms of ahead-of-time-compilation
****************************************
The above demonstrates compiling a :c:type:`gcc_jit_context *` directly
to an executable. It's also possible to compile it to an object file,
and to a dynamic library. See the documentation of
:c:func:`gcc_jit_context_compile_to_file` for more information.

View File

@ -17,14 +17,19 @@
.. default-domain:: c
Compilation results
Compiling a context
===================
.. type:: gcc_jit_result
Once populated, a :c:type:`gcc_jit_context *` can be compiled to
machine code, either in-memory via :c:func:`gcc_jit_context_compile` or
to disk via :c:func:`gcc_jit_context_compile_to_file`.
A `gcc_jit_result` encapsulates the result of compiling a context,
and the lifetimes of any machine code functions or globals that are
within it.
You can compile a context multiple times (using either form of
compilation), although any errors that occur on the context will
prevent any future compilation of that context.
In-memory compilation
*********************
.. function:: gcc_jit_result *\
gcc_jit_context_compile (gcc_jit_context *ctxt)
@ -36,6 +41,12 @@ Compilation results
calling :func:`gcc_jit_result_release` on it once they're done
with it.
.. type:: gcc_jit_result
A `gcc_jit_result` encapsulates the result of compiling a context
in-memory, and the lifetimes of any machine code functions or globals
that are within the resuilt.
.. function:: void *\
gcc_jit_result_get_code (gcc_jit_result *result,\
const char *funcname)
@ -125,3 +136,64 @@ Compilation results
valid to use the result, or any code or globals that were obtained
by calling :func:`gcc_jit_result_get_code` or
:func:`gcc_jit_result_get_global` on it.
Ahead-of-time compilation
*************************
Although libgccjit is primarily aimed at just-in-time compilation, it
can also be used for implementing more traditional ahead-of-time
compilers, via the :c:func:`gcc_jit_context_compile_to_file`
API entrypoint.
.. function:: void \
gcc_jit_context_compile_to_file (gcc_jit_context *ctxt, \
enum gcc_jit_output_kind output_kind,\
const char *output_path)
Compile the :c:type:`gcc_jit_context *` to a file of the given
kind.
:c:func:`gcc_jit_context_compile_to_file` ignores the suffix of
``output_path``, and insteads uses the given
:c:type:`enum gcc_jit_output_kind` to decide what to do.
.. note::
This is different from the ``gcc`` program, which does make use of the
suffix of the output file when determining what to do.
.. type:: enum gcc_jit_output_kind
The available kinds of output are:
============================================== ==============
Output kind Typical suffix
============================================== ==============
:c:macro:`GCC_JIT_OUTPUT_KIND_ASSEMBLER` .s
:c:macro:`GCC_JIT_OUTPUT_KIND_OBJECT_FILE` .o
:c:macro:`GCC_JIT_OUTPUT_KIND_DYNAMIC_LIBRARY` .so or .dll
:c:macro:`GCC_JIT_OUTPUT_KIND_EXECUTABLE` None, or .exe
============================================== ==============
.. c:macro:: GCC_JIT_OUTPUT_KIND_ASSEMBLER
Compile the context to an assembler file.
.. c:macro:: GCC_JIT_OUTPUT_KIND_OBJECT_FILE
Compile the context to an object file.
.. c:macro:: GCC_JIT_OUTPUT_KIND_DYNAMIC_LIBRARY
Compile the context to a dynamic library.
There is currently no support for specifying other libraries to link
against.
.. c:macro:: GCC_JIT_OUTPUT_KIND_EXECUTABLE
Compile the context to an executable.
There is currently no support for specifying libraries to link
against.

View File

@ -27,4 +27,4 @@ Topic Reference
expressions.rst
functions.rst
locations.rst
results.rst
compilation.rst

View File

@ -1668,26 +1668,37 @@ auto_argvec::~auto_argvec ()
- Use the context's options to cconstruct command-line options, and
call into the rest of GCC (toplev::main).
- Assuming it succeeds, we have a .s file; we want a .so file.
Invoke another gcc to convert the .s file to a .so file.
- dlopen the .so file
- Wrap the result up as a playback::result and return it. */
- Assuming it succeeds, we have a .s file.
- We then run the "postprocess" vfunc:
result *
(A) In-memory compile ("gcc_jit_context_compile")
For an in-memory compile we have the playback::compile_to_memory
subclass; "postprocess" will convert the .s file to a .so DSO,
and load it in memory (via dlopen), wrapping the result up as
a jit::result and returning it.
(B) Compile to file ("gcc_jit_context_compile_to_file")
When compiling to a file, we have the playback::compile_to_file
subclass; "postprocess" will either copy the .s file to the
destination (for GCC_JIT_OUTPUT_KIND_ASSEMBLER), or invoke
the driver to convert it as necessary, copying the result. */
void
playback::context::
compile ()
{
JIT_LOG_SCOPE (get_logger ());
const char *ctxt_progname;
result *result_obj = NULL;
int keep_intermediates =
get_bool_option (GCC_JIT_BOOL_OPTION_KEEP_INTERMEDIATES);
m_tempdir = new tempdir (get_logger (), keep_intermediates);
if (!m_tempdir->create ())
return NULL;
return;
/* Call into the rest of gcc.
For now, we have to assemble command-line options to pass into
@ -1706,7 +1717,7 @@ compile ()
auto_argvec fake_args;
make_fake_args (&fake_args, ctxt_progname, &requested_dumps);
if (errors_occurred ())
return NULL;
return;
/* Acquire the JIT mutex and set "this" as the active playback ctxt. */
acquire_mutex ();
@ -1737,24 +1748,258 @@ compile ()
if (errors_occurred ())
{
release_mutex ();
return NULL;
return;
}
if (get_bool_option (GCC_JIT_BOOL_OPTION_DUMP_GENERATED_CODE))
dump_generated_code ();
convert_to_dso (ctxt_progname);
if (errors_occurred ())
{
release_mutex ();
return NULL;
}
/* We now have a .s file.
result_obj = dlopen_built_dso ();
Run any postprocessing steps. This will either convert the .s file to
a .so DSO, and load it in memory (playback::compile_to_memory), or
convert the .s file to the requested output format, and copy it to a
given file (playback::compile_to_file). */
postprocess (ctxt_progname);
release_mutex ();
}
return result_obj;
/* Implementation of class gcc::jit::playback::compile_to_memory,
a subclass of gcc::jit::playback::context. */
/* playback::compile_to_memory's trivial constructor. */
playback::compile_to_memory::compile_to_memory (recording::context *ctxt) :
playback::context (ctxt),
m_result (NULL)
{
JIT_LOG_SCOPE (get_logger ());
}
/* Implementation of the playback::context::process vfunc for compiling
to memory.
Convert the .s file to a .so DSO, and load it in memory (via dlopen),
wrapping the result up as a jit::result and returning it. */
void
playback::compile_to_memory::postprocess (const char *ctxt_progname)
{
JIT_LOG_SCOPE (get_logger ());
convert_to_dso (ctxt_progname);
if (errors_occurred ())
return;
m_result = dlopen_built_dso ();
}
/* Implementation of class gcc::jit::playback::compile_to_file,
a subclass of gcc::jit::playback::context. */
/* playback::compile_to_file's trivial constructor. */
playback::compile_to_file::compile_to_file (recording::context *ctxt,
enum gcc_jit_output_kind output_kind,
const char *output_path) :
playback::context (ctxt),
m_output_kind (output_kind),
m_output_path (output_path)
{
JIT_LOG_SCOPE (get_logger ());
}
/* Implementation of the playback::context::process vfunc for compiling
to a file.
Either copy the .s file to the given destination (for
GCC_JIT_OUTPUT_KIND_ASSEMBLER), or invoke the driver to convert it
as necessary, copying the result. */
void
playback::compile_to_file::postprocess (const char *ctxt_progname)
{
JIT_LOG_SCOPE (get_logger ());
/* The driver takes different actions based on the filename, so
we provide a filename with an appropriate suffix for the
output kind, and then copy it up to the user-provided path,
rather than directly compiling it to the requested output path. */
switch (m_output_kind)
{
default:
gcc_unreachable ();
case GCC_JIT_OUTPUT_KIND_ASSEMBLER:
copy_file (get_tempdir ()->get_path_s_file (),
m_output_path);
break;
case GCC_JIT_OUTPUT_KIND_OBJECT_FILE:
{
char *tmp_o_path = ::concat (get_tempdir ()->get_path (),
"/fake.o",
NULL);
invoke_driver (ctxt_progname,
get_tempdir ()->get_path_s_file (),
tmp_o_path,
TV_ASSEMBLE,
false, /* bool shared, */
false);/* bool run_linker */
if (!errors_occurred ())
copy_file (tmp_o_path,
m_output_path);
free (tmp_o_path);
}
break;
case GCC_JIT_OUTPUT_KIND_DYNAMIC_LIBRARY:
invoke_driver (ctxt_progname,
get_tempdir ()->get_path_s_file (),
get_tempdir ()->get_path_so_file (),
TV_ASSEMBLE,
true, /* bool shared, */
true);/* bool run_linker */
if (!errors_occurred ())
copy_file (get_tempdir ()->get_path_so_file (),
m_output_path);
break;
case GCC_JIT_OUTPUT_KIND_EXECUTABLE:
{
char *tmp_exe_path = ::concat (get_tempdir ()->get_path (),
"/fake.exe",
NULL);
invoke_driver (ctxt_progname,
get_tempdir ()->get_path_s_file (),
tmp_exe_path,
TV_ASSEMBLE,
false, /* bool shared, */
true);/* bool run_linker */
if (!errors_occurred ())
copy_file (tmp_exe_path,
m_output_path);
free (tmp_exe_path);
}
break;
}
}
/* Copy SRC_PATH to DST_PATH, preserving permission bits (in particular,
the "executable" bits).
Any errors that occur are reported on the context and hence count as
a failure of the compile.
We can't in general hardlink or use "rename" from the tempdir since
it might be on a different filesystem to the destination. For example,
I get EXDEV: "Invalid cross-device link". */
void
playback::compile_to_file::copy_file (const char *src_path,
const char *dst_path)
{
JIT_LOG_SCOPE (get_logger ());
if (get_logger ())
{
get_logger ()->log ("src_path: %s", src_path);
get_logger ()->log ("dst_path: %s", dst_path);
}
FILE *f_in = NULL;
FILE *f_out = NULL;
size_t total_sz_in = 0;
size_t total_sz_out = 0;
char buf[4096];
size_t sz_in;
struct stat stat_buf;
f_in = fopen (src_path, "rb");
if (!f_in)
{
add_error (NULL,
"unable to open %s for reading: %s",
src_path,
xstrerror (errno));
return;
}
/* Use stat on the filedescriptor to get the mode,
so that we can copy it over (in particular, the
"executable" bits). */
if (-1 == fstat (fileno (f_in), &stat_buf))
{
add_error (NULL,
"unable to fstat %s: %s",
src_path,
xstrerror (errno));
fclose (f_in);
return;
}
f_out = fopen (dst_path, "wb");
if (!f_out)
{
add_error (NULL,
"unable to open %s for writing: %s",
dst_path,
xstrerror (errno));
fclose (f_in);
return;
}
while ( (sz_in = fread (buf, 1, sizeof (buf), f_in)) )
{
total_sz_in += sz_in;
size_t sz_out_remaining = sz_in;
size_t sz_out_so_far = 0;
while (sz_out_remaining)
{
size_t sz_out = fwrite (buf + sz_out_so_far,
1,
sz_out_remaining,
f_out);
gcc_assert (sz_out <= sz_out_remaining);
if (!sz_out)
{
add_error (NULL,
"error writing to %s: %s",
dst_path,
xstrerror (errno));
fclose (f_in);
fclose (f_out);
return;
}
total_sz_out += sz_out;
sz_out_so_far += sz_out;
sz_out_remaining -= sz_out;
}
gcc_assert (sz_out_so_far == sz_in);
}
if (!feof (f_in))
add_error (NULL,
"error reading from %s: %s",
src_path,
xstrerror (errno));
fclose (f_in);
gcc_assert (total_sz_in == total_sz_out);
if (get_logger ())
get_logger ()->log ("total bytes copied: %ld", total_sz_out);
/* Set the permissions of the copy to those of the original file,
in particular the "executable" bits. */
if (-1 == fchmod (fileno (f_out), stat_buf.st_mode))
add_error (NULL,
"error setting mode of %s: %s",
dst_path,
xstrerror (errno));
fclose (f_out);
}
/* Helper functions for gcc::jit::playback::context::compile. */
@ -1973,11 +2218,30 @@ playback::context::read_dump_file (const char *path)
void
playback::context::
convert_to_dso (const char *ctxt_progname)
{
JIT_LOG_SCOPE (get_logger ());
invoke_driver (ctxt_progname,
m_tempdir->get_path_s_file (),
m_tempdir->get_path_so_file (),
TV_ASSEMBLE,
true, /* bool shared, */
true);/* bool run_linker */
}
void
playback::context::
invoke_driver (const char *ctxt_progname,
const char *input_file,
const char *output_file,
timevar_id_t tv_id,
bool shared,
bool run_linker)
{
JIT_LOG_SCOPE (get_logger ());
/* Currently this lumps together both assembling and linking into
TV_ASSEMBLE. */
auto_timevar assemble_timevar (TV_ASSEMBLE);
auto_timevar assemble_timevar (tv_id);
const char *errmsg;
auto_vec <const char *> argvec;
#define ADD_ARG(arg) argvec.safe_push (arg)
@ -1986,12 +2250,16 @@ convert_to_dso (const char *ctxt_progname)
const char *gcc_driver_name = GCC_DRIVER_NAME;
ADD_ARG (gcc_driver_name);
ADD_ARG ("-shared");
/* The input: assembler. */
ADD_ARG (m_tempdir->get_path_s_file ());
/* The output: shared library. */
if (shared)
ADD_ARG ("-shared");
if (!run_linker)
ADD_ARG ("-c");
ADD_ARG (input_file);
ADD_ARG ("-o");
ADD_ARG (m_tempdir->get_path_so_file ());
ADD_ARG (output_file);
/* Don't use the linker plugin.
If running with just a "make" and not a "make install", then we'd

View File

@ -23,6 +23,8 @@ along with GCC; see the file COPYING3. If not see
#include <utility> // for std::pair
#include "timevar.h"
#include "jit-recording.h"
namespace gcc {
@ -35,6 +37,12 @@ namespace jit {
namespace playback {
/* playback::context is an abstract base class.
The two concrete subclasses are:
- playback::compile_to_memory
- playback::compile_to_file. */
class context : public log_user
{
public:
@ -174,7 +182,7 @@ public:
return m_recording_ctxt->get_builtins_manager ();
}
result *
void
compile ();
void
@ -252,9 +260,22 @@ private:
char *
read_dump_file (const char *path);
virtual void postprocess (const char *ctxt_progname) = 0;
protected:
tempdir *get_tempdir () { return m_tempdir; }
void
convert_to_dso (const char *ctxt_progname);
void
invoke_driver (const char *ctxt_progname,
const char *input_file,
const char *output_file,
timevar_id_t tv_id,
bool shared,
bool run_linker);
result *
dlopen_built_dso ();
@ -274,6 +295,37 @@ private:
auto_vec<std::pair<tree, location *> > m_cached_locations;
};
class compile_to_memory : public context
{
public:
compile_to_memory (recording::context *ctxt);
void postprocess (const char *ctxt_progname);
result *get_result_obj () const { return m_result; }
private:
result *m_result;
};
class compile_to_file : public context
{
public:
compile_to_file (recording::context *ctxt,
enum gcc_jit_output_kind output_kind,
const char *output_path);
void postprocess (const char *ctxt_progname);
private:
void
copy_file (const char *src_path,
const char *dst_path);
private:
enum gcc_jit_output_kind m_output_kind;
const char *m_output_path;
};
/* A temporary wrapper object.
These objects are (mostly) only valid during replay.
We allocate them on the GC heap, so that they will be cleaned

View File

@ -1152,8 +1152,8 @@ recording::context::enable_dump (const char *dumpname,
m_requested_dumps.safe_push (d);
}
/* Validate this context, and if it passes, compile it within a
mutex.
/* Validate this context, and if it passes, compile it to memory
(within a mutex).
Implements the post-error-checking part of
gcc_jit_context_compile. */
@ -1168,13 +1168,41 @@ recording::context::compile ()
if (errors_occurred ())
return NULL;
/* Set up a playback context. */
::gcc::jit::playback::context replayer (this);
/* Set up a compile_to_memory playback context. */
::gcc::jit::playback::compile_to_memory replayer (this);
/* Use it. */
result *result_obj = replayer.compile ();
replayer.compile ();
return result_obj;
/* Get the jit::result (or NULL) from the
compile_to_memory playback context. */
return replayer.get_result_obj ();
}
/* Validate this context, and if it passes, compile it to a file
(within a mutex).
Implements the post-error-checking part of
gcc_jit_context_compile_to_file. */
void
recording::context::compile_to_file (enum gcc_jit_output_kind output_kind,
const char *output_path)
{
JIT_LOG_SCOPE (get_logger ());
validate ();
if (errors_occurred ())
return;
/* Set up a compile_to_file playback context. */
::gcc::jit::playback::compile_to_file replayer (this,
output_kind,
output_path);
/* Use it. */
replayer.compile ();
}
/* Format the given error using printf's conventions, print

View File

@ -220,6 +220,10 @@ public:
result *
compile ();
void
compile_to_file (enum gcc_jit_output_kind output_kind,
const char *output_path);
void
add_error (location *loc, const char *fmt, ...)
GNU_PRINTF(3, 4);

View File

@ -99,6 +99,9 @@ namespace gccjit
gcc_jit_result *compile ();
void compile_to_file (enum gcc_jit_output_kind output_kind,
const char *output_path);
void dump_to_file (const std::string &path,
bool update_locations);
@ -540,6 +543,15 @@ context::compile ()
return result;
}
inline void
context::compile_to_file (enum gcc_jit_output_kind output_kind,
const char *output_path)
{
gcc_jit_context_compile_to_file (m_inner_ctxt,
output_kind,
output_path);
}
inline void
context::dump_to_file (const std::string &path,
bool update_locations)

View File

@ -2196,7 +2196,7 @@ gcc_jit_context_compile (gcc_jit_context *ctxt)
JIT_LOG_FUNC (ctxt->get_logger ());
ctxt->log ("compiling ctxt: %p", (void *)ctxt);
ctxt->log ("in-memory compile of ctxt: %p", (void *)ctxt);
gcc_jit_result *result = (gcc_jit_result *)ctxt->compile ();
@ -2206,6 +2206,35 @@ gcc_jit_context_compile (gcc_jit_context *ctxt)
return result;
}
/* Public entrypoint. See description in libgccjit.h.
After error-checking, the real work is done by the
gcc::jit::recording::context::compile_to_file method in
jit-recording.c. */
void
gcc_jit_context_compile_to_file (gcc_jit_context *ctxt,
enum gcc_jit_output_kind output_kind,
const char *output_path)
{
RETURN_IF_FAIL (ctxt, NULL, NULL, "NULL context");
JIT_LOG_FUNC (ctxt->get_logger ());
RETURN_IF_FAIL_PRINTF1 (
((output_kind >= GCC_JIT_OUTPUT_KIND_ASSEMBLER)
&& (output_kind <= GCC_JIT_OUTPUT_KIND_EXECUTABLE)),
ctxt, NULL,
"unrecognized output_kind: %i",
output_kind);
RETURN_IF_FAIL (output_path, ctxt, NULL, "NULL output_path");
ctxt->log ("compile_to_file of ctxt: %p", (void *)ctxt);
ctxt->log ("output_kind: %i", output_kind);
ctxt->log ("output_path: %s", output_path);
ctxt->compile_to_file (output_kind, output_path);
}
/* Public entrypoint. See description in libgccjit.h.
After error-checking, the real work is done by the

View File

@ -36,17 +36,20 @@ extern "C" {
the API below.
Invoking gcc_jit_context_compile on it gives you a gcc_jit_result *
(or NULL).
(or NULL), representing in-memory machine code.
You can call gcc_jit_context_compile repeatedly on one context, giving
multiple independent results.
Similarly, you can call gcc_jit_context_compile_to_file on a context
to compile to disk.
Eventually you can call gcc_jit_context_release to clean up the
context; any results created from it are still usable, and should be
cleaned up via gcc_jit_result_release. */
context; any in-memory results created from it are still usable, and
should be cleaned up via gcc_jit_result_release. */
typedef struct gcc_jit_context gcc_jit_context;
/* A gcc_jit_result encapsulates the result of a compilation. */
/* A gcc_jit_result encapsulates the result of an in-memory compilation. */
typedef struct gcc_jit_result gcc_jit_result;
/* An object created within a context. Such objects are automatically
@ -240,12 +243,42 @@ gcc_jit_context_set_bool_option (gcc_jit_context *ctxt,
enum gcc_jit_bool_option opt,
int value);
/* This actually calls into GCC and runs the build, all
in a mutex for now. The result is a wrapper around a .so file.
It can only be called once on a given context. */
/* Compile the context to in-memory machine code.
This can be called more that once on a given context,
although any errors that occur will block further compilation. */
extern gcc_jit_result *
gcc_jit_context_compile (gcc_jit_context *ctxt);
/* Kinds of ahead-of-time compilation, for use with
gcc_jit_context_compile_to_file. */
enum gcc_jit_output_kind
{
/* Compile the context to an assembler file. */
GCC_JIT_OUTPUT_KIND_ASSEMBLER,
/* Compile the context to an object file. */
GCC_JIT_OUTPUT_KIND_OBJECT_FILE,
/* Compile the context to a dynamic library. */
GCC_JIT_OUTPUT_KIND_DYNAMIC_LIBRARY,
/* Compile the context to an executable. */
GCC_JIT_OUTPUT_KIND_EXECUTABLE
};
/* Compile the context to a file of the given kind.
This can be called more that once on a given context,
although any errors that occur will block further compilation. */
extern void
gcc_jit_context_compile_to_file (gcc_jit_context *ctxt,
enum gcc_jit_output_kind output_kind,
const char *output_path);
/* To help with debugging: dump a C-like representation to the given path,
describing what's been set up on the context.
@ -1079,14 +1112,15 @@ gcc_jit_context_dump_reproducer_to_file (gcc_jit_context *ctxt,
The context directly stores the dumpname as a (const char *), so the
passed string must outlive the context.
gcc_jit_context_compile will capture the dump as a
dynamically-allocated buffer, writing it to ``*out_ptr``.
gcc_jit_context_compile and gcc_jit_context_to_file
will capture the dump as a dynamically-allocated buffer, writing
it to ``*out_ptr``.
The caller becomes responsible for calling
free (*out_ptr)
each time that gcc_jit_context_compile is called. *out_ptr will be
written to, either with the address of a buffer, or with NULL if an
error occurred.
each time that gcc_jit_context_compile or gcc_jit_context_to_file
are called. *out_ptr will be written to, either with the address of a
buffer, or with NULL if an error occurred.
This API entrypoint is likely to be less stable than the others.
In particular, both the precise dumpnames, and the format and content

View File

@ -32,6 +32,7 @@
gcc_jit_block_get_function;
gcc_jit_context_acquire;
gcc_jit_context_compile;
gcc_jit_context_compile_to_file;
gcc_jit_context_dump_to_file;
gcc_jit_context_dump_reproducer_to_file;
gcc_jit_context_enable_dump;

View File

@ -74,9 +74,16 @@ Client Code . Generated . libgccjit.so
. . . . │ (purge internal state)
. . <──────────────────────── end of toplev::finalize
. . │ . .
. . │ Convert assembler to DSO ("fake.so")
. . │ . .
. . │ Load DSO (dlopen "fake.so")
. . V─> playback::context::postprocess:
. . │ . .
. . │ (assuming an in-memory compile):
. . │ . .
. . │ . Convert assembler to DSO ("fake.so")
. . │ . .
. . │ . Load DSO (dlopen "fake.so")
. . │ . .
. . │ . Bundle it up in a jit::result
. . <── . .
. . │ . .
. . │ RELEASE MUTEX .
. . │ . .

View File

@ -1,3 +1,28 @@
2015-01-19 David Malcolm <dmalcolm@redhat.com>
* jit.dg/harness.h: Include <unistd.h>.
(CHECK_NO_ERRORS): New.
(verify_code): Wrap prototype in #ifndef TEST_COMPILING_TO_FILE.
(test_jit): Support new macro TEST_COMPILING_TO_FILE for exercising
gcc_jit_context_compile_to_file.
* jit.dg/jit.exp (fixed_host_execute): Fix the code for passing on
args to the spawned executable.
(jit-expand-vars): New function.
(jit-exe-params): New variable.
(dg-jit-set-exe-params): New function.
(jit-dg-test): Detect testcases that use
jit-verify-compile-to-file and call jit-setup-compile-to-file.
Set arguments of spawned process to jit-exe-params.
(jit-get-output-filename): New function.
(jit-setup-compile-to-file): New function.
(jit-verify-compile-to-file): New function.
(jit-run-executable): New function.
(jit-verify-executable): New function.
* jit.dg/test-compile-to-assembler.c: New testcase.
* jit.dg/test-compile-to-dynamic-library.c: New testcase.
* jit.dg/test-compile-to-executable.c: New testcase.
* jit.dg/test-compile-to-object.c: New testcase.
2015-01-19 Oleg Endo <olegendo@gcc.gnu.org>
PR target/64652

View File

@ -7,12 +7,14 @@
extern void
create_code (gcc_jit_context *ctxt, void * user_data);
and, #ifndef TEST_COMPILING_TO_FILE,
extern void
verify_code (gcc_jit_context *ctxt, gcc_jit_result *result);
*/
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
/* test-threads.c use threads, but dejagnu.h isn't thread-safe; there's a
shared "buffer", and the counts of passed/failed etc are globals.
@ -106,12 +108,23 @@ static char test[1024];
} \
} while (0)
#define CHECK_NO_ERRORS(CTXT) \
do { \
const char *err = gcc_jit_context_get_first_error (CTXT); \
if (err) \
fail ("error unexpectedly occurred: %s", err); \
else \
pass ("no errors occurred"); \
} while (0)
/* Hooks that testcases should provide. */
extern void
create_code (gcc_jit_context *ctxt, void * user_data);
#ifndef TEST_COMPILING_TO_FILE
extern void
verify_code (gcc_jit_context *ctxt, gcc_jit_result *result);
#endif
extern void check_string_value (const char *funcname,
const char *actual, const char *expected);
@ -322,7 +335,13 @@ test_jit (const char *argv0, void *user_data)
{
gcc_jit_context *ctxt;
FILE *logfile;
#ifndef TEST_COMPILING_TO_FILE
gcc_jit_result *result;
#endif
#ifdef TEST_COMPILING_TO_FILE
unlink (OUTPUT_FILENAME);
#endif
ctxt = gcc_jit_context_acquire ();
if (!ctxt)
@ -339,16 +358,24 @@ test_jit (const char *argv0, void *user_data)
dump_reproducer (ctxt, argv0);
#ifdef TEST_COMPILING_TO_FILE
gcc_jit_context_compile_to_file (ctxt,
(OUTPUT_KIND),
(OUTPUT_FILENAME));
#else /* #ifdef TEST_COMPILING_TO_FILE */
/* This actually calls into GCC and runs the build, all
in a mutex for now. */
result = gcc_jit_context_compile (ctxt);
verify_code (ctxt, result);
#endif
gcc_jit_context_release (ctxt);
#ifndef TEST_COMPILING_TO_FILE
/* Once we're done with the code, this unloads the built .so file: */
gcc_jit_result_release (result);
#endif
if (logfile)
fclose (logfile);

View File

@ -139,6 +139,8 @@ proc fixed_host_execute {args} {
global text
global spawn_id
verbose "fixed_host_execute: $args"
set timeoutmsg "Timed out: Never got started, "
set timeout 100
set file all
@ -148,9 +150,8 @@ proc fixed_host_execute {args} {
if { [llength $args] == 0} {
set executable $args
} else {
set executable [string trimleft [lindex [split $args " "] 0] "\{"]
set params [string trimleft [lindex [split $args " "] 1] "\{"]
set params [string trimright $params "\}"]
set executable [lindex $args 0]
set params [lindex $args 1]
}
verbose "The executable is $executable" 2
@ -159,6 +160,8 @@ proc fixed_host_execute {args} {
return "No source file found"
}
verbose "params: $params" 2
# spawn the executable and look for the DejaGnu output messages from the
# test case.
# spawn -noecho -open [open "|./${executable}" "r"]
@ -328,6 +331,39 @@ proc get_path_of_driver {} {
return [file dirname $binary]
}
# Expand "SRCDIR" within ARG to the location of the top-level
# src directory
proc jit-expand-vars {arg} {
verbose "jit-expand-vars: $arg"
global srcdir
verbose " srcdir: $srcdir"
# "srcdir" is that of the gcc/testsuite directory, so
# we need to go up two levels.
set arg [string map [list "SRCDIR" $srcdir/../..] $arg]
verbose " new arg: $arg"
return $arg
}
# Parameters used when invoking the executables built from the test cases.
global jit-exe-params
set jit-exe-params {}
# Set "jit-exe-params", expanding "SRCDIR" in each arg to the location of
# the top-level srcdir.
proc dg-jit-set-exe-params { args } {
verbose "dg-jit-set-exe-params: $args"
global jit-exe-params
set jit-exe-params {}
# Skip initial arg (line number)
foreach arg [lrange $args 1 [llength $args] ] {
lappend jit-exe-params [jit-expand-vars $arg]
}
}
proc jit-dg-test { prog do_what extra_tool_flags } {
verbose "within jit-dg-test..."
verbose " prog: $prog"
@ -339,6 +375,15 @@ proc jit-dg-test { prog do_what extra_tool_flags } {
append extra_tool_flags " -lpthread"
}
# Any test case that uses
# { dg-final { jit-verify-compile-to-file FOO } }
# needs to call jit-setup-compile-to-file here.
# (is there a better way to handle setup/finish pairs in dg?)
set tmp [grep $prog "jit-verify-compile-to-file"]
if {![string match "" $tmp]} {
jit-setup-compile-to-file $prog
}
# Determine what to name the built executable.
#
# We simply append .exe to the filename, e.g.
@ -464,7 +509,12 @@ proc jit-dg-test { prog do_what extra_tool_flags } {
# http://lists.gnu.org/archive/html/dejagnu/2014-10/msg00000.html
# We instead call a patched local copy, "fixed_host_execute", defined
# above.
set result [fixed_host_execute $output_file]
global jit-exe-params
set args ${jit-exe-params}
set jit-exe-params {}
set result [fixed_host_execute $output_file $args ]
verbose "result: $result"
# Restore PATH
@ -530,6 +580,157 @@ proc jit-dg-test { prog do_what extra_tool_flags } {
return [list $comp_output $output_file]
}
# Given source file PROG, scrape out the value of
# #define OUTPUT_FILENAME
# failing if it's not found.
proc jit-get-output-filename {prog} {
set tmp [grep $prog "#define OUTPUT_FILENAME (.*)"]
if {![string match "" $tmp]} {
foreach i $tmp {
verbose "i: $i"
if {[regexp "^\#define OUTPUT_FILENAME\[ \t\]\+\"(.*)\"$" $i i group] } {
verbose "group: '$group'"
return $group
} else {
fail "Unable to parse line: $i"
}
}
}
fail "Unable to locate OUTPUT_FILENAME"
return ""
}
# For testcases that use jit-verify-compile-to-file,
# delete OUTPUT_FILENAME beforehand, to ensure that the
# testcase is indeed creating it.
proc jit-setup-compile-to-file { prog } {
verbose "jit-setup-compile-to-file: $prog"
set output_filename [jit-get-output-filename $prog]
verbose " output_filename: $output_filename"
if {![string match "" $output_filename]} {
catch "exec rm -f $output_filename"
}
}
# Locate OUTPUT_FILENAME within the testcase. Verify
# that a file with that name was created, and that
# the output of running the "file" utility on it
# matches the given regex.
# For use by the various test-compile-to-*.c testcases.
proc jit-verify-compile-to-file {args} {
verbose "jit-verify-compile-to-file: $args"
set file_regex [lindex $args 0]
verbose "file_regex: $file_regex"
upvar 2 prog prog
verbose "prog: $prog"
set output_filename [jit-get-output-filename $prog]
verbose " output_filename: $output_filename"
# Verify that the expected file was written out
if { [file exists $output_filename] == 1} {
pass "$output_filename exists"
} else {
fail "$output_filename does not exist"
return
}
# Run "file" on OUTPUT_FILENAME, and verify that the output
# matches $file_regex.
spawn -noecho "file" $output_filename
set file_id $spawn_id
expect "\n" {
verbose "got newline: $expect_out(buffer)"
set classification $expect_out(buffer)
verbose "classification: $classification"
if { [regexp $file_regex $classification] } {
pass "'file' output on $output_filename matched: $file_regex"
} else {
fail "'file' output on $output_filename did not match: $file_regex"
}
}
set $spawn_id $file_id
close
}
# Verify that the given file exists, and is executable.
# Attempt to execute it, and verify that its stdout matches
# the given regex.
proc jit-run-executable { args } {
verbose "jit-run-executable: $args"
set executable-name [lindex $args 0]
verbose "executable-name: ${executable-name}"
set dg-output-text [lindex $args 1]
verbose "dg-output-text: ${dg-output-text}"
if { [file executable ${executable-name}] } {
pass "${executable-name} has executable bit set"
} else {
fail "${executable-name} does not have executable bit set"
}
# Attempt to run the executable; adapted from dg.exp's dg-test
set status -1
set result [jit_load ./${executable-name}]
set status [lindex $result 0]
set output [lindex $result 1]
verbose " status: $status"
verbose " output: $output"
# send_user "After exec, status: $status\n"
if { "$status" == "pass" } {
pass "${executable-name} execution test"
verbose "Exec succeeded." 3
set texttmp ${dg-output-text}
if { ![regexp $texttmp ${output}] } {
fail "${executable-name} output pattern test, is ${output}, should match $texttmp"
verbose "Failed test for output pattern $texttmp" 3
} else {
pass "${executable-name} output pattern test, $texttmp"
verbose "Passed test for output pattern $texttmp" 3
}
unset texttmp
} elseif { "$status" == "fail" } {
# It would be nice to get some info out of errorCode.
if {[info exists errorCode]} {
verbose "Exec failed, errorCode: $errorCode" 3
} else {
verbose "Exec failed, errorCode not defined!" 3
}
fail "${executable-name} execution test"
} else {
$status "${executable-name} execution test"
}
}
# A way to invoke "jit-run-executable" with the given regex,
# using OUTPUT_FILENAME within the testcase to determine
# the name of the executable to run.
# For use by the test-compile-to-executable.c testcase.
proc jit-verify-executable { args } {
verbose "jit-verify-executable: $args"
set dg-output-text [lindex $args 0]
verbose "dg-output-text: ${dg-output-text}"
upvar 2 name name
verbose "name: $name"
upvar 2 prog prog
verbose "prog: $prog"
set output_filename [jit-get-output-filename $prog]
verbose " output_filename: $output_filename"
jit-run-executable $output_filename ${dg-output-text}
}
# We need to link with --export-dynamic for test-calling-external-function.c
# so that the JIT-built code can call into functions from the main program.
set DEFAULT_CFLAGS "-I$srcdir/../jit -lgccjit -g -Wall -Werror -Wl,--export-dynamic"

View File

@ -0,0 +1,65 @@
#include <stdlib.h>
#include <stdio.h>
#include "libgccjit.h"
#define TEST_COMPILING_TO_FILE
#define OUTPUT_KIND GCC_JIT_OUTPUT_KIND_ASSEMBLER
#define OUTPUT_FILENAME "output-of-test-compile-to-assembler.c.s"
#include "harness.h"
void
create_code (gcc_jit_context *ctxt, void *user_data)
{
/* Let's try to inject the equivalent of:
void
hello_world (const char *name)
{
// a test comment
printf ("hello %s\n", name);
}
*/
gcc_jit_type *void_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_VOID);
gcc_jit_type *const_char_ptr_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CONST_CHAR_PTR);
gcc_jit_param *param_name =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "name");
gcc_jit_function *func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_EXPORTED,
void_type,
"hello_world",
1, &param_name,
0);
gcc_jit_param *param_format =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "format");
gcc_jit_function *printf_func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
gcc_jit_context_get_type (
ctxt, GCC_JIT_TYPE_INT),
"printf",
1, &param_format,
1);
gcc_jit_rvalue *args[2];
args[0] = gcc_jit_context_new_string_literal (ctxt, "hello %s\n");
args[1] = gcc_jit_param_as_rvalue (param_name);
gcc_jit_block *block = gcc_jit_function_new_block (func, NULL);
gcc_jit_block_add_comment (
block, NULL,
"a test comment");
gcc_jit_block_add_eval (
block, NULL,
gcc_jit_context_new_call (ctxt,
NULL,
printf_func,
2, args));
gcc_jit_block_end_with_void_return (block, NULL);
}
/* { dg-final { jit-verify-compile-to-file "assembler source text" } } */

View File

@ -0,0 +1,65 @@
#include <stdlib.h>
#include <stdio.h>
#include "libgccjit.h"
#define TEST_COMPILING_TO_FILE
#define OUTPUT_KIND GCC_JIT_OUTPUT_KIND_DYNAMIC_LIBRARY
#define OUTPUT_FILENAME "output-of-test-compile-to-dynamic-library.c.so"
#include "harness.h"
void
create_code (gcc_jit_context *ctxt, void *user_data)
{
/* Let's try to inject the equivalent of:
void
hello_world (const char *name)
{
// a test comment
printf ("hello %s\n", name);
}
*/
gcc_jit_type *void_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_VOID);
gcc_jit_type *const_char_ptr_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CONST_CHAR_PTR);
gcc_jit_param *param_name =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "name");
gcc_jit_function *func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_EXPORTED,
void_type,
"hello_world",
1, &param_name,
0);
gcc_jit_param *param_format =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "format");
gcc_jit_function *printf_func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
gcc_jit_context_get_type (
ctxt, GCC_JIT_TYPE_INT),
"printf",
1, &param_format,
1);
gcc_jit_rvalue *args[2];
args[0] = gcc_jit_context_new_string_literal (ctxt, "hello %s\n");
args[1] = gcc_jit_param_as_rvalue (param_name);
gcc_jit_block *block = gcc_jit_function_new_block (func, NULL);
gcc_jit_block_add_comment (
block, NULL,
"a test comment");
gcc_jit_block_add_eval (
block, NULL,
gcc_jit_context_new_call (ctxt,
NULL,
printf_func,
2, args));
gcc_jit_block_end_with_void_return (block, NULL);
}
/* { dg-final { jit-verify-compile-to-file "shared object.+dynamically linked" } } */

View File

@ -0,0 +1,110 @@
#include <stdlib.h>
#include <stdio.h>
#include "libgccjit.h"
#define TEST_COMPILING_TO_FILE
#define OUTPUT_KIND GCC_JIT_OUTPUT_KIND_EXECUTABLE
#define OUTPUT_FILENAME "output-of-test-compile-to-executable.c.exe"
#include "harness.h"
void
create_code (gcc_jit_context *ctxt, void *user_data)
{
/* Let's try to inject the equivalent of:
static void
hello_world (const char *name)
{
// a test comment
printf ("hello %s\n", name);
}
extern int
main (int argc, char **argv)
{
hello_world (argv[0]);
return 0;
}
*/
gcc_jit_type *void_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_VOID);
gcc_jit_type *const_char_ptr_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CONST_CHAR_PTR);
gcc_jit_param *param_name =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "name");
gcc_jit_function *func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_INTERNAL,
void_type,
"hello_world",
1, &param_name,
0);
gcc_jit_param *param_format =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "format");
gcc_jit_function *printf_func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
gcc_jit_context_get_type (
ctxt, GCC_JIT_TYPE_INT),
"printf",
1, &param_format,
1);
gcc_jit_rvalue *args[2];
args[0] = gcc_jit_context_new_string_literal (ctxt, "hello %s\n");
args[1] = gcc_jit_param_as_rvalue (param_name);
gcc_jit_block *block = gcc_jit_function_new_block (func, NULL);
gcc_jit_block_add_comment (
block, NULL,
"a test comment");
gcc_jit_block_add_eval (
block, NULL,
gcc_jit_context_new_call (ctxt,
NULL,
printf_func,
2, args));
gcc_jit_block_end_with_void_return (block, NULL);
gcc_jit_type *int_type = gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_INT);
gcc_jit_param *param_argc =
gcc_jit_context_new_param (ctxt, NULL, int_type, "argc");
gcc_jit_type *char_ptr_ptr_type =
gcc_jit_type_get_pointer (
gcc_jit_type_get_pointer (
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CHAR)));
gcc_jit_param *param_argv =
gcc_jit_context_new_param (ctxt, NULL, char_ptr_ptr_type, "argv");
gcc_jit_param *params[2] = {param_argc, param_argv};
gcc_jit_function *func_main =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_EXPORTED,
int_type,
"main",
2, params,
0);
block = gcc_jit_function_new_block (func_main, NULL);
gcc_jit_rvalue *zero = gcc_jit_context_zero (ctxt, int_type);
args[0] = gcc_jit_context_new_cast (
ctxt,
NULL,
gcc_jit_lvalue_as_rvalue (
gcc_jit_context_new_array_access (
ctxt,
NULL,
gcc_jit_param_as_rvalue (param_argv),
zero)),
const_char_ptr_type);
gcc_jit_block_add_eval (
block, NULL,
gcc_jit_context_new_call (ctxt,
NULL,
func,
1, args));
gcc_jit_block_end_with_return (block, NULL, zero);
}
/* { dg-final { jit-verify-compile-to-file "executable" } } */
/* { dg-final { jit-verify-executable "hello .*" } } */

View File

@ -0,0 +1,65 @@
#include <stdlib.h>
#include <stdio.h>
#include "libgccjit.h"
#define TEST_COMPILING_TO_FILE
#define OUTPUT_KIND GCC_JIT_OUTPUT_KIND_OBJECT_FILE
#define OUTPUT_FILENAME "output-of-test-compile-to-object.c.o"
#include "harness.h"
void
create_code (gcc_jit_context *ctxt, void *user_data)
{
/* Let's try to inject the equivalent of:
void
hello_world (const char *name)
{
// a test comment
printf ("hello %s\n", name);
}
*/
gcc_jit_type *void_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_VOID);
gcc_jit_type *const_char_ptr_type =
gcc_jit_context_get_type (ctxt, GCC_JIT_TYPE_CONST_CHAR_PTR);
gcc_jit_param *param_name =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "name");
gcc_jit_function *func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_EXPORTED,
void_type,
"hello_world",
1, &param_name,
0);
gcc_jit_param *param_format =
gcc_jit_context_new_param (ctxt, NULL, const_char_ptr_type, "format");
gcc_jit_function *printf_func =
gcc_jit_context_new_function (ctxt, NULL,
GCC_JIT_FUNCTION_IMPORTED,
gcc_jit_context_get_type (
ctxt, GCC_JIT_TYPE_INT),
"printf",
1, &param_format,
1);
gcc_jit_rvalue *args[2];
args[0] = gcc_jit_context_new_string_literal (ctxt, "hello %s\n");
args[1] = gcc_jit_param_as_rvalue (param_name);
gcc_jit_block *block = gcc_jit_function_new_block (func, NULL);
gcc_jit_block_add_comment (
block, NULL,
"a test comment");
gcc_jit_block_add_eval (
block, NULL,
gcc_jit_context_new_call (ctxt,
NULL,
printf_func,
2, args));
gcc_jit_block_end_with_void_return (block, NULL);
}
/* { dg-final { jit-verify-compile-to-file "relocatable" } } */