/* * Copyright (C) 2015 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 "ring.h" #include #include #include namespace android { RingBuffer::RingBuffer(size_t size) : mSize(size), mData((sensors_event_t *)malloc(sizeof(sensors_event_t) * mSize)), mReadPos(0), mWritePos(0) { } RingBuffer::~RingBuffer() { free(mData); mData = NULL; } ssize_t RingBuffer::write(const sensors_event_t *ev, size_t size) { Mutex::Autolock autoLock(mLock); size_t numAvailableToRead = mWritePos - mReadPos; size_t numAvailableToWrite = mSize - numAvailableToRead; if (size > numAvailableToWrite) { size = numAvailableToWrite; } size_t writePos = (mWritePos % mSize); size_t copy = mSize - writePos; if (copy > size) { copy = size; } memcpy(&mData[writePos], ev, copy * sizeof(sensors_event_t)); if (size > copy) { memcpy(mData, &ev[copy], (size - copy) * sizeof(sensors_event_t)); } mWritePos += size; if (numAvailableToRead == 0 && size > 0) { mNotEmptyCondition.broadcast(); } return size; } ssize_t RingBuffer::read(sensors_event_t *ev, size_t size) { Mutex::Autolock autoLock(mLock); size_t numAvailableToRead; for (;;) { numAvailableToRead = mWritePos - mReadPos; if (numAvailableToRead > 0) { break; } mNotEmptyCondition.wait(mLock); } if (size > numAvailableToRead) { size = numAvailableToRead; } size_t readPos = (mReadPos % mSize); size_t copy = mSize - readPos; if (copy > size) { copy = size; } memcpy(ev, &mData[readPos], copy * sizeof(sensors_event_t)); if (size > copy) { memcpy(&ev[copy], mData, (size - copy) * sizeof(sensors_event_t)); } mReadPos += size; return size; } LockfreeBuffer::LockfreeBuffer(void* buf, size_t size) : mData((sensors_event_t *)buf), mSize(size/sizeof(sensors_event_t)), mWritePos(0), mCounter(1) { memset(mData, 0, size); } LockfreeBuffer::~LockfreeBuffer() { memset(mData, 0, mSize*sizeof(sensors_event_t)); } void LockfreeBuffer::write(const sensors_event_t *ev, size_t size) { if (!mSize) { return; } while(size--) { // part before reserved0 field memcpy(&mData[mWritePos], ev, offsetof(sensors_event_t, reserved0)); // part after reserved0 field memcpy(reinterpret_cast(&mData[mWritePos]) + offsetof(sensors_event_t, timestamp), reinterpret_cast(ev) + offsetof(sensors_event_t, timestamp), sizeof(sensors_event_t) - offsetof(sensors_event_t, timestamp)); // barrier before writing the atomic counter std::atomic_thread_fence(std::memory_order_release); mData[mWritePos].reserved0 = mCounter++; // barrier after writing the atomic counter std::atomic_thread_fence(std::memory_order_release); ++ev; if (++mWritePos >= mSize) { mWritePos = 0; } } } } // namespace android