On Wed, Sep 23, 2020 at 10:44 AM Jan Hubicka <[email protected]> wrote:
>
> Hi,
> this is first of cleanup patches for mod-ref interfaces. It removes code
> duplication between ipa-pure-const and ipa-modref that both wants to check
> whether given memory access can interfere with memory acesses before function
> call or after function return.
>
> I pulled the logic out to refs_local_or_readonly_memory_p for references and
> points_to_local_or_readonly_memory_p for pointers. It is in ipa-fnsummary.c
> because I have incremental patches to track this info about function parameter
> so we can ignore dereferences from parameters that are local or readonly in
> modref oracle.
>
> I am not sure this is right placement. We could also move it to more general
> place, ipa-utils or some of gimple infrastructure.
> The extra includes in ipa-modref will however soon be necessary.
>
> Bootstrapped/regtested x86_64-linux, OK?
>
> Honza
>
> * ipa-fnsummary.c (refs_local_or_readonly_memory_p): New function.
> (points_to_local_or_readonly_memory_p): New function.
> * ipa-fnsummary.h (refs_local_or_readonly_memory_p,
> points_to_local_or_readonly_memory_p): Declare.
> * ipa-modref.c: Include value-range.h, ipa-prop.h and ipa-fnsummary.h
> (record_access_p): Use refs_local_or_readonly_memory_p.
> * ipa-pure-const.c (check_op): Likewise.
>
> * testsuite/gcc.dg/tree-ssa/local-pure-const.c: Update template.
> diff --git a/gcc/ipa-fnsummary.c b/gcc/ipa-fnsummary.c
> index 86d01addb44..ce168230105 100644
> --- a/gcc/ipa-fnsummary.c
> +++ b/gcc/ipa-fnsummary.c
> @@ -2430,6 +2433,51 @@ fp_expression_p (gimple *stmt)
> return false;
> }
>
> +/* Return true if T references memory location that is local
> + for the function (that means, dead after return) or read-only. */
> +
> +bool
> +refs_local_or_readonly_memory_p (tree t)
> +{
> + /* Non-escaping memory is fine */
> + t = get_base_address (t);
> + if ((TREE_CODE (t) == MEM_REF
> + || TREE_CODE (t) == TARGET_MEM_REF))
> + return points_to_local_or_readonly_memory_p (TREE_OPERAND (t, 0));
> +
> + /* Automatic variables are fine. */
> + if (DECL_P (t)
> + && auto_var_in_fn_p (t, current_function_decl))
> + return true;
> +
> + /* Read-only variables are fine. */
> + if (DECL_P (t) && TREE_READONLY (t))
> + return true;
> +
> + return false;
> +}
> +
> +/* Return true if T is a pointer pointing to memory location that is local
> + for the function (that means, dead after return) or read-only. */
> +
> +bool
> +points_to_local_or_readonly_memory_p (tree t)
> +{
> + /*if (!POINTER_TYPE_P (TREE_TYPE (t)))
> + return false; */
remove ^^^
> + STRIP_NOPS (t);
This wasn't in the original code - did you really run into (long)&x or what?
> + /* See if memory location is clearly invalid. */
> + if (TREE_CODE (t) == INTEGER_CST)
> + return flag_delete_null_pointer_checks;
> + if (TREE_CODE (t) == SSA_NAME)
> + return SSA_NAME_POINTS_TO_READONLY_MEMORY (t)
> + || !ptr_deref_may_alias_global_p (t);
> + if (TREE_CODE (t) == ADDR_EXPR)
> + return refs_local_or_readonly_memory_p (TREE_OPERAND (t, 0));
> + return false;
> +}
> +
> +
> /* Analyze function body for NODE.
> EARLY indicates run from early optimization pipeline. */
>
> diff --git a/gcc/ipa-fnsummary.h b/gcc/ipa-fnsummary.h
> index c6ddc9f3199..a3223d06acc 100644
> --- a/gcc/ipa-fnsummary.h
> +++ b/gcc/ipa-fnsummary.h
> @@ -357,6 +357,8 @@ void estimate_ipcp_clone_size_and_time (struct
> cgraph_node *,
> void ipa_merge_fn_summary_after_inlining (struct cgraph_edge *edge);
> void ipa_update_overall_fn_summary (struct cgraph_node *node, bool reset =
> true);
> void compute_fn_summary (struct cgraph_node *, bool);
> +bool refs_local_or_readonly_memory_p (tree);
> +bool points_to_local_or_readonly_memory_p (tree);
>
>
> void evaluate_properties_for_edge (struct cgraph_edge *e,
> diff --git a/gcc/ipa-modref.c b/gcc/ipa-modref.c
> index af0b710333e..d1b091ae29e 100644
> --- a/gcc/ipa-modref.c
> +++ b/gcc/ipa-modref.c
> @@ -62,6 +62,9 @@ along with GCC; see the file COPYING3. If not see
> #include "calls.h"
> #include "ipa-modref-tree.h"
> #include "ipa-modref.h"
> +#include "value-range.h"
> +#include "ipa-prop.h"
> +#include "ipa-fnsummary.h"
>
> /* Class (from which there is one global instance) that holds modref
> summaries
> for all analyzed functions. */
> @@ -317,36 +415,12 @@ record_access_lto (modref_records_lto *tt, ao_ref *ref)
> static bool
> record_access_p (tree expr)
> {
> - /* Non-escaping memory is fine */
> - tree t = get_base_address (expr);
> - if (t && (INDIRECT_REF_P (t)
> - || TREE_CODE (t) == MEM_REF
> - || TREE_CODE (t) == TARGET_MEM_REF)
> - && TREE_CODE (TREE_OPERAND (t, 0)) == SSA_NAME
> - && !ptr_deref_may_alias_global_p (TREE_OPERAND (t, 0)))
> + if (refs_local_or_readonly_memory_p (expr))
> {
> if (dump_file)
> - fprintf (dump_file, " - Non-escaping memory, ignoring.\n");
> + fprintf (dump_file, " - Read-only or local, ignoring.\n");
> return false;
> }
> -
> - /* Automatic variables are fine. */
> - if (DECL_P (t)
> - && auto_var_in_fn_p (t, current_function_decl))
> - {
> - if (dump_file)
> - fprintf (dump_file, " - Automatic variable, ignoring.\n");
> - return false;
> - }
> -
> - /* Read-only variables are fine. */
> - if (DECL_P (t) && TREE_READONLY (t))
> - {
> - if (dump_file)
> - fprintf (dump_file, " - Read-only variable, ignoring.\n");
> - return false;
> - }
> -
> return true;
> }
>
> diff --git a/gcc/ipa-pure-const.c b/gcc/ipa-pure-const.c
> index 564c6629c92..bdbccd010dc 100644
> --- a/gcc/ipa-pure-const.c
> +++ b/gcc/ipa-pure-const.c
> @@ -381,13 +381,11 @@ check_op (funct_state local, tree t, bool
> checking_write)
> fprintf (dump_file, " Volatile indirect ref is not const/pure\n");
> return;
> }
> - else if (t
> - && (INDIRECT_REF_P (t) || TREE_CODE (t) == MEM_REF)
> - && TREE_CODE (TREE_OPERAND (t, 0)) == SSA_NAME
> - && !ptr_deref_may_alias_global_p (TREE_OPERAND (t, 0)))
> + else if (refs_local_or_readonly_memory_p (t))
> {
> if (dump_file)
> - fprintf (dump_file, " Indirect ref to local memory is OK\n");
> + fprintf (dump_file, " Indirect ref to local or readonly "
> + "memory is OK\n");
> return;
> }
> else if (checking_write)
> diff --git a/gcc/testsuite/gcc.dg/tree-ssa/local-pure-const.c
> b/gcc/testsuite/gcc.dg/tree-ssa/local-pure-const.c
> index 3c358e01393..6746758ca88 100644
> --- a/gcc/testsuite/gcc.dg/tree-ssa/local-pure-const.c
> +++ b/gcc/testsuite/gcc.dg/tree-ssa/local-pure-const.c
> @@ -12,5 +12,5 @@ t(int a, int b, int c)
> p = &c;
> return *p;
> }
> -/* { dg-final { scan-tree-dump-times "local memory is OK" 1
> "local-pure-const1"} } */
> +/* { dg-final { scan-tree-dump-times "local or readonly memory is OK" 1
> "local-pure-const1"} } */
> /* { dg-final { scan-tree-dump-times "found to be const" 1
> "local-pure-const1"} } */