/* * Copyright (C) 2021 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. */ #if defined(ART_TARGET_ANDROID) #include #include "native_loader_test.h" #include "nativehelper/scoped_utf_chars.h" #include "nativeloader/native_loader.h" namespace android { namespace nativeloader { using ::testing::StrEq; // Only need to test that the trivial lazy lib wrappers call through to the real // functions, but still have to mock things well enough to avoid null pointer // dereferences. class NativeLoaderLazyTest : public ::testing::Test { protected: void SetUp() override { mock = std::make_unique>(false); env = std::make_unique(); env->functions = CreateJNINativeInterface(); } void TearDown() override { // ResetNativeLoader isn't accessible through the lazy library, so we cannot // reset libnativeloader internal state. Hence be sure to not reuse the same // class loader/namespace names. delete env->functions; mock.reset(); } void CallCreateClassLoaderNamespace(const char* class_loader) { ON_CALL(*mock, JniObject_getParent(StrEq(class_loader))).WillByDefault(Return(nullptr)); EXPECT_CALL(*mock, mock_create_namespace) .WillOnce(Return(TO_MOCK_NAMESPACE(TO_ANDROID_NAMESPACE(class_loader)))); ON_CALL(*mock, mock_link_namespaces).WillByDefault(Return(true)); jstring err = CreateClassLoaderNamespace(env.get(), 17, env.get()->NewStringUTF(class_loader), false, env.get()->NewStringUTF("/data/app/foo/classes.dex"), env.get()->NewStringUTF("/data/app/foo"), /*permitted_path=*/nullptr, /*uses_library_list=*/nullptr); EXPECT_EQ(err, nullptr) << "Error is: " << std::string(ScopedUtfChars(env.get(), err).c_str()); } std::unique_ptr env; }; TEST_F(NativeLoaderLazyTest, CreateClassLoaderNamespace) { CallCreateClassLoaderNamespace("my_classloader_1"); } TEST_F(NativeLoaderLazyTest, OpenNativeLibrary) { bool needs_native_bridge; char* errmsg = nullptr; EXPECT_EQ(nullptr, OpenNativeLibrary(env.get(), 17, "libnotfound.so", env.get()->NewStringUTF("my_classloader"), /*caller_location=*/nullptr, /*library_path=*/nullptr, &needs_native_bridge, &errmsg)); EXPECT_NE(nullptr, errmsg); NativeLoaderFreeErrorMessage(errmsg); } TEST_F(NativeLoaderLazyTest, CloseNativeLibrary) { char* errmsg = nullptr; EXPECT_FALSE(CloseNativeLibrary(nullptr, false, &errmsg)); EXPECT_NE(nullptr, errmsg); NativeLoaderFreeErrorMessage(errmsg); } TEST_F(NativeLoaderLazyTest, OpenNativeLibraryInNamespace) { CallCreateClassLoaderNamespace("my_classloader_2"); struct NativeLoaderNamespace* ns = FindNativeLoaderNamespaceByClassLoader( env.get(), env.get()->NewStringUTF("my_classloader_2")); ASSERT_NE(nullptr, ns); bool needs_native_bridge; char* errmsg = nullptr; EXPECT_FALSE(OpenNativeLibraryInNamespace(ns, "libnotfound.so", &needs_native_bridge, &errmsg)); EXPECT_NE(nullptr, errmsg); NativeLoaderFreeErrorMessage(errmsg); } TEST_F(NativeLoaderLazyTest, FindNamespaceByClassLoader) { EXPECT_EQ(nullptr, FindNamespaceByClassLoader(env.get(), env.get()->NewStringUTF("namespace"))); } TEST_F(NativeLoaderLazyTest, FindNativeLoaderNamespaceByClassLoader) { EXPECT_EQ( nullptr, FindNativeLoaderNamespaceByClassLoader(env.get(), env.get()->NewStringUTF("namespace"))); } TEST_F(NativeLoaderLazyTest, NativeLoaderFreeErrorMessage) { NativeLoaderFreeErrorMessage(nullptr); } } // namespace nativeloader } // namespace android #endif // defined(ART_TARGET_ANDROID)