aboutsummaryrefslogtreecommitdiff
path: root/gcc/ipa-modref-tree.c
diff options
context:
space:
mode:
authorJan Hubicka <jh@suse.cz>2020-09-20 07:25:16 +0200
committerJan Hubicka <jh@suse.cz>2020-09-20 07:27:48 +0200
commitd119f34c952f8718fdbabc63e2f369a16e92fa07 (patch)
tree3f1a460afbcfeafd21f63e7d843251c5b07ea9e6 /gcc/ipa-modref-tree.c
parent2fe5b7d1f66457c637b8bd2543a60a5faff34c40 (diff)
downloadgcc-d119f34c952f8718fdbabc63e2f369a16e92fa07.zip
gcc-d119f34c952f8718fdbabc63e2f369a16e92fa07.tar.gz
gcc-d119f34c952f8718fdbabc63e2f369a16e92fa07.tar.bz2
New modref/ipa_modref optimization passes
2020-09-19 David Cepelik <d@dcepelik.cz> Jan Hubicka <hubicka@ucw.cz> * Makefile.in: Add ipa-modref.c and ipa-modref-tree.c. * alias.c: (reference_alias_ptr_type_1): Export. * alias.h (reference_alias_ptr_type_1): Declare. * common.opt (fipa-modref): New. * gengtype.c (open_base_files): Add ipa-modref-tree.h and ipa-modref.h * ipa-modref-tree.c: New file. * ipa-modref-tree.h: New file. * ipa-modref.c: New file. * ipa-modref.h: New file. * lto-section-in.c (lto_section_name): Add ipa_modref. * lto-streamer.h (enum lto_section_type): Add LTO_section_ipa_modref. * opts.c (default_options_table): Enable ipa-modref at -O1+. * params.opt (-param=modref-max-bases, -param=modref-max-refs, -param=modref-max-tests): New params. * passes.def: Schedule pass_modref and pass_ipa_modref. * timevar.def (TV_IPA_MODREF): New timevar. (TV_TREE_MODREF): New timevar. * tree-pass.h (make_pass_modref): Declare. (make_pass_ipa_modref): Declare. * tree-ssa-alias.c (dump_alias_stats): Include ipa-modref-tree.h and ipa-modref.h (alias_stats): Add modref_use_may_alias, modref_use_no_alias, modref_clobber_may_alias, modref_clobber_no_alias, modref_tests. (dump_alias_stats): Dump new stats. (nonoverlapping_array_refs_p): Fix formating. (modref_may_conflict): New function. (ref_maybe_used_by_call_p_1): Use it. (call_may_clobber_ref_p_1): Use it. (call_may_clobber_ref_p): Update. (stmt_may_clobber_ref_p_1): Update. * tree-ssa-alias.h (call_may_clobber_ref_p_1): Update.
Diffstat (limited to 'gcc/ipa-modref-tree.c')
-rw-r--r--gcc/ipa-modref-tree.c236
1 files changed, 236 insertions, 0 deletions
diff --git a/gcc/ipa-modref-tree.c b/gcc/ipa-modref-tree.c
new file mode 100644
index 0000000..e37dee6
--- /dev/null
+++ b/gcc/ipa-modref-tree.c
@@ -0,0 +1,236 @@
+/* Data structure for the modref pass.
+ Copyright (C) 2020 Free Software Foundation, Inc.
+ Contributed by David Cepelik and Jan Hubicka
+
+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 3, 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 COPYING3. If not see
+<http://www.gnu.org/licenses/>. */
+
+#include "config.h"
+#include "system.h"
+#include "coretypes.h"
+#include "backend.h"
+#include "tree.h"
+#include "ipa-modref-tree.h"
+#include "selftest.h"
+
+#if CHECKING_P
+
+
+static void
+test_insert_search_collapse ()
+{
+ modref_base_node<alias_set_type> *base_node;
+ modref_ref_node<alias_set_type> *ref_node;
+
+ modref_tree<alias_set_type> *t = new modref_tree<alias_set_type>(1, 2);
+ ASSERT_FALSE (t->every_base);
+
+ /* Insert into an empty tree. */
+ t->insert (1, 2);
+ ASSERT_NE (t->bases, NULL);
+ ASSERT_EQ (t->bases->length (), 1);
+ ASSERT_FALSE (t->every_base);
+ ASSERT_EQ (t->search (2), NULL);
+
+ base_node = t->search (1);
+ ASSERT_NE (base_node, NULL);
+ ASSERT_EQ (base_node->base, 1);
+ ASSERT_NE (base_node->refs, NULL);
+ ASSERT_EQ (base_node->refs->length (), 1);
+ ASSERT_EQ (base_node->search (1), NULL);
+
+ ref_node = base_node->search (2);
+ ASSERT_NE (ref_node, NULL);
+ ASSERT_EQ (ref_node->ref, 2);
+
+ /* Insert when base exists but ref does not. */
+ t->insert (1, 3);
+ ASSERT_NE (t->bases, NULL);
+ ASSERT_EQ (t->bases->length (), 1);
+ ASSERT_EQ (t->search (1), base_node);
+ ASSERT_EQ (t->search (2), NULL);
+ ASSERT_NE (base_node->refs, NULL);
+ ASSERT_EQ (base_node->refs->length (), 2);
+
+ ref_node = base_node->search (3);
+ ASSERT_NE (ref_node, NULL);
+
+ /* Insert when base and ref exist, but access is not dominated by nor
+ dominates other accesses. */
+ t->insert (1, 2);
+ ASSERT_EQ (t->bases->length (), 1);
+ ASSERT_EQ (t->search (1), base_node);
+
+ ref_node = base_node->search (2);
+ ASSERT_NE (ref_node, NULL);
+
+ /* Insert when base and ref exist and access is dominated. */
+ t->insert (1, 2);
+ ASSERT_EQ (t->search (1), base_node);
+ ASSERT_EQ (base_node->search (2), ref_node);
+
+ /* Insert ref to trigger ref list collapse for base 1. */
+ t->insert (1, 4);
+ ASSERT_EQ (t->search (1), base_node);
+ ASSERT_EQ (base_node->refs, NULL);
+ ASSERT_EQ (base_node->search (2), NULL);
+ ASSERT_EQ (base_node->search (3), NULL);
+ ASSERT_TRUE (base_node->every_ref);
+
+ /* Further inserts to collapsed ref list are ignored. */
+ t->insert (1, 5);
+ ASSERT_EQ (t->search (1), base_node);
+ ASSERT_EQ (base_node->refs, NULL);
+ ASSERT_EQ (base_node->search (2), NULL);
+ ASSERT_EQ (base_node->search (3), NULL);
+ ASSERT_TRUE (base_node->every_ref);
+
+ /* Insert base to trigger base list collapse. */
+ t->insert (5, 6);
+ ASSERT_TRUE (t->every_base);
+ ASSERT_EQ (t->bases, NULL);
+ ASSERT_EQ (t->search (1), NULL);
+
+ /* Further inserts to collapsed base list are ignored. */
+ t->insert (7, 8);
+ ASSERT_TRUE (t->every_base);
+ ASSERT_EQ (t->bases, NULL);
+ ASSERT_EQ (t->search (1), NULL);
+}
+
+static void
+test_merge ()
+{
+ modref_tree<alias_set_type> *t1, *t2;
+ modref_base_node<alias_set_type> *base_node;
+
+ t1 = new modref_tree<alias_set_type>(3, 4);
+ t1->insert (1, 1);
+ t1->insert (1, 2);
+ t1->insert (1, 3);
+ t1->insert (2, 1);
+ t1->insert (3, 1);
+
+ t2 = new modref_tree<alias_set_type>(10, 10);
+ t2->insert (1, 2);
+ t2->insert (1, 3);
+ t2->insert (1, 4);
+ t2->insert (3, 2);
+ t2->insert (3, 3);
+ t2->insert (3, 4);
+ t2->insert (3, 5);
+
+ t1->merge (t2);
+
+ ASSERT_FALSE (t1->every_base);
+ ASSERT_NE (t1->bases, NULL);
+ ASSERT_EQ (t1->bases->length (), 3);
+
+ base_node = t1->search (1);
+ ASSERT_NE (base_node->refs, NULL);
+ ASSERT_FALSE (base_node->every_ref);
+ ASSERT_EQ (base_node->refs->length (), 4);
+
+ base_node = t1->search (2);
+ ASSERT_NE (base_node->refs, NULL);
+ ASSERT_FALSE (base_node->every_ref);
+ ASSERT_EQ (base_node->refs->length (), 1);
+
+ base_node = t1->search (3);
+ ASSERT_EQ (base_node->refs, NULL);
+ ASSERT_TRUE (base_node->every_ref);
+}
+
+
+void
+modref_tree_c_tests ()
+{
+ test_insert_search_collapse ();
+ test_merge ();
+}
+
+#endif
+
+void
+gt_ggc_mx (modref_tree < int >*const &tt)
+{
+ if (tt->bases)
+ {
+ ggc_test_and_set_mark (tt->bases);
+ gt_ggc_mx (tt->bases);
+ }
+}
+
+void
+gt_ggc_mx (modref_tree < tree_node * >*const &tt)
+{
+ if (tt->bases)
+ {
+ ggc_test_and_set_mark (tt->bases);
+ gt_ggc_mx (tt->bases);
+ }
+}
+
+void gt_pch_nx (modref_tree<int>* const&) {}
+void gt_pch_nx (modref_tree<tree_node*>* const&) {}
+void gt_pch_nx (modref_tree<int>* const&, gt_pointer_operator, void *) {}
+void gt_pch_nx (modref_tree<tree_node*>* const&, gt_pointer_operator, void *) {}
+
+void gt_ggc_mx (modref_base_node<int>* &b)
+{
+ ggc_test_and_set_mark (b);
+ if (b->refs)
+ {
+ ggc_test_and_set_mark (b->refs);
+ gt_ggc_mx (b->refs);
+ }
+}
+
+void gt_ggc_mx (modref_base_node<tree_node*>* &b)
+{
+ ggc_test_and_set_mark (b);
+ if (b->refs)
+ {
+ ggc_test_and_set_mark (b->refs);
+ gt_ggc_mx (b->refs);
+ }
+ if (b->base)
+ gt_ggc_mx (b->base);
+}
+
+void gt_pch_nx (modref_base_node<int>*) {}
+void gt_pch_nx (modref_base_node<tree_node*>*) {}
+void gt_pch_nx (modref_base_node<int>*, gt_pointer_operator, void *) {}
+void gt_pch_nx (modref_base_node<tree_node*>*, gt_pointer_operator, void *) {}
+
+void gt_ggc_mx (modref_ref_node<int>* &r)
+{
+ ggc_test_and_set_mark (r);
+}
+
+void gt_ggc_mx (modref_ref_node<tree_node*>* &r)
+{
+ ggc_test_and_set_mark (r);
+ if (r->ref)
+ gt_ggc_mx (r->ref);
+}
+
+void gt_pch_nx (modref_ref_node<int>* ) {}
+void gt_pch_nx (modref_ref_node<tree_node*>*) {}
+void gt_pch_nx (modref_ref_node<int>*, gt_pointer_operator, void *) {}
+void gt_pch_nx (modref_ref_node<tree_node*>*, gt_pointer_operator, void *) {}
+
+