Fix hiddenapi::MemberSignature for proxies

Proxy classes are classes generated at runtime which implement
a given interface. Because they do not inherit the associated
dex file form the interface(s), names/signatures of methods cannot
be requested directly, but rather through the original interface
method. Calling getName() on a proxy mirror::Class also triggers
a DCHECK.

This patch will refer to the interface method when printing the
signature instead of the proxy method.

This fixes the warning printed for the proxy method, printing
even the class name of the interface instead of the name of the
proxy class. This is meant to provide useful info to the devloper.

Proxies do not define fields except for the synthetic 'interfaces'
and 'throws' fields. Their signatures remain unchanged.

This patch also continues to check the access flags of the proxy
method for performance reasons. De-proxying the method would
introduce two new memory accesses into the fast path. That means
deduplication of warnings remains independent between the original
and proxy methods.

Bug: 78327881
Test: make test-art-host-gtest-hidden_api_test
Merged-In: I8f334e5e2b62ca38691c94524edaf198eb73574b
Change-Id: I8f334e5e2b62ca38691c94524edaf198eb73574b
(cherry picked from commit 73a64f6a2a475c2fe018c7ab1151e3f44d316533)
diff --git a/runtime/hidden_api.cc b/runtime/hidden_api.cc
index e8918e7..9445ae0 100644
--- a/runtime/hidden_api.cc
+++ b/runtime/hidden_api.cc
@@ -83,6 +83,10 @@
 }
 
 MemberSignature::MemberSignature(ArtMethod* method) {
+  // If this is a proxy method, print the signature of the interface method.
+  method = method->GetInterfaceMethodIfProxy(
+      Runtime::Current()->GetClassLinker()->GetImagePointerSize());
+
   class_name_ = method->GetDeclaringClass()->GetDescriptor(&tmp_);
   member_name_ = method->GetName();
   type_signature_ = method->GetSignature().ToString();
diff --git a/runtime/hidden_api_test.cc b/runtime/hidden_api_test.cc
index 1985c6b..68d4eec 100644
--- a/runtime/hidden_api_test.cc
+++ b/runtime/hidden_api_test.cc
@@ -18,6 +18,7 @@
 
 #include "common_runtime_test.h"
 #include "jni_internal.h"
+#include "proxy_test.h"
 
 namespace art {
 
@@ -31,7 +32,7 @@
     CommonRuntimeTest::SetUp();
     self_ = Thread::Current();
     self_->TransitionFromSuspendedToRunnable();
-    LoadDex("HiddenApiSignatures");
+    jclass_loader_ = LoadDex("HiddenApiSignatures");
     bool started = runtime_->Start();
     CHECK(started);
 
@@ -69,6 +70,7 @@
 
  protected:
   Thread* self_;
+  jobject jclass_loader_;
   ArtField* class1_field1_;
   ArtField* class1_field12_;
   ArtMethod* class1_init_;
@@ -311,4 +313,56 @@
   ASSERT_FALSE(MemberSignature(class1_field1_).DoesPrefixMatch(prefix));
 }
 
+TEST_F(HiddenApiTest, CheckMemberSignatureForProxyClass) {
+  ScopedObjectAccess soa(self_);
+  StackHandleScope<4> hs(soa.Self());
+  Handle<mirror::ClassLoader> class_loader(
+      hs.NewHandle(soa.Decode<mirror::ClassLoader>(jclass_loader_)));
+
+  // Find interface we will create a proxy for.
+  Handle<mirror::Class> h_iface(hs.NewHandle(
+      class_linker_->FindClass(soa.Self(), "Lmypackage/packagea/Interface;", class_loader)));
+  ASSERT_TRUE(h_iface != nullptr);
+
+  // Create the proxy class.
+  std::vector<mirror::Class*> interfaces;
+  interfaces.push_back(h_iface.Get());
+  Handle<mirror::Class> proxyClass = hs.NewHandle(proxy_test::GenerateProxyClass(
+      soa, jclass_loader_, runtime_->GetClassLinker(), "$Proxy1234", interfaces));
+  ASSERT_TRUE(proxyClass != nullptr);
+  ASSERT_TRUE(proxyClass->IsProxyClass());
+  ASSERT_TRUE(proxyClass->IsInitialized());
+
+  // Find the "method" virtual method.
+  ArtMethod* method = nullptr;
+  for (auto& m : proxyClass->GetDeclaredVirtualMethods(kRuntimePointerSize)) {
+    if (strcmp("method", m.GetInterfaceMethodIfProxy(kRuntimePointerSize)->GetName()) == 0) {
+      method = &m;
+      break;
+    }
+  }
+  ASSERT_TRUE(method != nullptr);
+
+  // Find the "interfaces" static field. This is generated for all proxies.
+  ArtField* field = nullptr;
+  for (size_t i = 0; i < proxyClass->NumStaticFields(); ++i) {
+    ArtField* f = proxyClass->GetStaticField(i);
+    if (strcmp("interfaces", f->GetName()) == 0) {
+      field = f;
+      break;
+    }
+  }
+  ASSERT_TRUE(field != nullptr);
+
+  // Test the signature. We expect the signature from the interface class.
+  std::ostringstream ss_method;
+  MemberSignature(method).Dump(ss_method);
+  ASSERT_EQ("Lmypackage/packagea/Interface;->method()V", ss_method.str());
+
+  // Test the signature. We expect the signature of the proxy class.
+  std::ostringstream ss_field;
+  MemberSignature(field).Dump(ss_field);
+  ASSERT_EQ("L$Proxy1234;->interfaces:[Ljava/lang/Class;", ss_field.str());
+}
+
 }  // namespace art
