blob: 162a73022ba2702885cc58a7db6f7783dd4fb450 [file] [log] [blame]
// 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