blob: 7aa60fd3b252c4d8d8df31c01b35ff251c4b54ab [file] [log] [blame]
/*
* Copyright (C) 2024 The Android Open Source Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#if __has_include (<android/dlext_private.h>)
#define IS_ANDROID_DL
#endif
#include "page_size_compat_helpers.h"
#include <android-base/parsebool.h>
#include <android-base/properties.h>
#include <android-base/silent_death_test.h>
using PageSize16KiBCompatTest_DeathTest = SilentDeathTest;
using ::android::base::ParseBool;
using ::android::base::ParseBoolResult;
#if defined(IS_ANDROID_DL)
#include <android/dlext_private.h>
#endif
static inline std::string CompatMode() {
return android::base::GetProperty("bionic.linker.16kb.app_compat.enabled", "false");
}
static inline bool CompatModeDisabled(const std::string& compat_mode) {
return ParseBool(compat_mode) == ParseBoolResult::kFalse;
}
static inline bool CompatModeEnabled(const std::string& compat_mode) {
return ParseBool(compat_mode) == ParseBoolResult::kTrue;
}
static inline bool CompatModeFatal(const std::string& compat_mode) {
return compat_mode == "fatal";
}
static inline std::string TestLibPath() {
return GetTestLibRoot() + "/libtest_elf_max_page_size_4kib.so";
}
TEST(PageSize16KiBCompatTest, ElfAlignment4KiB_LoadElf) {
if (getpagesize() != 0x4000) {
GTEST_SKIP() << "This test is only applicable to 16kB page-size devices";
}
std::string compat_mode = CompatMode();
if (CompatModeFatal(compat_mode)) {
GTEST_SKIP() << "This test is only applicable if dlopen() errors are not fatal";
}
std::string lib = TestLibPath();
void* handle = nullptr;
OpenTestLibrary(lib, CompatModeDisabled(compat_mode), &handle);
if (CompatModeEnabled(compat_mode)) CallTestFunction(handle);
}
TEST(PageSize16KiBCompatTest, ElfAlignment4KiB_NonAdajacentWritableSegments_LoadElf) {
if (getpagesize() != 0x4000) {
GTEST_SKIP() << "This test is only applicable to 16kB page-size devices";
}
std::string lib = GetPrebuiltElfDir() + "/libtest_invalid-rw_rx_rw_load_segments.so";
std::string compat_mode = CompatMode();
void* handle = nullptr;
OpenTestLibrary(lib, CompatModeDisabled(compat_mode), &handle);
}
TEST(PageSize16KiBCompatTest,
ElfAlignment4KiB_ExecutalbeSegmentsSeparatedByWritableSegment_LoadElf) {
if (getpagesize() != 0x4000) {
GTEST_SKIP() << "This test is only applicable to 16kB page-size devices";
}
std::string lib = GetPrebuiltElfDir() + "/libtest_invalid-rw_rx_rw_load_segments.so";
std::string compat_mode = CompatMode();
void* handle = nullptr;
OpenTestLibrary(lib, CompatModeDisabled(compat_mode), &handle);
}
TEST(PageSize16KiBCompatTest, ElfAlignment4KiB_LoadElf_perAppOption) {
if (getpagesize() != 0x4000) {
GTEST_SKIP() << "This test is only applicable to 16kB page-size devices";
}
std::string compat_mode = CompatMode();
if (CompatModeFatal(compat_mode)) {
GTEST_SKIP() << "This test is only applicable if dlopen() errors are not fatal";
}
#if defined(IS_ANDROID_DL)
android_set_16kb_appcompat_mode(true);
#endif
std::string lib = TestLibPath();
void* handle = nullptr;
OpenTestLibrary(lib, false /*should_fail*/, &handle);
CallTestFunction(handle);
#if defined(IS_ANDROID_DL)
android_set_16kb_appcompat_mode(false);
#endif
}
static void FatalDlError() {
std::string lib = TestLibPath();
void* handle = nullptr;
OpenTestLibrary(lib, true /*should_fail*/, &handle);
}
TEST(PageSize16KiBCompatTest_DeathTest, AppDlopenErrIsFatal) {
if (getpagesize() != 0x4000) {
GTEST_SKIP() << "This test is only applicable to 16kB page-size devices";
}
std::string compat_mode = CompatMode();
if (!CompatModeFatal(compat_mode)) {
GTEST_SKIP() << "This test is only applicable if dlopen() errors are fatal";
}
ASSERT_DEATH(FatalDlError(),
".*program alignment (.*) cannot be smaller than system page size.*");
}