Index: runtime/platform/text_buffer.cc |
diff --git a/runtime/platform/text_buffer.cc b/runtime/platform/text_buffer.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..b9c6ac7515bee735dc004f2deb3b71d75e462ec3 |
--- /dev/null |
+++ b/runtime/platform/text_buffer.cc |
@@ -0,0 +1,157 @@ |
+// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
+// for details. All rights reserved. Use of this source code is governed by a |
+// BSD-style license that can be found in the LICENSE file. |
+ |
+#include "platform/text_buffer.h" |
+ |
+#include "platform/assert.h" |
+#include "platform/globals.h" |
+#include "platform/utils.h" |
+#include "vm/os.h" |
+ |
+namespace dart { |
+ |
+TextBuffer::TextBuffer(intptr_t buf_size) { |
+ ASSERT(buf_size > 0); |
+ buf_ = reinterpret_cast<char*>(malloc(buf_size)); |
+ buf_size_ = buf_size; |
+ Clear(); |
+} |
+ |
+ |
+TextBuffer::~TextBuffer() { |
+ free(buf_); |
+ buf_ = NULL; |
+} |
+ |
+ |
+void TextBuffer::Clear() { |
+ msg_len_ = 0; |
+ buf_[0] = '\0'; |
+} |
+ |
+ |
+const char* TextBuffer::Steal() { |
+ const char* r = buf_; |
+ buf_ = NULL; |
+ buf_size_ = 0; |
+ msg_len_ = 0; |
+ return r; |
+} |
+ |
+ |
+void TextBuffer::AddChar(char ch) { |
+ EnsureCapacity(sizeof(ch)); |
+ buf_[msg_len_] = ch; |
+ msg_len_++; |
+ buf_[msg_len_] = '\0'; |
+} |
+ |
+ |
+intptr_t TextBuffer::Printf(const char* format, ...) { |
+ va_list args; |
+ va_start(args, format); |
+ intptr_t remaining = buf_size_ - msg_len_; |
+ ASSERT(remaining >= 0); |
+ intptr_t len = OS::VSNPrint(buf_ + msg_len_, remaining, format, args); |
+ va_end(args); |
+ if (len >= remaining) { |
+ EnsureCapacity(len); |
+ remaining = buf_size_ - msg_len_; |
+ ASSERT(remaining > len); |
+ va_list args2; |
+ va_start(args2, format); |
+ intptr_t len2 = OS::VSNPrint(buf_ + msg_len_, remaining, format, args2); |
+ va_end(args2); |
+ ASSERT(len == len2); |
+ } |
+ msg_len_ += len; |
+ buf_[msg_len_] = '\0'; |
+ return len; |
+} |
+ |
+ |
+// Write a UTF-16 code unit so it can be read by a JSON parser in a string |
+// literal. Use escape sequences for characters other than printable ASCII. |
+void TextBuffer::EscapeAndAddCodeUnit(uint32_t codeunit) { |
+ switch (codeunit) { |
+ case '"': |
+ Printf("%s", "\\\""); |
+ break; |
+ case '\\': |
+ Printf("%s", "\\\\"); |
+ break; |
+ case '/': |
+ Printf("%s", "\\/"); |
+ break; |
+ case '\b': |
+ Printf("%s", "\\b"); |
+ break; |
+ case '\f': |
+ Printf("%s", "\\f"); |
+ break; |
+ case '\n': |
+ Printf("%s", "\\n"); |
+ break; |
+ case '\r': |
+ Printf("%s", "\\r"); |
+ break; |
+ case '\t': |
+ Printf("%s", "\\t"); |
+ break; |
+ default: |
+ if (codeunit < 0x20) { |
+ // Encode character as \u00HH. |
+ uint32_t digit2 = (codeunit >> 4) & 0xf; |
+ uint32_t digit3 = (codeunit & 0xf); |
+ Printf("\\u00%c%c", |
+ digit2 > 9 ? 'A' + (digit2 - 10) : '0' + digit2, |
+ digit3 > 9 ? 'A' + (digit3 - 10) : '0' + digit3); |
+ } else if (codeunit > 127) { |
+ // Encode character as \uHHHH. |
+ uint32_t digit0 = (codeunit >> 12) & 0xf; |
+ uint32_t digit1 = (codeunit >> 8) & 0xf; |
+ uint32_t digit2 = (codeunit >> 4) & 0xf; |
+ uint32_t digit3 = (codeunit & 0xf); |
+ Printf("\\u%c%c%c%c", |
+ digit0 > 9 ? 'A' + (digit0 - 10) : '0' + digit0, |
+ digit1 > 9 ? 'A' + (digit1 - 10) : '0' + digit1, |
+ digit2 > 9 ? 'A' + (digit2 - 10) : '0' + digit2, |
+ digit3 > 9 ? 'A' + (digit3 - 10) : '0' + digit3); |
+ } else { |
+ AddChar(codeunit); |
+ } |
+ } |
+} |
+ |
+ |
+void TextBuffer::AddString(const char* s) { |
+ Printf("%s", s); |
+} |
+ |
+ |
+void TextBuffer::AddEscapedString(const char* s) { |
+ intptr_t len = strlen(s); |
+ for (int i = 0; i < len; i++) { |
+ EscapeAndAddCodeUnit(s[i]); |
+ } |
+} |
+ |
+ |
+void TextBuffer::EnsureCapacity(intptr_t len) { |
+ intptr_t remaining = buf_size_ - msg_len_; |
+ if (remaining <= len) { |
+ const int kBufferSpareCapacity = 64; // Somewhat arbitrary. |
+ // TODO(turnidge): do we need to guard against overflow or other |
+ // security issues here? Text buffers are used by the debugger |
+ // to send user-controlled data (e.g. values of string variables) to |
+ // the debugger front-end. |
+ intptr_t new_size = buf_size_ + len + kBufferSpareCapacity; |
+ char* new_buf = reinterpret_cast<char*>(realloc(buf_, new_size)); |
+ ASSERT(new_buf != NULL); |
+ buf_ = new_buf; |
+ buf_size_ = new_size; |
+ } |
+} |
+ |
+} // namespace dart |