• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2014 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef V8_STRINGS_STRING_STREAM_H_
6 #define V8_STRINGS_STRING_STREAM_H_
7 
8 #include <memory>
9 
10 #include "src/base/small-vector.h"
11 #include "src/base/strings.h"
12 #include "src/base/vector.h"
13 #include "src/handles/handles.h"
14 #include "src/objects/heap-object.h"
15 #include "src/utils/allocation.h"
16 
17 namespace v8 {
18 namespace internal {
19 
20 // Forward declarations.
21 class ByteArray;
22 
23 class StringAllocator {
24  public:
25   virtual ~StringAllocator() = default;
26   // Allocate a number of bytes.
27   virtual char* allocate(unsigned bytes) = 0;
28   // Allocate a larger number of bytes and copy the old buffer to the new one.
29   // bytes is an input and output parameter passing the old size of the buffer
30   // and returning the new size.  If allocation fails then we return the old
31   // buffer and do not increase the size.
32   virtual char* grow(unsigned* bytes) = 0;
33 };
34 
35 // Normal allocator uses new[] and delete[].
36 class HeapStringAllocator final : public StringAllocator {
37  public:
~HeapStringAllocator()38   ~HeapStringAllocator() override { DeleteArray(space_); }
39   char* allocate(unsigned bytes) override;
40   char* grow(unsigned* bytes) override;
41 
42  private:
43   char* space_;
44 };
45 
46 class FixedStringAllocator final : public StringAllocator {
47  public:
FixedStringAllocator(char * buffer,unsigned length)48   FixedStringAllocator(char* buffer, unsigned length)
49       : buffer_(buffer), length_(length) {}
50   ~FixedStringAllocator() override = default;
51   FixedStringAllocator(const FixedStringAllocator&) = delete;
52   FixedStringAllocator& operator=(const FixedStringAllocator&) = delete;
53 
54   char* allocate(unsigned bytes) override;
55   char* grow(unsigned* bytes) override;
56 
57  private:
58   char* buffer_;
59   unsigned length_;
60 };
61 
62 template <std::size_t kInlineSize>
63 class SmallStringOptimizedAllocator final : public StringAllocator {
64  public:
65   using SmallVector = base::SmallVector<char, kInlineSize>;
66 
SmallStringOptimizedAllocator(SmallVector * vector)67   explicit SmallStringOptimizedAllocator(SmallVector* vector) V8_NOEXCEPT
68       : vector_(vector) {}
69 
allocate(unsigned bytes)70   char* allocate(unsigned bytes) override {
71     vector_->resize_no_init(bytes);
72     return vector_->data();
73   }
74 
grow(unsigned * bytes)75   char* grow(unsigned* bytes) override {
76     unsigned new_bytes = *bytes * 2;
77     // Check for overflow.
78     if (new_bytes <= *bytes) {
79       return vector_->data();
80     }
81     vector_->resize_no_init(new_bytes);
82     *bytes = new_bytes;
83     return vector_->data();
84   }
85 
86  private:
87   SmallVector* vector_;
88 };
89 
90 class StringStream final {
91   class FmtElm final {
92    public:
FmtElm(int value)93     FmtElm(int value) : FmtElm(INT) {  // NOLINT
94       data_.u_int_ = value;
95     }
FmtElm(double value)96     explicit FmtElm(double value) : FmtElm(DOUBLE) {  // NOLINT
97       data_.u_double_ = value;
98     }
FmtElm(const char * value)99     FmtElm(const char* value) : FmtElm(C_STR) {  // NOLINT
100       data_.u_c_str_ = value;
101     }
FmtElm(const base::Vector<const base::uc16> & value)102     FmtElm(const base::Vector<const base::uc16>& value)  // NOLINT
103         : FmtElm(LC_STR) {
104       data_.u_lc_str_ = &value;
105     }
FmtElm(Object value)106     FmtElm(Object value) : FmtElm(OBJ) {  // NOLINT
107       data_.u_obj_ = value.ptr();
108     }
FmtElm(Handle<Object> value)109     FmtElm(Handle<Object> value) : FmtElm(HANDLE) {  // NOLINT
110       data_.u_handle_ = value.location();
111     }
FmtElm(void * value)112     FmtElm(void* value) : FmtElm(POINTER) {  // NOLINT
113       data_.u_pointer_ = value;
114     }
115 
116    private:
117     friend class StringStream;
118     enum Type { INT, DOUBLE, C_STR, LC_STR, OBJ, HANDLE, POINTER };
119 
120 #ifdef DEBUG
121     Type type_;
FmtElm(Type type)122     explicit FmtElm(Type type) : type_(type) {}
123 #else
FmtElm(Type)124     explicit FmtElm(Type) {}
125 #endif
126 
127     union {
128       int u_int_;
129       double u_double_;
130       const char* u_c_str_;
131       const base::Vector<const base::uc16>* u_lc_str_;
132       Address u_obj_;
133       Address* u_handle_;
134       void* u_pointer_;
135     } data_;
136   };
137 
138  public:
139   enum ObjectPrintMode { kPrintObjectConcise, kPrintObjectVerbose };
140   explicit StringStream(StringAllocator* allocator,
141                         ObjectPrintMode object_print_mode = kPrintObjectVerbose)
allocator_(allocator)142       : allocator_(allocator),
143         object_print_mode_(object_print_mode),
144         capacity_(kInitialCapacity),
145         length_(0),
146         buffer_(allocator_->allocate(kInitialCapacity)) {
147     buffer_[0] = 0;
148   }
149 
150   bool Put(char c);
151   bool Put(String str);
152   bool Put(String str, int start, int end);
Add(const char * format)153   void Add(const char* format) { Add(base::CStrVector(format)); }
Add(base::Vector<const char> format)154   void Add(base::Vector<const char> format) {
155     Add(format, base::Vector<FmtElm>());
156   }
157 
158   template <typename... Args>
Add(const char * format,Args...args)159   void Add(const char* format, Args... args) {
160     Add(base::CStrVector(format), args...);
161   }
162 
163   template <typename... Args>
Add(base::Vector<const char> format,Args...args)164   void Add(base::Vector<const char> format, Args... args) {
165     FmtElm elems[]{args...};
166     Add(format, base::ArrayVector(elems));
167   }
168 
169   // Getting the message out.
170   void OutputToFile(FILE* out);
OutputToStdOut()171   void OutputToStdOut() { OutputToFile(stdout); }
172   void Log(Isolate* isolate);
173   Handle<String> ToString(Isolate* isolate);
174   std::unique_ptr<char[]> ToCString() const;
length()175   int length() const { return length_; }
176 
177   // Object printing support.
178   void PrintName(Object o);
179   void PrintFixedArray(FixedArray array, unsigned int limit);
180   void PrintByteArray(ByteArray ba);
181   void PrintUsingMap(JSObject js_object);
182   void PrintPrototype(JSFunction fun, Object receiver);
183   void PrintSecurityTokenIfChanged(JSFunction function);
184   // NOTE: Returns the code in the output parameter.
185   void PrintFunction(JSFunction function, Object receiver, Code* code);
186 
187   // Reset the stream.
Reset()188   void Reset() {
189     length_ = 0;
190     buffer_[0] = 0;
191   }
192 
193   // Mentioned object cache support.
194   void PrintMentionedObjectCache(Isolate* isolate);
195   V8_EXPORT_PRIVATE static void ClearMentionedObjectCache(Isolate* isolate);
196 #ifdef DEBUG
197   bool IsMentionedObjectCacheClear(Isolate* isolate);
198 #endif
199 
200   static const int kInitialCapacity = 16;
201 
202  private:
203   void Add(base::Vector<const char> format, base::Vector<FmtElm> elms);
204   void PrintObject(Object obj);
205 
206   StringAllocator* allocator_;
207   ObjectPrintMode object_print_mode_;
208   unsigned capacity_;
209   unsigned length_;  // does not include terminating 0-character
210   char* buffer_;
211 
full()212   bool full() const { return (capacity_ - length_) == 1; }
space()213   int space() const { return capacity_ - length_; }
214 
215   DISALLOW_IMPLICIT_CONSTRUCTORS(StringStream);
216 };
217 
218 }  // namespace internal
219 }  // namespace v8
220 
221 #endif  // V8_STRINGS_STRING_STREAM_H_
222