aboutsummaryrefslogtreecommitdiff
path: root/gcc/ipa-modref.c
diff options
context:
space:
mode:
authorJan Hubicka <jh@suse.cz>2021-11-13 01:51:25 +0100
committerJan Hubicka <jh@suse.cz>2021-11-13 01:51:25 +0100
commit2f3d43a35155685b1795b4392e20e1c14a33c38f (patch)
tree62c93bdcc3a21cb6f0f23479da2bc320a6c33e57 /gcc/ipa-modref.c
parent60f761c7e54f96a287c73a71d0b09ee2b2f8426d (diff)
downloadgcc-2f3d43a35155685b1795b4392e20e1c14a33c38f.zip
gcc-2f3d43a35155685b1795b4392e20e1c14a33c38f.tar.gz
gcc-2f3d43a35155685b1795b4392e20e1c14a33c38f.tar.bz2
Fix wrong code with modref and some builtins.
ipa-modref gets confused by EAF flags of memcpy becuase parameter 1 is escaping but used only directly. In modref we do not track values saved to memory and thus we clear all other flags on each store. This needs to also happen when called function escapes parameter. gcc/ChangeLog: PR tree-optimization/103182 * ipa-modref.c (callee_to_caller_flags): Fix merging of flags. (modref_eaf_analysis::analyze_ssa_name): Fix merging of flags.
Diffstat (limited to 'gcc/ipa-modref.c')
-rw-r--r--gcc/ipa-modref.c30
1 files changed, 14 insertions, 16 deletions
diff --git a/gcc/ipa-modref.c b/gcc/ipa-modref.c
index e999c2c..90985cc 100644
--- a/gcc/ipa-modref.c
+++ b/gcc/ipa-modref.c
@@ -1888,19 +1888,18 @@ callee_to_caller_flags (int call_flags, bool ignore_stores,
that is not the same as caller returning it. */
call_flags |= EAF_NOT_RETURNED_DIRECTLY
| EAF_NOT_RETURNED_INDIRECTLY;
- /* TODO: We miss return value propagation.
- Be conservative and if value escapes to memory
- also mark it as escaping. */
if (!ignore_stores && !(call_flags & EAF_UNUSED))
{
+ /* If value escapes we are no longer able to track what happens
+ with it because we can read it from the escaped location
+ anytime. */
if (!(call_flags & EAF_NO_DIRECT_ESCAPE))
- lattice.merge (~(EAF_NOT_RETURNED_DIRECTLY
- | EAF_NOT_RETURNED_INDIRECTLY
- | EAF_NO_DIRECT_READ
- | EAF_UNUSED));
- if (!(call_flags & EAF_NO_INDIRECT_ESCAPE))
+ lattice.merge (0);
+ else if (!(call_flags & EAF_NO_INDIRECT_ESCAPE))
lattice.merge (~(EAF_NOT_RETURNED_INDIRECTLY
| EAF_NO_DIRECT_READ
+ | EAF_NO_INDIRECT_READ
+ | EAF_NO_INDIRECT_CLOBBER
| EAF_UNUSED));
}
else
@@ -2036,18 +2035,17 @@ modref_eaf_analysis::analyze_ssa_name (tree name)
not_returned and escape has same meaning.
However passing arg to return slot is different. If
the callee's return slot is returned it means that
- arg is written to itself which is an escape. */
+ arg is written to itself which is an escape.
+ Since we do not track the memory it is written to we
+ need to give up on analysisng it. */
if (!isretslot)
{
if (!(call_flags & (EAF_NOT_RETURNED_DIRECTLY
| EAF_UNUSED)))
- m_lattice[index].merge (~(EAF_NO_DIRECT_ESCAPE
- | EAF_UNUSED));
- if (!(call_flags & (EAF_NOT_RETURNED_INDIRECTLY
- | EAF_UNUSED)))
- m_lattice[index].merge (~(EAF_NO_INDIRECT_ESCAPE
- | EAF_NO_DIRECT_READ
- | EAF_UNUSED));
+ m_lattice[index].merge (0);
+ else gcc_checking_assert
+ (call_flags & (EAF_NOT_RETURNED_INDIRECTLY
+ | EAF_UNUSED));
call_flags = callee_to_caller_flags
(call_flags, false,
m_lattice[index]);