| // Copyright 2014 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 "aemu/base/synchronization/AndroidMessageChannel.h" |
| |
| namespace android { |
| namespace base { |
| namespace guest { |
| |
| MessageChannelBase::MessageChannelBase(size_t capacity) : mCapacity(capacity) {} |
| |
| size_t MessageChannelBase::size() const { |
| AutoLock<Lock> lock(mLock); |
| return mCount; |
| } |
| |
| void MessageChannelBase::stop() { |
| android::base::guest::AutoLock<Lock> lock(mLock); |
| mStopped = true; |
| mCount = 0; |
| mCanRead.broadcast(); |
| mCanWrite.broadcastAndUnlock(&lock); |
| } |
| |
| bool MessageChannelBase::isStopped() const { |
| AutoLock<Lock> lock(mLock); |
| return isStoppedLocked(); |
| } |
| |
| void MessageChannelBase::waitForEmpty() { |
| AutoLock<Lock> lock(mLock); |
| while (mCount > 0) { |
| mCanWrite.wait(&lock); |
| } |
| } |
| |
| size_t MessageChannelBase::beforeWrite() { |
| mLock.lock(); |
| while (mCount >= mCapacity && !mStopped) { |
| mCanWrite.wait(&mLock); |
| } |
| // Return value is undefined if stopped, so let's save a branch and skip the |
| // check for it. |
| size_t result = mPos + mCount; |
| if (result >= mCapacity) { |
| result -= mCapacity; |
| } |
| return result; |
| } |
| |
| Optional<size_t> MessageChannelBase::beforeTryWrite() { |
| mLock.lock(); |
| |
| if (mCount >= mCapacity || mStopped) { |
| return {}; |
| } |
| size_t result = mPos + mCount; |
| if (result >= mCapacity) { |
| result -= mCapacity; |
| } |
| return result; |
| } |
| |
| void MessageChannelBase::afterWrite(bool success) { |
| if (success) { |
| ++mCount; |
| } |
| mCanRead.signalAndUnlock(&mLock); |
| } |
| |
| size_t MessageChannelBase::beforeRead() { |
| mLock.lock(); |
| while (mCount == 0 && !mStopped) { |
| mCanRead.wait(&mLock); |
| } |
| return mPos; // return value is undefined if stopped, so let's save a branch |
| } |
| |
| Optional<size_t> MessageChannelBase::beforeTryRead() { |
| mLock.lock(); |
| |
| if (mCount == 0 || mStopped) { |
| return {}; |
| } |
| return mPos; |
| } |
| |
| Optional<size_t> MessageChannelBase::beforeTimedRead(uint64_t wallTimeUs) { |
| mLock.lock(); |
| |
| while (mCount == 0 && !mStopped) { |
| if (!mCanRead.timedWait(&mLock, wallTimeUs)) { |
| return {}; |
| } |
| } |
| return mPos; |
| } |
| |
| void MessageChannelBase::afterRead(bool success) { |
| if (success) { |
| if (++mPos == mCapacity) { |
| mPos = 0U; |
| } |
| --mCount; |
| } |
| mCanWrite.signalAndUnlock(&mLock); |
| } |
| |
| } // namespace guest |
| } // namespace base |
| } // namespace android |