/* * 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 "src/profiling/memory/scoped_spinlock.h" #include #include #include "perfetto/ext/base/utils.h" namespace { constexpr bool IsPowerOfTwo(size_t v) { return (v != 0 && ((v & (v - 1)) == 0)); } // Wait for ~1s before timing out (+- spurious wakeups from the sleeps). constexpr unsigned kSleepAttempts = 1000; constexpr unsigned kLockAttemptsPerSleep = 1024; constexpr unsigned kSleepDurationUs = 1000; static_assert(IsPowerOfTwo(kLockAttemptsPerSleep), "lock attempts of power of 2 produce faster code."); } // namespace namespace perfetto { namespace profiling { void PoisonSpinlock(Spinlock* lock) { lock->poisoned.store(true, std::memory_order_relaxed); } void ScopedSpinlock::LockSlow(Mode mode) { size_t sleeps = 0; // We need to start with attempt = 1, otherwise // attempt % kLockAttemptsPerSleep is zero for the first iteration. for (size_t attempt = 1; mode == Mode::Blocking || attempt < kLockAttemptsPerSleep * kSleepAttempts; attempt++) { if (!lock_->locked.load(std::memory_order_relaxed) && PERFETTO_LIKELY( !lock_->locked.exchange(true, std::memory_order_acquire))) { locked_ = true; break; } if (attempt % kLockAttemptsPerSleep == 0) { usleep(kSleepDurationUs); sleeps++; } } blocked_us_ = kSleepDurationUs * sleeps; } } // namespace profiling } // namespace perfetto