blob: 9743a4927fee62a10d0b141d39a10a748d773e76 [file] [log] [blame]
// 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.
#ifndef RUNTIME_PLATFORM_TEXT_BUFFER_H_
#define RUNTIME_PLATFORM_TEXT_BUFFER_H_
#include "platform/allocation.h"
#include "platform/globals.h"
namespace dart {
// BaseTextBuffer maintains a dynamic character buffer with a printf-style way
// to append text. Internal buffer management is handled by subclasses.
class BaseTextBuffer : public ValueObject {
public:
BaseTextBuffer() : buffer_(nullptr), capacity_(0), length_(0) {}
BaseTextBuffer(char* buffer, intptr_t capacity)
: buffer_(buffer), capacity_(capacity), length_(0) {}
virtual ~BaseTextBuffer() {}
intptr_t Printf(const char* format, ...) PRINTF_ATTRIBUTE(2, 3);
intptr_t VPrintf(const char* format, va_list args);
void AddChar(char ch);
void AddString(const char* s);
void AddRaw(const uint8_t* buffer, intptr_t buffer_length);
void AddEscapedString(const char* s);
void AddEscapedUTF8(const char* s, intptr_t len);
void AddEscapedLatin1(const uint8_t* code_units, intptr_t len);
void AddEscapedUTF16(const uint16_t* code_units, intptr_t len);
// Returns a pointer to the current internal buffer. Whether the pointer is
// still valid after the BaseTextBuffer dies depends on the subclass.
char* buffer() const { return buffer_; }
intptr_t length() const { return length_; }
// Clears the stored contents. Unless specified otherwise by the subclass,
// should be assumed to invalidate the contents of previous calls to buffer().
virtual void Clear() = 0;
private:
intptr_t EscapedCodeUnitLength(uint32_t cu);
void EscapeAndAddCodeUnit(uint32_t cu);
void EscapeAndAddUTF16CodeUnit(uint16_t cu);
protected:
virtual bool EnsureCapacity(intptr_t len) = 0;
char* buffer_;
intptr_t capacity_;
intptr_t length_;
DISALLOW_COPY_AND_ASSIGN(BaseTextBuffer);
};
// TextBuffer uses manual memory management for the character buffer. Unless
// Steal() is used, the internal buffer is deallocated when the object dies.
class TextBuffer : public BaseTextBuffer {
public:
explicit TextBuffer(intptr_t buf_size);
~TextBuffer();
// Resets the contents of the internal buffer.
void Clear() { set_length(0); }
void set_length(intptr_t len) {
ASSERT(len >= 0);
ASSERT(len <= length_);
length_ = len;
buffer_[len] = '\0';
}
// Take ownership of the buffer contents. Future uses of the TextBuffer object
// will not affect the contents of the returned buffer.
// NOTE: TextBuffer is empty afterwards.
char* Steal();
private:
bool EnsureCapacity(intptr_t len);
DISALLOW_COPY_AND_ASSIGN(TextBuffer);
};
class BufferFormatter : public BaseTextBuffer {
public:
BufferFormatter(char* buffer, intptr_t size) : BaseTextBuffer(buffer, size) {
buffer_[length_] = '\0';
}
void Clear() {
length_ = 0;
buffer_[length_] = '\0';
}
private:
// We can't extend, so only return true if there's room.
bool EnsureCapacity(intptr_t len) { return length_ + len <= capacity_ - 1; }
DISALLOW_COPY_AND_ASSIGN(BufferFormatter);
};
} // namespace dart
#endif // RUNTIME_PLATFORM_TEXT_BUFFER_H_