/* * 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. */ #include "membarrier.h" #include #include #if !defined(_WIN32) #include #include #include #endif #include "macros.h" #if __has_include() #include #define CHECK_MEMBARRIER_CMD(art_value, membarrier_value) \ static_assert(static_cast(art_value) == (membarrier_value), "Bad value for " # art_value) CHECK_MEMBARRIER_CMD(art::MembarrierCommand::kQuery, MEMBARRIER_CMD_QUERY); CHECK_MEMBARRIER_CMD(art::MembarrierCommand::kGlobal, MEMBARRIER_CMD_SHARED); CHECK_MEMBARRIER_CMD(art::MembarrierCommand::kPrivateExpedited, MEMBARRIER_CMD_PRIVATE_EXPEDITED); CHECK_MEMBARRIER_CMD(art::MembarrierCommand::kRegisterPrivateExpedited, MEMBARRIER_CMD_REGISTER_PRIVATE_EXPEDITED); CHECK_MEMBARRIER_CMD(art::MembarrierCommand::kPrivateExpedited, MEMBARRIER_CMD_PRIVATE_EXPEDITED); #undef CHECK_MEMBARRIER_CMD #endif // __has_include() namespace art { #if defined(__linux__) static bool IsMemBarrierSupported() { // Check kernel version supports membarrier(2). // MEMBARRIER_CMD_QUERY is supported since Linux 4.3. // MEMBARRIER_CMD_PRIVATE_EXPEDITED is supported since Linux 4.14. // MEMBARRIER_CMD_PRIVATE_EXPEDITED_SYNC_CORE is supported since Linux 4.16. // Lowest Linux version useful for ART is 4.14. static constexpr int kRequiredMajor = 4; static constexpr int kRequiredMinor = 14; struct utsname uts; int major, minor; if (uname(&uts) != 0 || strcmp(uts.sysname, "Linux") != 0 || sscanf(uts.release, "%d.%d", &major, &minor) != 2 || (major < kRequiredMajor || (major == kRequiredMajor && minor < kRequiredMinor))) { return false; } #if defined(__BIONIC__) // Avoid calling membarrier on older Android versions where membarrier may be barred by seccomp // causing the current process to be killed. The probing here could be considered expensive so // endeavour not to repeat too often. int api_level = android_get_device_api_level(); if (api_level < __ANDROID_API_Q__) { return false; } #endif // __BIONIC__ return true; } int membarrier(MembarrierCommand command) { static const bool membarrier_supported = IsMemBarrierSupported(); if (UNLIKELY(!membarrier_supported)) { errno = ENOSYS; return -1; } return syscall(__NR_membarrier, static_cast(command), 0); } #else // __linux__ int membarrier([[maybe_unused]] MembarrierCommand command) { errno = ENOSYS; return -1; } #endif // __linux__ } // namespace art