/* * Copyright (C) 2019 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. */ #include "palette/palette.h" #include #include #include #include #include #include "base/common_art_test.h" #include "gtest/gtest.h" #ifdef ART_TARGET_ANDROID #include "android-modules-utils/sdk_level.h" #include "android/api-level.h" #endif namespace { pid_t GetTid() { #ifdef __BIONIC__ return gettid(); #else // __BIONIC__ return syscall(__NR_gettid); #endif // __BIONIC__ } #ifdef ART_TARGET_ANDROID bool PaletteSetTaskProfilesIsSupported(palette_status_t res) { if (android::modules::sdklevel::IsAtLeastU()) { return true; } EXPECT_EQ(PALETTE_STATUS_NOT_SUPPORTED, res) << "Device API level: " << android_get_device_api_level(); return false; } bool PaletteDebugStoreIsSupported() { // TODO(b/345433959): Switch to android::modules::sdklevel::IsAtLeastW return android_get_device_api_level() >= 36; } #endif } // namespace class PaletteClientTest : public testing::Test {}; TEST_F(PaletteClientTest, SchedPriority) { int32_t tid = GetTid(); int32_t saved_priority; EXPECT_EQ(PALETTE_STATUS_OK, PaletteSchedGetPriority(tid, &saved_priority)); EXPECT_EQ(PALETTE_STATUS_INVALID_ARGUMENT, PaletteSchedSetPriority(tid, /*java_priority=*/ 0)); EXPECT_EQ(PALETTE_STATUS_INVALID_ARGUMENT, PaletteSchedSetPriority(tid, /*java_priority=*/ -1)); EXPECT_EQ(PALETTE_STATUS_INVALID_ARGUMENT, PaletteSchedSetPriority(tid, /*java_priority=*/ 11)); EXPECT_EQ(PALETTE_STATUS_OK, PaletteSchedSetPriority(tid, /*java_priority=*/ 1)); EXPECT_EQ(PALETTE_STATUS_OK, PaletteSchedSetPriority(tid, saved_priority)); } TEST_F(PaletteClientTest, Trace) { bool enabled = false; EXPECT_EQ(PALETTE_STATUS_OK, PaletteTraceEnabled(&enabled)); EXPECT_EQ(PALETTE_STATUS_OK, PaletteTraceBegin("Hello world!")); EXPECT_EQ(PALETTE_STATUS_OK, PaletteTraceEnd()); EXPECT_EQ(PALETTE_STATUS_OK, PaletteTraceIntegerValue("Beans", /*value=*/ 3)); } TEST_F(PaletteClientTest, Ashmem) { #ifndef ART_TARGET_ANDROID GTEST_SKIP() << "ashmem is only supported on Android"; #else int fd; EXPECT_EQ(PALETTE_STATUS_OK, PaletteAshmemCreateRegion("ashmem-test", 4096, &fd)); EXPECT_EQ(PALETTE_STATUS_OK, PaletteAshmemSetProtRegion(fd, PROT_READ | PROT_EXEC)); EXPECT_EQ(0, close(fd)); #endif } class PaletteClientJniTest : public art::CommonArtTest {}; TEST_F(PaletteClientJniTest, JniInvocation) { bool enabled; EXPECT_EQ(PALETTE_STATUS_OK, PaletteShouldReportJniInvocations(&enabled)); std::string boot_class_path_string = GetClassPathOption("-Xbootclasspath:", GetLibCoreDexFileNames()); std::string boot_class_path_locations_string = GetClassPathOption("-Xbootclasspath-locations:", GetLibCoreDexLocations()); JavaVMOption options[] = { {.optionString = boot_class_path_string.c_str(), .extraInfo = nullptr}, {.optionString = boot_class_path_locations_string.c_str(), .extraInfo = nullptr}, }; JavaVMInitArgs vm_args = { .version = JNI_VERSION_1_6, .nOptions = std::size(options), .options = options, .ignoreUnrecognized = JNI_TRUE, }; JavaVM* jvm = nullptr; JNIEnv* env = nullptr; EXPECT_EQ(JNI_OK, JNI_CreateJavaVM(&jvm, &env, &vm_args)); ASSERT_NE(nullptr, env); PaletteNotifyBeginJniInvocation(env); PaletteNotifyEndJniInvocation(env); EXPECT_EQ(JNI_OK, jvm->DestroyJavaVM()); } TEST_F(PaletteClientTest, SetTaskProfiles) { #ifndef ART_TARGET_ANDROID GTEST_SKIP() << "SetTaskProfiles is only supported on Android"; #else const char* profiles[] = {"ProcessCapacityHigh", "TimerSlackNormal"}; palette_status_t res = PaletteSetTaskProfiles(GetTid(), &profiles[0], 2); if (PaletteSetTaskProfilesIsSupported(res)) { // SetTaskProfiles will only work fully if we run as root. Otherwise it'll // return false which is mapped to PALETTE_STATUS_FAILED_CHECK_LOG. if (getuid() == 0) { EXPECT_EQ(PALETTE_STATUS_OK, res); } else { EXPECT_EQ(PALETTE_STATUS_FAILED_CHECK_LOG, res); } } #endif } TEST_F(PaletteClientTest, SetTaskProfilesCpp) { #ifndef ART_TARGET_ANDROID GTEST_SKIP() << "SetTaskProfiles is only supported on Android"; #else std::vector profiles = {"ProcessCapacityHigh", "TimerSlackNormal"}; palette_status_t res = PaletteSetTaskProfiles(GetTid(), profiles); if (PaletteSetTaskProfilesIsSupported(res)) { // SetTaskProfiles will only work fully if we run as root. Otherwise it'll // return false which is mapped to PALETTE_STATUS_FAILED_CHECK_LOG. if (getuid() == 0) { EXPECT_EQ(PALETTE_STATUS_OK, res); } else { EXPECT_EQ(PALETTE_STATUS_FAILED_CHECK_LOG, res); } } #endif } TEST_F(PaletteClientTest, DebugStore) { #ifndef ART_TARGET_ANDROID GTEST_SKIP() << "DebugStore is only supported on Android"; #else std::array result{}; // Make sure the we are on a correct API level. if (!PaletteDebugStoreIsSupported()) { GTEST_SKIP() << "DebugStore is only supported on API 36+"; } palette_status_t pstatus = PaletteDebugStoreGetString(result.data(), result.size()); EXPECT_EQ(PALETTE_STATUS_OK, pstatus); size_t len = strnlen(result.data(), result.size()); EXPECT_TRUE(len < result.size()); const char* start = "1,0,"; const char* end = "::"; EXPECT_TRUE(len > strlen(start) + strlen(end)); EXPECT_EQ(strncmp(result.data() + len - strlen(end), end, strlen(end)), 0); #endif }