// Copyright 2014 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_STRING_STREAM_H_ #define V8_STRING_STREAM_H_ #include "src/allocation.h" #include "src/handles.h" #include "src/vector.h" namespace v8 { namespace internal { // Forward declarations. class ByteArray; class StringAllocator { public: virtual ~StringAllocator() { } // Allocate a number of bytes. virtual char* allocate(unsigned bytes) = 0; // Allocate a larger number of bytes and copy the old buffer to the new one. // bytes is an input and output parameter passing the old size of the buffer // and returning the new size. If allocation fails then we return the old // buffer and do not increase the size. virtual char* grow(unsigned* bytes) = 0; }; // Normal allocator uses new[] and delete[]. class HeapStringAllocator final : public StringAllocator { public: ~HeapStringAllocator() { DeleteArray(space_); } char* allocate(unsigned bytes) override; char* grow(unsigned* bytes) override; private: char* space_; }; class FixedStringAllocator final : public StringAllocator { public: FixedStringAllocator(char* buffer, unsigned length) : buffer_(buffer), length_(length) {} ~FixedStringAllocator() override{}; char* allocate(unsigned bytes) override; char* grow(unsigned* bytes) override; private: char* buffer_; unsigned length_; DISALLOW_COPY_AND_ASSIGN(FixedStringAllocator); }; class StringStream final { class FmtElm final { public: FmtElm(int value) : FmtElm(INT) { // NOLINT data_.u_int_ = value; } explicit FmtElm(double value) : FmtElm(DOUBLE) { // NOLINT data_.u_double_ = value; } FmtElm(const char* value) : FmtElm(C_STR) { // NOLINT data_.u_c_str_ = value; } FmtElm(const Vector& value) : FmtElm(LC_STR) { // NOLINT data_.u_lc_str_ = &value; } FmtElm(Object* value) : FmtElm(OBJ) { // NOLINT data_.u_obj_ = value; } FmtElm(Handle value) : FmtElm(HANDLE) { // NOLINT data_.u_handle_ = value.location(); } FmtElm(void* value) : FmtElm(POINTER) { // NOLINT data_.u_pointer_ = value; } private: friend class StringStream; enum Type { INT, DOUBLE, C_STR, LC_STR, OBJ, HANDLE, POINTER }; #ifdef DEBUG Type type_; explicit FmtElm(Type type) : type_(type) {} #else explicit FmtElm(Type) {} #endif union { int u_int_; double u_double_; const char* u_c_str_; const Vector* u_lc_str_; Object* u_obj_; Object** u_handle_; void* u_pointer_; } data_; }; public: enum ObjectPrintMode { kPrintObjectConcise, kPrintObjectVerbose }; StringStream(StringAllocator* allocator, ObjectPrintMode object_print_mode = kPrintObjectVerbose) : allocator_(allocator), object_print_mode_(object_print_mode), capacity_(kInitialCapacity), length_(0), buffer_(allocator_->allocate(kInitialCapacity)) { buffer_[0] = 0; } bool Put(char c); bool Put(String* str); bool Put(String* str, int start, int end); void Add(const char* format) { Add(CStrVector(format)); } void Add(Vector format) { Add(format, Vector()); } template void Add(const char* format, Args... args) { Add(CStrVector(format), args...); } template void Add(Vector format, Args... args) { FmtElm elems[]{args...}; Add(format, ArrayVector(elems)); } // Getting the message out. void OutputToFile(FILE* out); void OutputToStdOut() { OutputToFile(stdout); } void Log(Isolate* isolate); Handle ToString(Isolate* isolate); std::unique_ptr ToCString() const; int length() const { return length_; } // Object printing support. void PrintName(Object* o); void PrintFixedArray(FixedArray* array, unsigned int limit); void PrintByteArray(ByteArray* ba); void PrintUsingMap(JSObject* js_object); void PrintPrototype(JSFunction* fun, Object* receiver); void PrintSecurityTokenIfChanged(JSFunction* function); // NOTE: Returns the code in the output parameter. void PrintFunction(JSFunction* function, Object* receiver, Code** code); // Reset the stream. void Reset() { length_ = 0; buffer_[0] = 0; } // Mentioned object cache support. void PrintMentionedObjectCache(Isolate* isolate); static void ClearMentionedObjectCache(Isolate* isolate); #ifdef DEBUG bool IsMentionedObjectCacheClear(Isolate* isolate); #endif static const int kInitialCapacity = 16; private: void Add(Vector format, Vector elms); void PrintObject(Object* obj); StringAllocator* allocator_; ObjectPrintMode object_print_mode_; unsigned capacity_; unsigned length_; // does not include terminating 0-character char* buffer_; bool full() const { return (capacity_ - length_) == 1; } int space() const { return capacity_ - length_; } DISALLOW_IMPLICIT_CONSTRUCTORS(StringStream); }; } // namespace internal } // namespace v8 #endif // V8_STRING_STREAM_H_