re PR ipa/88561 (PGO devirtualization miscompilation of firefox)
authorJan Hubicka <hubicka@ucw.cz>
Fri, 21 Dec 2018 19:13:06 +0000 (20:13 +0100)
committerJan Hubicka <hubicka@gcc.gnu.org>
Fri, 21 Dec 2018 19:13:06 +0000 (19:13 +0000)
PR ipa/88561
* ipa-polymorphic-call.c
(ipa_polymorphic_call_context::ipa_polymorphic_call_context): Handle
arguments of thunks correctly.
(ipa_polymorphic_call_context::get_dynamic_context): Be ready for
NULL instance pinter.
* lto-cgraph.c (lto_output_node): Always stream thunk info.

From-SVN: r267338

gcc/ChangeLog
gcc/ipa-polymorphic-call.c
gcc/lto-cgraph.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/tree-prof/devirt.C [new file with mode: 0644]

index 622e024..b8db488 100644 (file)
@@ -1,3 +1,13 @@
+2018-12-15  Jan Hubicka  <hubicka@ucw.cz>
+
+       PR ipa/88561
+       * ipa-polymorphic-call.c
+       (ipa_polymorphic_call_context::ipa_polymorphic_call_context): Handle
+       arguments of thunks correctly.
+       (ipa_polymorphic_call_context::get_dynamic_context): Be ready for
+       NULL instance pinter.
+       * lto-cgraph.c (lto_output_node): Always stream thunk info.
+
 2018-12-21  Andreas Krebbel  <krebbel@linux.ibm.com>
 
        * config/s390/vector.md ("floatv2div2df2", "floatunsv2div2df2")
