Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 1 | /* |
| 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 "rsContext.h" |
| 18 | |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 19 | #include "rsThreadIO.h" |
| 20 | |
| 21 | using namespace android; |
| 22 | using namespace android::renderscript; |
| 23 | |
Alex Sakhartchouk | 6b0c004 | 2011-07-13 17:32:05 -0700 | [diff] [blame] | 24 | ThreadIO::ThreadIO() : mUsingSocket(false) { |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 25 | } |
| 26 | |
Alex Sakhartchouk | afb743a | 2010-11-09 17:00:54 -0800 | [diff] [blame] | 27 | ThreadIO::~ThreadIO() { |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 28 | } |
| 29 | |
Jason Sams | 1a4efa3 | 2011-05-17 15:01:29 -0700 | [diff] [blame] | 30 | void ThreadIO::init(bool useSocket) { |
| 31 | mUsingSocket = useSocket; |
Alex Sakhartchouk | 6b0c004 | 2011-07-13 17:32:05 -0700 | [diff] [blame] | 32 | mToCore.init(16 * 1024); |
Jason Sams | 1a4efa3 | 2011-05-17 15:01:29 -0700 | [diff] [blame] | 33 | |
| 34 | if (mUsingSocket) { |
| 35 | mToClientSocket.init(); |
| 36 | mToCoreSocket.init(); |
| 37 | } else { |
| 38 | mToClient.init(1024); |
| 39 | } |
Jason Sams | 8c0ee65 | 2009-08-25 14:49:07 -0700 | [diff] [blame] | 40 | } |
| 41 | |
Jason Sams | 1a4efa3 | 2011-05-17 15:01:29 -0700 | [diff] [blame] | 42 | void ThreadIO::shutdown() { |
| 43 | //LOGE("shutdown 1"); |
| 44 | mToCore.shutdown(); |
| 45 | //LOGE("shutdown 2"); |
| 46 | } |
| 47 | |
| 48 | void ThreadIO::coreFlush() { |
| 49 | //LOGE("coreFlush 1"); |
| 50 | if (mUsingSocket) { |
| 51 | } else { |
| 52 | mToCore.flush(); |
| 53 | } |
| 54 | //LOGE("coreFlush 2"); |
| 55 | } |
| 56 | |
| 57 | void * ThreadIO::coreHeader(uint32_t cmdID, size_t dataLen) { |
| 58 | //LOGE("coreHeader %i %i", cmdID, dataLen); |
| 59 | if (mUsingSocket) { |
| 60 | CoreCmdHeader hdr; |
| 61 | hdr.bytes = dataLen; |
| 62 | hdr.cmdID = cmdID; |
| 63 | mToCoreSocket.writeAsync(&hdr, sizeof(hdr)); |
| 64 | } else { |
| 65 | mCoreCommandSize = dataLen; |
| 66 | mCoreCommandID = cmdID; |
| 67 | mCoreDataPtr = (uint8_t *)mToCore.reserve(dataLen); |
| 68 | mCoreDataBasePtr = mCoreDataPtr; |
| 69 | } |
| 70 | //LOGE("coreHeader ret %p", mCoreDataPtr); |
| 71 | return mCoreDataPtr; |
| 72 | } |
| 73 | |
| 74 | void ThreadIO::coreData(const void *data, size_t dataLen) { |
| 75 | //LOGE("coreData %p %i", data, dataLen); |
| 76 | mToCoreSocket.writeAsync(data, dataLen); |
| 77 | //LOGE("coreData ret %p", mCoreDataPtr); |
| 78 | } |
| 79 | |
| 80 | void ThreadIO::coreCommit() { |
| 81 | //LOGE("coreCommit %p %p %i", mCoreDataPtr, mCoreDataBasePtr, mCoreCommandSize); |
| 82 | if (mUsingSocket) { |
| 83 | } else { |
| 84 | rsAssert((size_t)(mCoreDataPtr - mCoreDataBasePtr) <= mCoreCommandSize); |
| 85 | mToCore.commit(mCoreCommandID, mCoreCommandSize); |
| 86 | } |
| 87 | //LOGE("coreCommit ret"); |
| 88 | } |
| 89 | |
| 90 | void ThreadIO::coreCommitSync() { |
| 91 | //LOGE("coreCommitSync %p %p %i", mCoreDataPtr, mCoreDataBasePtr, mCoreCommandSize); |
| 92 | if (mUsingSocket) { |
| 93 | } else { |
| 94 | rsAssert((size_t)(mCoreDataPtr - mCoreDataBasePtr) <= mCoreCommandSize); |
| 95 | mToCore.commitSync(mCoreCommandID, mCoreCommandSize); |
| 96 | } |
| 97 | //LOGE("coreCommitSync ret"); |
| 98 | } |
| 99 | |
| 100 | void ThreadIO::clientShutdown() { |
| 101 | //LOGE("coreShutdown 1"); |
| 102 | mToClient.shutdown(); |
| 103 | //LOGE("coreShutdown 2"); |
| 104 | } |
| 105 | |
| 106 | void ThreadIO::coreSetReturn(const void *data, size_t dataLen) { |
| 107 | rsAssert(dataLen <= sizeof(mToCoreRet)); |
| 108 | memcpy(&mToCoreRet, data, dataLen); |
| 109 | } |
| 110 | |
| 111 | void ThreadIO::coreGetReturn(void *data, size_t dataLen) { |
| 112 | memcpy(data, &mToCoreRet, dataLen); |
| 113 | } |
| 114 | |
| 115 | |
Jason Sams | e0aab4a | 2011-08-12 15:05:15 -0700 | [diff] [blame^] | 116 | bool ThreadIO::playCoreCommands(Context *con, bool waitForCommand, uint64_t timeToWait) { |
Jason Sams | a44cb29 | 2009-06-04 17:58:03 -0700 | [diff] [blame] | 117 | bool ret = false; |
Jason Sams | e0aab4a | 2011-08-12 15:05:15 -0700 | [diff] [blame^] | 118 | uint64_t startTime = con->getTime(); |
| 119 | |
Alex Sakhartchouk | afb743a | 2010-11-09 17:00:54 -0800 | [diff] [blame] | 120 | while (!mToCore.isEmpty() || waitForCommand) { |
Jason Sams | 24371d9 | 2009-08-19 12:17:14 -0700 | [diff] [blame] | 121 | uint32_t cmdID = 0; |
| 122 | uint32_t cmdSize = 0; |
Jason Sams | a44cb29 | 2009-06-04 17:58:03 -0700 | [diff] [blame] | 123 | ret = true; |
Jason Sams | 1fddd90 | 2009-09-25 15:25:00 -0700 | [diff] [blame] | 124 | if (con->props.mLogTimes) { |
Joe Onorato | 76371ff | 2009-09-23 16:37:36 -0700 | [diff] [blame] | 125 | con->timerSet(Context::RS_TIMER_IDLE); |
| 126 | } |
Jason Sams | e0aab4a | 2011-08-12 15:05:15 -0700 | [diff] [blame^] | 127 | |
| 128 | uint64_t delay = 0; |
| 129 | if (waitForCommand) { |
| 130 | delay = timeToWait - (con->getTime() - startTime); |
| 131 | if (delay > timeToWait) { |
| 132 | delay = 0; |
| 133 | } |
| 134 | } |
| 135 | const void * data = mToCore.get(&cmdID, &cmdSize, delay); |
Jason Sams | e514b45 | 2009-09-25 14:51:22 -0700 | [diff] [blame] | 136 | if (!cmdSize) { |
Jason Sams | e0aab4a | 2011-08-12 15:05:15 -0700 | [diff] [blame^] | 137 | // exception or timeout occurred. |
Jason Sams | e514b45 | 2009-09-25 14:51:22 -0700 | [diff] [blame] | 138 | return false; |
| 139 | } |
Jason Sams | 1fddd90 | 2009-09-25 15:25:00 -0700 | [diff] [blame] | 140 | if (con->props.mLogTimes) { |
Joe Onorato | 76371ff | 2009-09-23 16:37:36 -0700 | [diff] [blame] | 141 | con->timerSet(Context::RS_TIMER_INTERNAL); |
| 142 | } |
Jason Sams | 732f1c0 | 2009-06-18 16:58:42 -0700 | [diff] [blame] | 143 | waitForCommand = false; |
Jason Sams | 992a0b7 | 2009-06-23 12:22:47 -0700 | [diff] [blame] | 144 | //LOGV("playCoreCommands 3 %i %i", cmdID, cmdSize); |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 145 | |
Jason Sams | 185b8b0 | 2011-01-16 14:54:28 -0800 | [diff] [blame] | 146 | if (cmdID >= (sizeof(gPlaybackFuncs) / sizeof(void *))) { |
| 147 | rsAssert(cmdID < (sizeof(gPlaybackFuncs) / sizeof(void *))); |
| 148 | LOGE("playCoreCommands error con %p, cmd %i", con, cmdID); |
Jason Sams | 1dda675 | 2011-01-18 18:12:26 -0800 | [diff] [blame] | 149 | mToCore.printDebugData(); |
Jason Sams | 185b8b0 | 2011-01-16 14:54:28 -0800 | [diff] [blame] | 150 | } |
Jason Sams | 5fb1aeb | 2011-04-27 15:12:49 -0700 | [diff] [blame] | 151 | gPlaybackFuncs[cmdID](con, data, cmdSize << 2); |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 152 | mToCore.next(); |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 153 | } |
Jason Sams | a44cb29 | 2009-06-04 17:58:03 -0700 | [diff] [blame] | 154 | return ret; |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 155 | } |
| 156 | |
Jason Sams | 1a4efa3 | 2011-05-17 15:01:29 -0700 | [diff] [blame] | 157 | RsMessageToClientType ThreadIO::getClientHeader(size_t *receiveLen, uint32_t *usrID) { |
| 158 | if (mUsingSocket) { |
| 159 | mToClientSocket.read(&mLastClientHeader, sizeof(mLastClientHeader)); |
| 160 | } else { |
| 161 | size_t bytesData = 0; |
Alex Sakhartchouk | b81a0eb | 2011-06-03 10:18:01 -0700 | [diff] [blame] | 162 | const uint32_t *d = (const uint32_t *)mToClient.get(&mLastClientHeader.cmdID, (uint32_t*)&bytesData); |
Jason Sams | 1a4efa3 | 2011-05-17 15:01:29 -0700 | [diff] [blame] | 163 | if (bytesData >= sizeof(uint32_t)) { |
| 164 | mLastClientHeader.userID = d[0]; |
| 165 | mLastClientHeader.bytes = bytesData - sizeof(uint32_t); |
| 166 | } else { |
| 167 | mLastClientHeader.userID = 0; |
| 168 | mLastClientHeader.bytes = 0; |
| 169 | } |
| 170 | } |
| 171 | receiveLen[0] = mLastClientHeader.bytes; |
| 172 | usrID[0] = mLastClientHeader.userID; |
| 173 | return (RsMessageToClientType)mLastClientHeader.cmdID; |
| 174 | } |
| 175 | |
| 176 | RsMessageToClientType ThreadIO::getClientPayload(void *data, size_t *receiveLen, |
| 177 | uint32_t *usrID, size_t bufferLen) { |
| 178 | receiveLen[0] = mLastClientHeader.bytes; |
| 179 | usrID[0] = mLastClientHeader.userID; |
| 180 | if (bufferLen < mLastClientHeader.bytes) { |
| 181 | return RS_MESSAGE_TO_CLIENT_RESIZE; |
| 182 | } |
| 183 | if (mUsingSocket) { |
| 184 | if (receiveLen[0]) { |
| 185 | mToClientSocket.read(data, receiveLen[0]); |
| 186 | } |
| 187 | return (RsMessageToClientType)mLastClientHeader.cmdID; |
| 188 | } else { |
| 189 | uint32_t bytesData = 0; |
| 190 | uint32_t commandID = 0; |
| 191 | const uint32_t *d = (const uint32_t *)mToClient.get(&commandID, &bytesData); |
| 192 | //LOGE("getMessageToClient 3 %i %i", commandID, bytesData); |
| 193 | //LOGE("getMessageToClient %i %i", commandID, *subID); |
| 194 | if (bufferLen >= receiveLen[0]) { |
| 195 | memcpy(data, d+1, receiveLen[0]); |
| 196 | mToClient.next(); |
| 197 | return (RsMessageToClientType)commandID; |
| 198 | } |
| 199 | } |
| 200 | return RS_MESSAGE_TO_CLIENT_RESIZE; |
| 201 | } |
| 202 | |
| 203 | bool ThreadIO::sendToClient(RsMessageToClientType cmdID, uint32_t usrID, const void *data, |
| 204 | size_t dataLen, bool waitForSpace) { |
| 205 | ClientCmdHeader hdr; |
| 206 | hdr.bytes = dataLen; |
| 207 | hdr.cmdID = cmdID; |
| 208 | hdr.userID = usrID; |
| 209 | if (mUsingSocket) { |
| 210 | mToClientSocket.writeAsync(&hdr, sizeof(hdr)); |
| 211 | if (dataLen) { |
| 212 | mToClientSocket.writeAsync(data, dataLen); |
| 213 | } |
| 214 | return true; |
| 215 | } else { |
| 216 | if (!waitForSpace) { |
| 217 | if (!mToClient.makeSpaceNonBlocking(dataLen + sizeof(hdr))) { |
| 218 | // Not enough room, and not waiting. |
| 219 | return false; |
| 220 | } |
| 221 | } |
| 222 | |
| 223 | //LOGE("sendMessageToClient 2"); |
| 224 | uint32_t *p = (uint32_t *)mToClient.reserve(dataLen + sizeof(usrID)); |
| 225 | p[0] = usrID; |
| 226 | if (dataLen > 0) { |
| 227 | memcpy(p+1, data, dataLen); |
| 228 | } |
| 229 | mToClient.commit(cmdID, dataLen + sizeof(usrID)); |
| 230 | //LOGE("sendMessageToClient 3"); |
| 231 | return true; |
| 232 | } |
| 233 | return false; |
| 234 | } |
Jason Sams | 326e0dd | 2009-05-22 14:03:28 -0700 | [diff] [blame] | 235 | |