1 /* 2 * Copyright (C) 2006 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 <binder/BufferedTextOutput.h> 18 #include <binder/Debug.h> 19 20 #include <cutils/atomic.h> 21 #include <cutils/threads.h> 22 #include <utils/Log.h> 23 #include <utils/RefBase.h> 24 #include <utils/Vector.h> 25 26 #include <private/binder/Static.h> 27 28 #include <pthread.h> 29 #include <stdio.h> 30 #include <stdlib.h> 31 32 // --------------------------------------------------------------------------- 33 34 namespace android { 35 36 struct BufferedTextOutput::BufferState : public RefBase 37 { 38 explicit BufferState(int32_t _seq) 39 : seq(_seq) 40 , buffer(nullptr) 41 , bufferPos(0) 42 , bufferSize(0) 43 , atFront(true) 44 , indent(0) 45 , bundle(0) { 46 } 47 ~BufferState() { 48 free(buffer); 49 } 50 51 status_t append(const char* txt, size_t len) { 52 if ((len+bufferPos) > bufferSize) { 53 size_t newSize = ((len+bufferPos)*3)/2; 54 if (newSize < (len+bufferPos)) return NO_MEMORY; // overflow 55 void* b = realloc(buffer, newSize); 56 if (!b) return NO_MEMORY; 57 buffer = (char*)b; 58 bufferSize = newSize; 59 } 60 memcpy(buffer+bufferPos, txt, len); 61 bufferPos += len; 62 return NO_ERROR; 63 } 64 65 void restart() { 66 bufferPos = 0; 67 atFront = true; 68 if (bufferSize > 256) { 69 void* b = realloc(buffer, 256); 70 if (b) { 71 buffer = (char*)b; 72 bufferSize = 256; 73 } 74 } 75 } 76 77 const int32_t seq; 78 char* buffer; 79 size_t bufferPos; 80 size_t bufferSize; 81 bool atFront; 82 int32_t indent; 83 int32_t bundle; 84 }; 85 86 struct BufferedTextOutput::ThreadState 87 { 88 Vector<sp<BufferedTextOutput::BufferState> > states; 89 }; 90 91 static pthread_mutex_t gMutex = PTHREAD_MUTEX_INITIALIZER; 92 93 static thread_store_t tls; 94 95 BufferedTextOutput::ThreadState* BufferedTextOutput::getThreadState() 96 { 97 ThreadState* ts = (ThreadState*) thread_store_get( &tls ); 98 if (ts) return ts; 99 ts = new ThreadState; 100 thread_store_set( &tls, ts, threadDestructor ); 101 return ts; 102 } 103 104 void BufferedTextOutput::threadDestructor(void *st) 105 { 106 delete ((ThreadState*)st); 107 } 108 109 static volatile int32_t gSequence = 0; 110 111 static volatile int32_t gFreeBufferIndex = -1; 112 113 static int32_t allocBufferIndex() 114 { 115 int32_t res = -1; 116 117 pthread_mutex_lock(&gMutex); 118 119 if (gFreeBufferIndex >= 0) { 120 res = gFreeBufferIndex; 121 gFreeBufferIndex = gTextBuffers[res]; 122 gTextBuffers.editItemAt(res) = -1; 123 124 } else { 125 res = gTextBuffers.size(); 126 gTextBuffers.add(-1); 127 } 128 129 pthread_mutex_unlock(&gMutex); 130 131 return res; 132 } 133 134 static void freeBufferIndex(int32_t idx) 135 { 136 pthread_mutex_lock(&gMutex); 137 gTextBuffers.editItemAt(idx) = gFreeBufferIndex; 138 gFreeBufferIndex = idx; 139 pthread_mutex_unlock(&gMutex); 140 } 141 142 // --------------------------------------------------------------------------- 143 144 BufferedTextOutput::BufferedTextOutput(uint32_t flags) 145 : mFlags(flags) 146 , mSeq(android_atomic_inc(&gSequence)) 147 , mIndex(allocBufferIndex()) 148 { 149 mGlobalState = new BufferState(mSeq); 150 if (mGlobalState) mGlobalState->incStrong(this); 151 } 152 153 BufferedTextOutput::~BufferedTextOutput() 154 { 155 if (mGlobalState) mGlobalState->decStrong(this); 156 freeBufferIndex(mIndex); 157 } 158 159 status_t BufferedTextOutput::print(const char* txt, size_t len) 160 { 161 //printf("BufferedTextOutput: printing %d\n", len); 162 163 AutoMutex _l(mLock); 164 BufferState* b = getBuffer(); 165 166 const char* const end = txt+len; 167 168 status_t err; 169 170 while (txt < end) { 171 // Find the next line. 172 const char* first = txt; 173 while (txt < end && *txt != '\n') txt++; 174 175 // Include this and all following empty lines. 176 while (txt < end && *txt == '\n') txt++; 177 178 // Special cases for first data on a line. 179 if (b->atFront) { 180 if (b->indent > 0) { 181 // If this is the start of a line, add the indent. 182 const char* prefix = stringForIndent(b->indent); 183 err = b->append(prefix, strlen(prefix)); 184 if (err != NO_ERROR) return err; 185 186 } else if (*(txt-1) == '\n' && !b->bundle) { 187 // Fast path: if we are not indenting or bundling, and 188 // have been given one or more complete lines, just write 189 // them out without going through the buffer. 190 191 // Slurp up all of the lines. 192 const char* lastLine = txt; 193 while (txt < end) { 194 if (*txt++ == '\n') lastLine = txt; 195 } 196 struct iovec vec; 197 vec.iov_base = (void*)first; 198 vec.iov_len = lastLine-first; 199 //printf("Writing %d bytes of data!\n", vec.iov_len); 200 writeLines(vec, 1); 201 txt = lastLine; 202 continue; 203 } 204 } 205 206 // Append the new text to the buffer. 207 err = b->append(first, txt-first); 208 if (err != NO_ERROR) return err; 209 b->atFront = *(txt-1) == '\n'; 210 211 // If we have finished a line and are not bundling, write 212 // it out. 213 //printf("Buffer is now %d bytes\n", b->bufferPos); 214 if (b->atFront && !b->bundle) { 215 struct iovec vec; 216 vec.iov_base = b->buffer; 217 vec.iov_len = b->bufferPos; 218 //printf("Writing %d bytes of data!\n", vec.iov_len); 219 writeLines(vec, 1); 220 b->restart(); 221 } 222 } 223 224 return NO_ERROR; 225 } 226 227 void BufferedTextOutput::moveIndent(int delta) 228 { 229 AutoMutex _l(mLock); 230 BufferState* b = getBuffer(); 231 b->indent += delta; 232 if (b->indent < 0) b->indent = 0; 233 } 234 235 void BufferedTextOutput::pushBundle() 236 { 237 AutoMutex _l(mLock); 238 BufferState* b = getBuffer(); 239 b->bundle++; 240 } 241 242 void BufferedTextOutput::popBundle() 243 { 244 AutoMutex _l(mLock); 245 BufferState* b = getBuffer(); 246 b->bundle--; 247 LOG_FATAL_IF(b->bundle < 0, 248 "TextOutput::popBundle() called more times than pushBundle()"); 249 if (b->bundle < 0) b->bundle = 0; 250 251 if (b->bundle == 0) { 252 // Last bundle, write out data if it is complete. If it is not 253 // complete, don't write until the last line is done... this may 254 // or may not be the write thing to do, but it's the easiest. 255 if (b->bufferPos > 0 && b->atFront) { 256 struct iovec vec; 257 vec.iov_base = b->buffer; 258 vec.iov_len = b->bufferPos; 259 writeLines(vec, 1); 260 b->restart(); 261 } 262 } 263 } 264 265 BufferedTextOutput::BufferState* BufferedTextOutput::getBuffer() const 266 { 267 if ((mFlags&MULTITHREADED) != 0) { 268 ThreadState* ts = getThreadState(); 269 if (ts) { 270 while (ts->states.size() <= (size_t)mIndex) ts->states.add(nullptr); 271 BufferState* bs = ts->states[mIndex].get(); 272 if (bs != nullptr && bs->seq == mSeq) return bs; 273 274 ts->states.editItemAt(mIndex) = new BufferState(mIndex); 275 bs = ts->states[mIndex].get(); 276 if (bs != nullptr) return bs; 277 } 278 } 279 280 return mGlobalState; 281 } 282 283 }; // namespace android 284