index 13aca94..6e54733 100644 (file)
@@ -995,9 +995,22 @@ ipa_polymorphic_call_context::ipa_polymorphic_call_context (tree fndecl,
        {
          outer_type
             = TYPE_MAIN_VARIANT (TREE_TYPE (TREE_TYPE (base_pointer)));
+         cgraph_node *node = cgraph_node::get (current_function_decl);
          gcc_assert (TREE_CODE (outer_type) == RECORD_TYPE
                      || TREE_CODE (outer_type) == UNION_TYPE);
 
+         /* Handle the case we inlined into a thunk.  In this case
+            thunk has THIS pointer of type bar, but it really receives
+            address to its base type foo which sits in bar at 
+            0-thunk.fixed_offset.  It starts with code that adds
+            think.fixed_offset to the pointer to compensate for this.
+
+            Because we walked all the way to the begining of thunk, we now
+            see pointer &bar-thunk.fixed_offset and need to compensate
+            for it.  */
+         if (node->thunk.fixed_offset)
+           offset -= node->thunk.fixed_offset * BITS_PER_UNIT;
+
          /* Dynamic casting has possibly upcasted the type
             in the hiearchy.  In this case outer type is less
             informative than inner type and we should forget
@@ -1005,7 +1018,11 @@ ipa_polymorphic_call_context::ipa_polymorphic_call_context (tree fndecl,
          if ((otr_type
               && !contains_type_p (outer_type, offset,
                                    otr_type))
-             || !contains_polymorphic_type_p (outer_type))
+             || !contains_polymorphic_type_p (outer_type)
+             /* If we compile thunk with virtual offset, the THIS pointer
+                is adjusted by unknown value.  We can't thus use outer info
+                at all.  */
+             || node->thunk.virtual_offset_p)
            {
              outer_type = NULL;
              if (instance)
@@ -1030,7 +1047,15 @@ ipa_polymorphic_call_context::ipa_polymorphic_call_context (tree fndecl,
              maybe_in_construction = false;
            }
          if (instance)
-           *instance = base_pointer;
+           {
+             /* If method is expanded thunk, we need to apply thunk offset
+                to instance pointer.  */
+             if (node->thunk.virtual_offset_p
+                 || node->thunk.fixed_offset)
+               *instance = NULL;
+             else
+               *instance = base_pointer;
+           }
          return;
        }
       /* Non-PODs passed by value are really passed by invisible
@@ -1547,6 +1572,9 @@ ipa_polymorphic_call_context::get_dynamic_type (tree instance,
   HOST_WIDE_INT instance_offset = offset;
   tree instance_outer_type = outer_type;
 
+  if (!instance)
+    return false;
+
   if (otr_type)
     otr_type = TYPE_MAIN_VARIANT (otr_type);
 
index 536e73c..45138fd 100644 (file)
@@ -547,7 +547,11 @@ lto_output_node (struct lto_simple_output_block *ob, struct cgraph_node *node,
   streamer_write_bitpack (&bp);
   streamer_write_data_stream (ob->main_stream, section, strlen (section) + 1);
 
-  if (node->thunk.thunk_p)
+  /* Stream thunk info always because we use it in
+     ipa_polymorphic_call_context::ipa_polymorphic_call_context
+     to properly interpret THIS pointers for thunks that has been converted
+     to Gimple.  */
+  if (node->definition)
     {
       streamer_write_uhwi_stream
         (ob->main_stream,
@@ -1295,7 +1299,7 @@ input_node (struct lto_file_decl_data *file_data,
   if (section)
     node->set_section_for_node (section);
 
-  if (node->thunk.thunk_p)
+  if (node->definition)
     {
       int type = streamer_read_uhwi (ib);
       HOST_WIDE_INT fixed_offset = streamer_read_uhwi (ib);
index 687d700..0018d8b 100644 (file)
@@ -1,3 +1,8 @@
+2018-12-15  Jan Hubicka  <hubicka@ucw.cz>
+
+       PR ipa/88561
+       * g++.dg/tree-prof/devirt.C: New testcase.
+
 2018-12-21  Paul Thomas  <pault@gcc.gnu.org>
 
        PR fortran/87881
diff --git a/gcc/testsuite/g++.dg/tree-prof/devirt.C b/gcc/testsuite/g++.dg/tree-prof/devirt.C
new file mode 100644 (file)
index 0000000..05c9a26
--- /dev/null
@@ -0,0 +1,123 @@
+/* { dg-options "-O3 -fdump-tree-dom3" } */
+struct nsISupports
+{
+  virtual int QueryInterface (const int &aIID, void **aInstancePtr) = 0;
+  virtual __attribute__((noinline, noclone)) unsigned AddRef (void) = 0;
+  virtual unsigned Release (void) = 0;
+};
+
+struct nsIObserver : public nsISupports
+{
+  virtual int Observe (nsISupports * aSubject, const char *aTopic, const unsigned short *aData) = 0;
+};
+
+struct nsISupportsWeakReference : public nsISupports
+{
+  virtual int GetWeakReference (void **_retval) = 0;
+};
+
+struct nsSupportsWeakReference : public nsISupportsWeakReference
+{
+  nsSupportsWeakReference () : mProxy (0) {}
+  virtual int GetWeakReference (void **_retval) override { return 0; }
+  ~nsSupportsWeakReference () {}
+  void NoticeProxyDestruction () { mProxy = nullptr; }
+  void *mProxy;
+  void ClearWeakReferences ();
+  bool HasWeakReferences () const { return !!mProxy; }
+};
+
+struct mozIPersonalDictionary : public nsISupports
+{
+  virtual int Load (void) = 0;
+  virtual int Save (void) = 0;
+  virtual int GetWordList (void **aWordList) = 0;
+  virtual int Check (const int &word, bool * _retval) = 0;
+  virtual int AddWord (const int &word) = 0;
+  virtual int RemoveWord (const int &word) = 0;
+  virtual int IgnoreWord (const int &word) = 0;
+  virtual int EndSession (void) = 0;
+};
+
+struct mozPersonalDictionary final
+  : public mozIPersonalDictionary, public nsIObserver, public nsSupportsWeakReference
+{
+  virtual int QueryInterface (const int &aIID, void **aInstancePtr) override;
+  virtual __attribute__((noinline, noclone)) unsigned AddRef (void) override;
+  virtual unsigned Release (void) override;
+  unsigned long mRefCnt;
+  virtual int Load (void) override { return 0; }
+  virtual int Save (void) override { return 0; }
+  virtual int GetWordList (void **aWordList) override { return 0; }
+  virtual int Check (const int &word, bool * _retval) override { return 0; }
+  virtual int AddWord (const int &word) override { return 0; }
+  virtual int RemoveWord (const int &word) override { return 0; }
+  virtual int IgnoreWord (const int &word) override { return 0; }
+  virtual int EndSession (void) override { return 0; }
+  virtual int Observe (nsISupports * aSubject, const char *aTopic, const unsigned short *aData) override { return 0; }
+  mozPersonalDictionary () : mRefCnt(0) {}
+  int Init () { return 0; }
+  virtual ~mozPersonalDictionary () {}
+  bool mIsLoaded;
+  bool mSavePending;
+  void *mFile;
+  char mMonitor[96];
+  char mMonitorSave[96];
+  char mDictionaryTable[32];
+  char mIgnoreTable[32];
+};
+
+unsigned
+mozPersonalDictionary::AddRef (void)
+{
+  unsigned count = ++mRefCnt;
+  return count;
+}
+
+unsigned
+mozPersonalDictionary::Release (void)
+{
+  unsigned count = --mRefCnt;
+  if (count == 0)
+    {
+      mRefCnt = 1;
+      delete (this);
+      return 0;
+    }
+  return count;
+}
+
+int
+mozPersonalDictionary::QueryInterface (const int &aIID, void **aInstancePtr)
+{
+  nsISupports *foundInterface;
+  if (aIID == 122)
+    foundInterface = static_cast <mozIPersonalDictionary *>(this);
+  else
+    foundInterface = static_cast <nsISupportsWeakReference *>(this);
+  int status;
+  foundInterface->AddRef ();
+  *aInstancePtr = foundInterface;
+  return status;
+}
+
+__attribute__((noipa)) int
+foo (nsISupports *p, const int &i)
+{
+  void *q;
+  return p->QueryInterface (i, &q);
+}
+
+int
+main ()
+{
+  mozPersonalDictionary m;
+  int j = 123;
+  for (int i = 0; i < 100000; i++)
+    foo (static_cast <nsISupportsWeakReference *>(&m), j);
+  if (m.mRefCnt != 100000)
+    __builtin_abort ();
+}
+
+/* { dg-final-use-not-autofdo { scan-ipa-dump-times 3 "folding virtual function call to virtual unsigned int mozPersonalDictionary::_ZThn16" "dom3" } } */
+/* { dg-final-use-not-autofdo { scan-ipa-dump-times 3 "folding virtual function call to virtual unsigned int mozPersonalDictionary::AddRef" "dom3" } } */