diff --git a/runtime/proxy_test.cc b/runtime/proxy_test.cc
index 3ad3a4b..4e0bf89 100644
--- a/runtime/proxy_test.cc
+++ b/runtime/proxy_test.cc
@@ -18,97 +18,16 @@
 #include <vector>
 
 #include "art_field-inl.h"
-#include "art_method-inl.h"
 #include "base/enums.h"
-#include "class_linker-inl.h"
 #include "common_compiler_test.h"
-#include "mirror/class-inl.h"
 #include "mirror/field-inl.h"
-#include "mirror/method.h"
+#include "proxy_test.h"
 #include "scoped_thread_state_change-inl.h"
 
 namespace art {
+namespace proxy_test {
 
-class ProxyTest : public CommonCompilerTest {
- public:
-  // Generate a proxy class with the given name and interfaces. This is a simplification from what
-  // libcore does to fit to our test needs. We do not check for duplicated interfaces or methods and
-  // we do not declare exceptions.
-  mirror::Class* GenerateProxyClass(ScopedObjectAccess& soa, jobject jclass_loader,
-                                    const char* className,
-                                    const std::vector<mirror::Class*>& interfaces)
-      REQUIRES_SHARED(Locks::mutator_lock_) {
-    mirror::Class* javaLangObject = class_linker_->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
-    CHECK(javaLangObject != nullptr);
-
-    jclass javaLangClass = soa.AddLocalReference<jclass>(mirror::Class::GetJavaLangClass());
-
-    // Builds the interfaces array.
-    jobjectArray proxyClassInterfaces = soa.Env()->NewObjectArray(interfaces.size(), javaLangClass,
-                                                                  nullptr);
-    soa.Self()->AssertNoPendingException();
-    for (size_t i = 0; i < interfaces.size(); ++i) {
-      soa.Env()->SetObjectArrayElement(proxyClassInterfaces, i,
-                                       soa.AddLocalReference<jclass>(interfaces[i]));
-    }
-
-    // Builds the method array.
-    jsize methods_count = 3;  // Object.equals, Object.hashCode and Object.toString.
-    for (mirror::Class* interface : interfaces) {
-      methods_count += interface->NumVirtualMethods();
-    }
-    jobjectArray proxyClassMethods = soa.Env()->NewObjectArray(
-        methods_count, soa.AddLocalReference<jclass>(mirror::Method::StaticClass()), nullptr);
-    soa.Self()->AssertNoPendingException();
-
-    jsize array_index = 0;
-    // Fill the method array
-    DCHECK_EQ(Runtime::Current()->GetClassLinker()->GetImagePointerSize(), kRuntimePointerSize);
-    ArtMethod* method = javaLangObject->FindClassMethod(
-        "equals", "(Ljava/lang/Object;)Z", kRuntimePointerSize);
-    CHECK(method != nullptr);
-    CHECK(!method->IsDirect());
-    CHECK(method->GetDeclaringClass() == javaLangObject);
-    DCHECK(!Runtime::Current()->IsActiveTransaction());
-    soa.Env()->SetObjectArrayElement(
-        proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
-            mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
-    method = javaLangObject->FindClassMethod("hashCode", "()I", kRuntimePointerSize);
-    CHECK(method != nullptr);
-    CHECK(!method->IsDirect());
-    CHECK(method->GetDeclaringClass() == javaLangObject);
-    soa.Env()->SetObjectArrayElement(
-        proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
-            mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
-    method = javaLangObject->FindClassMethod(
-        "toString", "()Ljava/lang/String;", kRuntimePointerSize);
-    CHECK(method != nullptr);
-    CHECK(!method->IsDirect());
-    CHECK(method->GetDeclaringClass() == javaLangObject);
-    soa.Env()->SetObjectArrayElement(
-        proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
-            mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
-    // Now adds all interfaces virtual methods.
-    for (mirror::Class* interface : interfaces) {
-      for (auto& m : interface->GetDeclaredVirtualMethods(kRuntimePointerSize)) {
-        soa.Env()->SetObjectArrayElement(
-            proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
-                mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), &m)));
-      }
-    }
-    CHECK_EQ(array_index, methods_count);
-
-    // Builds an empty exception array.
-    jobjectArray proxyClassThrows = soa.Env()->NewObjectArray(0, javaLangClass, nullptr);
-    soa.Self()->AssertNoPendingException();
-
-    mirror::Class* proxyClass = class_linker_->CreateProxyClass(
-        soa, soa.Env()->NewStringUTF(className), proxyClassInterfaces, jclass_loader,
-        proxyClassMethods, proxyClassThrows);
-    soa.Self()->AssertNoPendingException();
-    return proxyClass;
-  }
-};
+class ProxyTest : public CommonRuntimeTest {};
 
 // Creates a proxy class and check ClassHelper works correctly.
 TEST_F(ProxyTest, ProxyClassHelper) {
@@ -129,7 +48,7 @@
   interfaces.push_back(I.Get());
   interfaces.push_back(J.Get());
   Handle<mirror::Class> proxy_class(hs.NewHandle(
-      GenerateProxyClass(soa, jclass_loader, "$Proxy1234", interfaces)));
+      GenerateProxyClass(soa, jclass_loader, class_linker_, "$Proxy1234", interfaces)));
   interfaces.clear();  // Don't least possibly stale objects in the array as good practice.
   ASSERT_TRUE(proxy_class != nullptr);
   ASSERT_TRUE(proxy_class->IsProxyClass());
@@ -164,7 +83,8 @@
     std::vector<mirror::Class*> interfaces;
     interfaces.push_back(I.Get());
     interfaces.push_back(J.Get());
-    proxyClass = hs.NewHandle(GenerateProxyClass(soa, jclass_loader, "$Proxy1234", interfaces));
+    proxyClass = hs.NewHandle(
+        GenerateProxyClass(soa, jclass_loader, class_linker_, "$Proxy1234", interfaces));
   }
 
   ASSERT_TRUE(proxyClass != nullptr);
