* tree-pretty-print.c (dump_generic_node): Handle OMP_PARALLEL_FN, OMP_PARALLEL_DATA_ARG and OMP_RETURN_EXPR. * cgraph.c (cgraph_expand_queue): Rename from cgraph_analyze_queue. Update all users. * cgraphunit.c (cgraph_assemble_pending_functions): Process cgraph_expand_queue. (cgraph_expand_all_functions): Likewise. (cgraph_finalize_pending_functions): Remove. Update callers. * tree.h (OMP_DIRECTIVE_P): Define. (OMP_PARALLEL_FN): Define. (OMP_PARALLEL_DATA_ARG): Define. (OMP_SECTIONS_SECTIONS): Define. * tree-pass.h (pass_expand_omp): Declare. * omp-low.c (struct omp_region): Declare. (struct omp_context): Remove fields 'parallel_type', 'parallel_start_ix' and 'parallel_start_additional_args'. Update all users. (struct omp_for_data): Rename from struct expand_omp_for_data. (omp_regions): New static variable. (root_omp_region): New static variable. (find_omp_clause): Make static. (is_in_combined_parallel_ctx): Remove. (is_combined_parallel): New. (extract_omp_for_data): Move earlier in the file. (workshare_safe_to_combine_p): New. (get_ws_args_for): New. (determine_parallel_type): Move earlier in the file. (omp_copy_decl_2): Do not set DECL_CONTEXT of new local to the child function. (omp_copy_decl): Likewise. (create_omp_child_function): Likewise. (lookup_omp_region): New. (dump_omp_region): New. (debug_omp_region): New. (debug_all_omp_regions): New. (new_omp_region): New. (scan_omp_parallel): If parallel_nesting_level > 1, the directive is nested within another parallel directive. Set OMP_PARALLEL_FN. (scan_omp_for): Do not try to handle combined parallel+for cases. Remove FIXME comment. (scan_omp_nested): Remove. (scan_omp_1): Do not call scan_omp_nested when parallel_nesting_level is > 1. Do not change the DECL_CONTEXT of local variables found. (lookup_decl_in_outer_ctx): New. (lower_rec_input_clauses): Rename from expand_rec_input_clauses. (lower_lastprivate_clauses): Rename from expand_lastprivate_clauses. (lower_reduction_clauses): Rename from expand_reduction_clauses. (lower_copyprivate_clauses): Rename from expand_copyprivate_clauses. If CTX is nested, lookup VAR in the outer context when building copy assignment. (lower_send_clauses): Rename from expand_send_clauses. If CTX is nested, lookup VAR in the outer context when building copy assignments. (lower_send_shared_vars): Rename from expand_send_shared_vars. If CTX is nested, lookup VAR in the outer context when building copy assignments. (expand_parallel_call): Rename from build_parallel_call. Handle combined parallel+workshare cases. Re-implement to emit code into the CFG. (list2chain): New. (expand_omp_parallel): Re-implement to emit code into the CFG. Call move_sese_region_to_fn to outline the sub-graph containing the parallel region. (expand_omp_for_1): Remove. (expand_omp_for_generic): Re-implement to emit code into the CFG. (expand_omp_for_static_nochunk): Likewise. (expand_omp_for_static_chunk): Likewise. (expand_omp_for): Likewise. (expand_omp_sections): Likewise. (remove_exit_barriers): New. (expand_omp_synch): New. (expand_omp): New. (build_omp_regions_1): New. (build_omp_regions): New. (execute_expand_omp): New. (gate_expand_omp): New. (pass_expand_omp): Define. (lower_omp_sections): Rename from expand_omp_sections. Set OMP_SECTIONS_SECTIONS. (lower_omp_single_simple): Rename from expand_omp_single_simple. (lower_omp_single_copy): Rename from expand_omp_single_copy. (lower_omp_single): Rename from expand_omp_simple. (lower_omp_master): Rename from expand_omp_master. (lower_omp_ordered): Rename from expand_omp_ordered. (lower_omp_critical): Rename from expand_omp_critical. (lower_omp_for_lastprivate): Rename from expand_omp_for_lastprivate. (lower_omp_for): Re-implement. (lower_omp_parallel): Re-implement. (lower_regimplify): Rename from expand_regimplify. (lower_omp_1): Rename from expand_omp_1. If there are syntax errors in the program, replace every OpenMP directive with NOP. Call lower_omp_* instead of expand_omp_*. (lower_omp): Rename from expand_omp. * tree-gimple.c (is_gimple_stmt): Handle OMP_RETURN_EXPR. * tree-gimple.h (enum omp_parallel_type): Remove. (gimple_boolify): Declare extern. (find_omp_clause, determine_parallel_type): Remove. * gimple-low.c (lower_omp_directive): New. (lower_stmt): Call it. (record_vars_into): Move from ... (record_vars): ... here. Call record_vars_into with current_function_decl. * gimplify.c (struct gimplify_ctx): Remove fields combined_pre_p and combined_ctxp. Update users. (get_formal_tmp_var): Add documentation. (gimple_boolify): Make extern. (gimplify_expr_in_ctx): Remove. Update callers. (gimplify_omp_parallel): Do not assume that OMP_PARALLEL_BODY will always be a BIND_EXPR. (gimplify_expr): Handle OMP_RETURN_EXPR. * tree.def (BLOCK): Remove documentation about BLOCK_TYPE_TAGS. (OMP_PARALLEL): Add 3 operands. (OMP_SECTIONS): Add 1 operand. (OMP_RETURN_EXPR): Define. * tree-inline.c (estimate_num_insns_1): Handle OpenMP directives. (copy_tree_r): Restore TREE_CHAIN in OMP_CLAUSE_*. * tree-iterator.c (alloc_stmt_list): Assert that we are not creating a circular free list. (free_stmt_list): Assert that we are not freeing stmt_list_cache. * tree-flow.h (move_sese_region_to_fn): Declare. (record_vars_into): Declare. * tree-cfg.c (make_omp_sections_edges): New. (make_exit_edges): Handle OMP_PARALLEL, OMP_FOR, OMP_SINGLE, OMP_MASTER, OMP_ORDERED, OMP_CRITICAL, OMP_RETURN_EXPR, OMP_SECTIONS and OMP_SECTION. (is_ctrl_altering_stmt): Return true for OMP_DIRECTIVE_P. (set_bb_for_stmt): Undo change to check currently_expanding_to_rtl. (verify_stmt): Do not handle OMP_DIRECTIVE_P. (gather_blocks_in_sese_region): New. (struct move_stmt_d): Declare. (move_stmt_r): New. (move_block_to_fn): New. (move_sese_region_to_fn): New. * passes.c (init_optimization_passes): Schedule pass_expand_omp after pass_init_datastructures. * tree-ssa-operands.c (get_expr_operands): Handle OMP_PARALLEL, OMP_SECTIONS, OMP_FOR, OMP_RETURN_EXPR, OMP_SINGLE, OMP_MASTER, OMP_ORDERED, OMP_CRITICAL. testsuite/ * testsuite/gcc.dg/gomp/for-13.c: Use -fdump-tree-ompexp. * testsuite/gcc.dg/gomp/critical-1.c: Likewise. * testsuite/gcc.dg/gomp/critical-3.c: Likewise. * testsuite/gcc.dg/gomp/empty.c: Likewise. * testsuite/gcc.dg/gomp/ordered-1.c: Likewise. * testsuite/gcc.dg/gomp/for-4.c: Likewise. * testsuite/gcc.dg/gomp/for-6.c: Likewise. * testsuite/gcc.dg/gomp/master-3.c: Likewise. * testsuite/gcc.dg/gomp/for-8.c: Likewise. * testsuite/gcc.dg/gomp/for-10.c: Likewise. * testsuite/gcc.dg/gomp/for-18.c: Likewise. * testsuite/gcc.dg/gomp/for-5.c: Likewise. * testsuite/gcc.dg/gomp/for-7.c: Likewise. * testsuite/gcc.dg/gomp/for-9.c: Likewise. From-SVN: r109969
206 lines
7.7 KiB
C
206 lines
7.7 KiB
C
/* Functions to analyze and validate GIMPLE trees.
|
|
Copyright (C) 2002, 2003, 2005 Free Software Foundation, Inc.
|
|
Contributed by Diego Novillo <dnovillo@redhat.com>
|
|
|
|
This file is part of GCC.
|
|
|
|
GCC 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 2, or (at your option)
|
|
any later version.
|
|
|
|
GCC 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 GCC; see the file COPYING. If not, write to
|
|
the Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA. */
|
|
|
|
#ifndef _TREE_SIMPLE_H
|
|
#define _TREE_SIMPLE_H 1
|
|
|
|
|
|
#include "tree-iterator.h"
|
|
|
|
extern tree create_tmp_var_raw (tree, const char *);
|
|
extern tree create_tmp_var_name (const char *);
|
|
extern tree create_tmp_var (tree, const char *);
|
|
extern tree get_initialized_tmp_var (tree, tree *, tree *);
|
|
extern tree get_formal_tmp_var (tree, tree *);
|
|
extern void declare_tmp_vars (tree, tree);
|
|
|
|
extern void annotate_all_with_locus (tree *, location_t);
|
|
|
|
/* Validation of GIMPLE expressions. Note that these predicates only check
|
|
the basic form of the expression, they don't recurse to make sure that
|
|
underlying nodes are also of the right form. */
|
|
|
|
typedef bool (*gimple_predicate)(tree);
|
|
|
|
/* Returns true iff T is a valid GIMPLE statement. */
|
|
extern bool is_gimple_stmt (tree);
|
|
|
|
/* Returns true iff TYPE is a valid type for a scalar register variable. */
|
|
extern bool is_gimple_reg_type (tree);
|
|
/* Returns true iff T is a scalar register variable. */
|
|
extern bool is_gimple_reg (tree);
|
|
/* Returns true if T is a GIMPLE temporary variable, false otherwise. */
|
|
extern bool is_gimple_formal_tmp_var (tree);
|
|
/* Returns true if T is a GIMPLE temporary register variable. */
|
|
extern bool is_gimple_formal_tmp_reg (tree);
|
|
/* Returns true iff T is any sort of variable. */
|
|
extern bool is_gimple_variable (tree);
|
|
/* Returns true iff T is any sort of symbol. */
|
|
extern bool is_gimple_id (tree);
|
|
/* Returns true iff T is a variable or an INDIRECT_REF (of a variable). */
|
|
extern bool is_gimple_min_lval (tree);
|
|
/* Returns true iff T is something whose address can be taken. */
|
|
extern bool is_gimple_addressable (tree);
|
|
/* Returns true iff T is any valid GIMPLE lvalue. */
|
|
extern bool is_gimple_lvalue (tree);
|
|
|
|
/* Returns true iff T is a GIMPLE restricted function invariant. */
|
|
extern bool is_gimple_min_invariant (tree);
|
|
/* Returns true iff T is a GIMPLE rvalue. */
|
|
extern bool is_gimple_val (tree);
|
|
/* Returns true iff T is a GIMPLE asm statement input. */
|
|
extern bool is_gimple_asm_val (tree);
|
|
/* Returns true iff T is a valid rhs for a MODIFY_EXPR where the LHS is a
|
|
GIMPLE temporary, a renamed user variable, or something else,
|
|
respectively. */
|
|
extern bool is_gimple_formal_tmp_rhs (tree);
|
|
extern bool is_gimple_reg_rhs (tree);
|
|
extern bool is_gimple_mem_rhs (tree);
|
|
/* Returns the appropriate one of the above three predicates for the LHS
|
|
T. */
|
|
extern gimple_predicate rhs_predicate_for (tree);
|
|
|
|
/* Returns true iff T is a valid if-statement condition. */
|
|
extern bool is_gimple_condexpr (tree);
|
|
|
|
/* Returns true iff T is a type conversion. */
|
|
extern bool is_gimple_cast (tree);
|
|
/* Returns true iff T is a variable that does not need to live in memory. */
|
|
extern bool is_gimple_non_addressable (tree t);
|
|
|
|
/* Returns true iff T is a valid call address expression. */
|
|
extern bool is_gimple_call_addr (tree);
|
|
/* If T makes a function call, returns the CALL_EXPR operand. */
|
|
extern tree get_call_expr_in (tree t);
|
|
|
|
extern void recalculate_side_effects (tree);
|
|
|
|
/* FIXME we should deduce this from the predicate. */
|
|
typedef enum fallback_t {
|
|
fb_none = 0,
|
|
fb_rvalue = 1,
|
|
fb_lvalue = 2,
|
|
fb_mayfail = 4,
|
|
fb_either= fb_rvalue | fb_lvalue
|
|
} fallback_t;
|
|
|
|
enum gimplify_status {
|
|
GS_ERROR = -2, /* Something Bad Seen. */
|
|
GS_UNHANDLED = -1, /* A langhook result for "I dunno". */
|
|
GS_OK = 0, /* We did something, maybe more to do. */
|
|
GS_ALL_DONE = 1 /* The expression is fully gimplified. */
|
|
};
|
|
|
|
extern enum gimplify_status gimplify_expr (tree *, tree *, tree *,
|
|
bool (*) (tree), fallback_t);
|
|
extern void gimplify_type_sizes (tree, tree *);
|
|
extern void gimplify_one_sizepos (tree *, tree *);
|
|
extern void gimplify_stmt (tree *);
|
|
extern void gimplify_to_stmt_list (tree *);
|
|
extern void gimplify_body (tree *, tree, bool);
|
|
extern void push_gimplify_context (void);
|
|
extern void pop_gimplify_context (tree);
|
|
extern void gimplify_and_add (tree, tree *);
|
|
|
|
/* Miscellaneous helpers. */
|
|
extern void gimple_add_tmp_var (tree);
|
|
extern tree gimple_current_bind_expr (void);
|
|
extern tree voidify_wrapper_expr (tree, tree);
|
|
extern tree gimple_build_eh_filter (tree, tree, tree);
|
|
extern tree build_and_jump (tree *);
|
|
extern tree alloc_stmt_list (void);
|
|
extern void free_stmt_list (tree);
|
|
extern tree force_labels_r (tree *, int *, void *);
|
|
extern enum gimplify_status gimplify_va_arg_expr (tree *, tree *, tree *);
|
|
struct gimplify_omp_ctx;
|
|
extern void omp_firstprivatize_variable (struct gimplify_omp_ctx *, tree);
|
|
extern tree gimple_boolify (tree);
|
|
|
|
/* In omp-low.c. */
|
|
extern void diagnose_omp_structured_block_errors (tree);
|
|
extern tree omp_reduction_init (tree, tree);
|
|
|
|
/* In tree-nested.c. */
|
|
extern void lower_nested_functions (tree);
|
|
extern void insert_field_into_struct (tree, tree);
|
|
|
|
/* Convenience routines to walk all statements of a gimple function.
|
|
The difference between these walkers and the generic walk_tree is
|
|
that walk_stmt provides context information to the callback
|
|
routine to know whether it is currently on the LHS or RHS of an
|
|
assignment (IS_LHS) or contexts where only GIMPLE values are
|
|
allowed (VAL_ONLY).
|
|
|
|
This is useful in walkers that need to re-write sub-expressions
|
|
inside statements while making sure the result is still in GIMPLE
|
|
form.
|
|
|
|
Note that this is useful exclusively before the code is converted
|
|
into SSA form. Once the program is in SSA form, the standard
|
|
operand interface should be used to analyze/modify statements. */
|
|
|
|
struct walk_stmt_info
|
|
{
|
|
/* For each statement, we invoke CALLBACK via walk_tree. The passed
|
|
data is a walk_stmt_info structure. */
|
|
walk_tree_fn callback;
|
|
|
|
/* Points to the current statement being walked. */
|
|
tree_stmt_iterator tsi;
|
|
|
|
/* Additional data that CALLBACK may want to carry through the
|
|
recursion. */
|
|
void *info;
|
|
|
|
/* Indicates whether the *TP being examined may be replaced
|
|
with something that matches is_gimple_val (if true) or something
|
|
slightly more complicated (if false). "Something" technically
|
|
means the common subset of is_gimple_lvalue and is_gimple_rhs,
|
|
but we never try to form anything more complicated than that, so
|
|
we don't bother checking.
|
|
|
|
Also note that CALLBACK should update this flag while walking the
|
|
sub-expressions of a statement. For instance, when walking the
|
|
statement 'foo (&var)', the flag VAL_ONLY will initially be set
|
|
to true, however, when walking &var, the operand of that
|
|
ADDR_EXPR does not need to be a GIMPLE value. */
|
|
bool val_only;
|
|
|
|
/* True if we are currently walking the LHS of an assignment. */
|
|
bool is_lhs;
|
|
|
|
/* Optional. Set to true by CALLBACK if it made any changes. */
|
|
bool changed;
|
|
|
|
/* True if we're interested in seeing BIND_EXPRs. */
|
|
bool want_bind_expr;
|
|
|
|
/* True if we're interested in seeing RETURN_EXPRs. */
|
|
bool want_return_expr;
|
|
|
|
/* True if we're interested in location information. */
|
|
bool want_locations;
|
|
};
|
|
|
|
void walk_stmts (struct walk_stmt_info *, tree *);
|
|
|
|
#endif /* _TREE_SIMPLE_H */
|