blob: e87525e438cc8f83eac656c2cf491042b29b955e [file] [log] [blame]
Jason Samsd19f10d2009-05-22 14:03:28 -07001/*
2 * Copyright (C) 2009 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "rsLocklessFifo.h"
Jason Sams3b9c52a2010-10-14 17:48:46 -070018#include "utils/Timers.h"
19#include "utils/StopWatch.h"
Jason Samsd19f10d2009-05-22 14:03:28 -070020
21using namespace android;
Jason Samsc1d726c2010-03-18 11:39:44 -070022using namespace android::renderscript;
Jason Samsd19f10d2009-05-22 14:03:28 -070023
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080024LocklessCommandFifo::LocklessCommandFifo() {
Jason Samsd19f10d2009-05-22 14:03:28 -070025}
26
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080027LocklessCommandFifo::~LocklessCommandFifo() {
Jason Samsf5b45962009-08-25 14:49:07 -070028 if (!mInShutdown) {
29 shutdown();
30 }
31 free(mBuffer);
32}
33
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080034void LocklessCommandFifo::shutdown() {
Jason Samsf5b45962009-08-25 14:49:07 -070035 mInShutdown = true;
36 mSignalToWorker.set();
Jason Samsd19f10d2009-05-22 14:03:28 -070037}
38
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080039bool LocklessCommandFifo::init(uint32_t sizeInBytes) {
Jason Samsd19f10d2009-05-22 14:03:28 -070040 // Add room for a buffer reset command
41 mBuffer = static_cast<uint8_t *>(malloc(sizeInBytes + 4));
42 if (!mBuffer) {
43 LOGE("LocklessFifo allocation failure");
44 return false;
45 }
46
Jason Sams5f7fc272009-06-18 16:58:42 -070047 if (!mSignalToControl.init() || !mSignalToWorker.init()) {
48 LOGE("Signal setup failed");
Jason Samsd19f10d2009-05-22 14:03:28 -070049 free(mBuffer);
50 return false;
51 }
52
Jason Samsf5b45962009-08-25 14:49:07 -070053 mInShutdown = false;
Jason Samsd19f10d2009-05-22 14:03:28 -070054 mSize = sizeInBytes;
55 mPut = mBuffer;
56 mGet = mBuffer;
57 mEnd = mBuffer + (sizeInBytes) - 1;
Jason Sams3bc47d42009-11-12 15:10:25 -080058 //dumpState("init");
Jason Samsd19f10d2009-05-22 14:03:28 -070059 return true;
60}
61
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080062uint32_t LocklessCommandFifo::getFreeSpace() const {
Jason Samsd19f10d2009-05-22 14:03:28 -070063 int32_t freeSpace = 0;
64 //dumpState("getFreeSpace");
65
66 if (mPut >= mGet) {
67 freeSpace = mEnd - mPut;
68 } else {
69 freeSpace = mGet - mPut;
70 }
71
72 if (freeSpace < 0) {
73 freeSpace = 0;
74 }
Jason Samsd19f10d2009-05-22 14:03:28 -070075 return freeSpace;
76}
77
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080078bool LocklessCommandFifo::isEmpty() const {
Jason Samsd19f10d2009-05-22 14:03:28 -070079 return mPut == mGet;
80}
81
82
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080083void * LocklessCommandFifo::reserve(uint32_t sizeInBytes) {
Jason Samse2ae85f2009-06-03 16:04:54 -070084 // Add space for command header and loop token;
85 sizeInBytes += 8;
Jason Samsd19f10d2009-05-22 14:03:28 -070086
87 //dumpState("reserve");
88 if (getFreeSpace() < sizeInBytes) {
89 makeSpace(sizeInBytes);
90 }
91
92 return mPut + 4;
93}
94
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080095void LocklessCommandFifo::commit(uint32_t command, uint32_t sizeInBytes) {
Jason Sams516c3192009-10-06 13:58:47 -070096 if (mInShutdown) {
97 return;
98 }
Jason Samsd19f10d2009-05-22 14:03:28 -070099 //dumpState("commit 1");
100 reinterpret_cast<uint16_t *>(mPut)[0] = command;
101 reinterpret_cast<uint16_t *>(mPut)[1] = sizeInBytes;
102 mPut += ((sizeInBytes + 3) & ~3) + 4;
103 //dumpState("commit 2");
Jason Sams5f7fc272009-06-18 16:58:42 -0700104 mSignalToWorker.set();
Jason Samsd19f10d2009-05-22 14:03:28 -0700105}
106
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800107void LocklessCommandFifo::commitSync(uint32_t command, uint32_t sizeInBytes) {
Jason Sams516c3192009-10-06 13:58:47 -0700108 if (mInShutdown) {
109 return;
110 }
Jason Sams3b9c52a2010-10-14 17:48:46 -0700111
112 //char buf[1024];
113 //sprintf(buf, "RenderScript LocklessCommandFifo::commitSync %p %i %i", this, command, sizeInBytes);
114 //StopWatch compileTimer(buf);
Jason Samsd19f10d2009-05-22 14:03:28 -0700115 commit(command, sizeInBytes);
116 flush();
117}
118
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800119void LocklessCommandFifo::flush() {
Jason Samsd19f10d2009-05-22 14:03:28 -0700120 //dumpState("flush 1");
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800121 while (mPut != mGet) {
Jason Sams5f7fc272009-06-18 16:58:42 -0700122 mSignalToControl.wait();
Jason Samsd19f10d2009-05-22 14:03:28 -0700123 }
124 //dumpState("flush 2");
125}
126
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800127void LocklessCommandFifo::wait() {
128 while (isEmpty() && !mInShutdown) {
Jason Samsc1d726c2010-03-18 11:39:44 -0700129 mSignalToControl.set();
130 mSignalToWorker.wait();
131 }
132}
133
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800134const void * LocklessCommandFifo::get(uint32_t *command, uint32_t *bytesData) {
135 while (1) {
Jason Sams5f7fc272009-06-18 16:58:42 -0700136 //dumpState("get");
Jason Samsc1d726c2010-03-18 11:39:44 -0700137 wait();
Jason Samsa9e7a052009-09-25 14:51:22 -0700138 if (mInShutdown) {
139 *command = 0;
140 *bytesData = 0;
141 return 0;
142 }
143
Jason Samsd19f10d2009-05-22 14:03:28 -0700144 *command = reinterpret_cast<const uint16_t *>(mGet)[0];
145 *bytesData = reinterpret_cast<const uint16_t *>(mGet)[1];
Jason Samsd19f10d2009-05-22 14:03:28 -0700146 if (*command) {
147 // non-zero command is valid
148 return mGet+4;
149 }
Jason Samsf5b45962009-08-25 14:49:07 -0700150
Jason Samsd19f10d2009-05-22 14:03:28 -0700151 // zero command means reset to beginning.
152 mGet = mBuffer;
153 }
154}
155
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800156void LocklessCommandFifo::next() {
Jason Samsd19f10d2009-05-22 14:03:28 -0700157 uint32_t bytes = reinterpret_cast<const uint16_t *>(mGet)[1];
158 mGet += ((bytes + 3) & ~3) + 4;
Jason Sams5f7fc272009-06-18 16:58:42 -0700159 if (isEmpty()) {
160 mSignalToControl.set();
161 }
Jason Samsd19f10d2009-05-22 14:03:28 -0700162 //dumpState("next");
163}
164
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800165bool LocklessCommandFifo::makeSpaceNonBlocking(uint32_t bytes) {
Jason Samsd0cb1062010-08-18 12:38:03 -0700166 //dumpState("make space non-blocking");
167 if ((mPut+bytes) > mEnd) {
168 // Need to loop regardless of where get is.
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800169 if ((mGet > mPut) && (mBuffer+4 >= mGet)) {
Jason Samsd0cb1062010-08-18 12:38:03 -0700170 return false;
171 }
172
173 // Toss in a reset then the normal wait for space will do the rest.
174 reinterpret_cast<uint16_t *>(mPut)[0] = 0;
175 reinterpret_cast<uint16_t *>(mPut)[1] = 0;
176 mPut = mBuffer;
177 mSignalToWorker.set();
178 }
179
180 // it will fit here so we just need to wait for space.
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800181 if (getFreeSpace() < bytes) {
Jason Samsd0cb1062010-08-18 12:38:03 -0700182 return false;
183 }
184
185 return true;
186}
187
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800188void LocklessCommandFifo::makeSpace(uint32_t bytes) {
Jason Samse2ae85f2009-06-03 16:04:54 -0700189 //dumpState("make space");
Jason Samsd19f10d2009-05-22 14:03:28 -0700190 if ((mPut+bytes) > mEnd) {
191 // Need to loop regardless of where get is.
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800192 while ((mGet > mPut) && (mBuffer+4 >= mGet)) {
Jason Samse60446b2009-09-24 14:55:38 -0700193 usleep(100);
Jason Samsd19f10d2009-05-22 14:03:28 -0700194 }
195
196 // Toss in a reset then the normal wait for space will do the rest.
197 reinterpret_cast<uint16_t *>(mPut)[0] = 0;
198 reinterpret_cast<uint16_t *>(mPut)[1] = 0;
Jason Samse2ae85f2009-06-03 16:04:54 -0700199 mPut = mBuffer;
Jason Samsd0cb1062010-08-18 12:38:03 -0700200 mSignalToWorker.set();
Jason Samsd19f10d2009-05-22 14:03:28 -0700201 }
202
203 // it will fit here so we just need to wait for space.
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800204 while (getFreeSpace() < bytes) {
Jason Samse60446b2009-09-24 14:55:38 -0700205 usleep(100);
Jason Samsd19f10d2009-05-22 14:03:28 -0700206 }
Jason Samsf5b45962009-08-25 14:49:07 -0700207
Jason Samsd19f10d2009-05-22 14:03:28 -0700208}
209
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800210void LocklessCommandFifo::dumpState(const char *s) const {
Jason Samsd0cb1062010-08-18 12:38:03 -0700211 LOGV("%s %p put %p, get %p, buf %p, end %p", s, this, mPut, mGet, mBuffer, mEnd);
Jason Samsd19f10d2009-05-22 14:03:28 -0700212}
213