@@ -212,8 +132,10 @@
   Handle<mirror::Class> proxyClass1;
   {
     std::vector<mirror::Class*> interfaces;
-    proxyClass0 = hs.NewHandle(GenerateProxyClass(soa, jclass_loader, "$Proxy0", interfaces));
-    proxyClass1 = hs.NewHandle(GenerateProxyClass(soa, jclass_loader, "$Proxy1", interfaces));
+    proxyClass0 = hs.NewHandle(
+        GenerateProxyClass(soa, jclass_loader, class_linker_, "$Proxy0", interfaces));
+    proxyClass1 = hs.NewHandle(
+        GenerateProxyClass(soa, jclass_loader, class_linker_, "$Proxy1", interfaces));
   }
 
   ASSERT_TRUE(proxyClass0 != nullptr);
@@ -255,4 +177,5 @@
   EXPECT_EQ(field11->GetArtField(), &static_fields1->At(1));
 }
 
+}  // namespace proxy_test
 }  // namespace art
diff --git a/runtime/proxy_test.h b/runtime/proxy_test.h
new file mode 100644
index 0000000..b559823
--- /dev/null
+++ b/runtime/proxy_test.h
@@ -0,0 +1,114 @@
+/*
+ * Copyright (C) 2018 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#ifndef ART_RUNTIME_PROXY_TEST_H_
+#define ART_RUNTIME_PROXY_TEST_H_
+
+#include <jni.h>
+#include <vector>
+
+#include "art_method-inl.h"
+#include "class_linker-inl.h"
+#include "mirror/class-inl.h"
+#include "mirror/method.h"
+
+namespace art {
+namespace proxy_test {
+
+// Generate a proxy class with the given name and interfaces. This is a simplification from what
+// libcore does to fit to our test needs. We do not check for duplicated interfaces or methods and
+// we do not declare exceptions.
+mirror::Class* GenerateProxyClass(ScopedObjectAccess& soa,
+                                  jobject jclass_loader,
+                                  ClassLinker* class_linker,
+                                  const char* className,
+                                  const std::vector<mirror::Class*>& interfaces)
+    REQUIRES_SHARED(Locks::mutator_lock_) {
+  mirror::Class* javaLangObject = class_linker->FindSystemClass(soa.Self(), "Ljava/lang/Object;");
+  CHECK(javaLangObject != nullptr);
+
+  jclass javaLangClass = soa.AddLocalReference<jclass>(mirror::Class::GetJavaLangClass());
+
+  // Builds the interfaces array.
+  jobjectArray proxyClassInterfaces = soa.Env()->NewObjectArray(interfaces.size(), javaLangClass,
+                                                                nullptr);
+  soa.Self()->AssertNoPendingException();
+  for (size_t i = 0; i < interfaces.size(); ++i) {
+    soa.Env()->SetObjectArrayElement(proxyClassInterfaces, i,
+                                     soa.AddLocalReference<jclass>(interfaces[i]));
+  }
+
+  // Builds the method array.
+  jsize methods_count = 3;  // Object.equals, Object.hashCode and Object.toString.
+  for (mirror::Class* interface : interfaces) {
+    methods_count += interface->NumVirtualMethods();
+  }
+  jobjectArray proxyClassMethods = soa.Env()->NewObjectArray(
+      methods_count, soa.AddLocalReference<jclass>(mirror::Method::StaticClass()), nullptr);
+  soa.Self()->AssertNoPendingException();
+
+  jsize array_index = 0;
+  // Fill the method array
+  DCHECK_EQ(Runtime::Current()->GetClassLinker()->GetImagePointerSize(), kRuntimePointerSize);
+  ArtMethod* method = javaLangObject->FindClassMethod(
+      "equals", "(Ljava/lang/Object;)Z", kRuntimePointerSize);
+  CHECK(method != nullptr);
+  CHECK(!method->IsDirect());
+  CHECK(method->GetDeclaringClass() == javaLangObject);
+  DCHECK(!Runtime::Current()->IsActiveTransaction());
+  soa.Env()->SetObjectArrayElement(
+      proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
+          mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
+  method = javaLangObject->FindClassMethod("hashCode", "()I", kRuntimePointerSize);
+  CHECK(method != nullptr);
+  CHECK(!method->IsDirect());
+  CHECK(method->GetDeclaringClass() == javaLangObject);
+  soa.Env()->SetObjectArrayElement(
+      proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
+          mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
+  method = javaLangObject->FindClassMethod(
+      "toString", "()Ljava/lang/String;", kRuntimePointerSize);
+  CHECK(method != nullptr);
+  CHECK(!method->IsDirect());
+  CHECK(method->GetDeclaringClass() == javaLangObject);
+  soa.Env()->SetObjectArrayElement(
+      proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
+          mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), method)));
+  // Now adds all interfaces virtual methods.
+  for (mirror::Class* interface : interfaces) {
+    for (auto& m : interface->GetDeclaredVirtualMethods(kRuntimePointerSize)) {
+      soa.Env()->SetObjectArrayElement(
+          proxyClassMethods, array_index++, soa.AddLocalReference<jobject>(
+              mirror::Method::CreateFromArtMethod<kRuntimePointerSize, false>(soa.Self(), &m)));
+    }
+  }
+  CHECK_EQ(array_index, methods_count);
+
+  // Builds an empty exception array.
+  jobjectArray proxyClassThrows = soa.Env()->NewObjectArray(0, javaLangClass, nullptr);
+  soa.Self()->AssertNoPendingException();
+
+  mirror::Class* proxyClass = class_linker->CreateProxyClass(
+      soa, soa.Env()->NewStringUTF(className), proxyClassInterfaces, jclass_loader,
+      proxyClassMethods, proxyClassThrows);
+  soa.Self()->AssertNoPendingException();
+  return proxyClass;
+}
+
+}  // namespace proxy_test
+}  // namespace art
+
+#endif  // ART_RUNTIME_PROXY_TEST_H_
diff --git a/test/HiddenApiSignatures/Interface.java b/test/HiddenApiSignatures/Interface.java
new file mode 100644
index 0000000..f141d09
--- /dev/null
+++ b/test/HiddenApiSignatures/Interface.java
@@ -0,0 +1,21 @@
+/*
+ * Copyright (C) 2018 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package mypackage.packagea;
+
+public interface Interface {
+    public void method();
+}