• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2011 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are
4 // met:
5 //
6 //     * Redistributions of source code must retain the above copyright
7 //       notice, this list of conditions and the following disclaimer.
8 //     * Redistributions in binary form must reproduce the above
9 //       copyright notice, this list of conditions and the following
10 //       disclaimer in the documentation and/or other materials provided
11 //       with the distribution.
12 //     * Neither the name of Google Inc. nor the names of its
13 //       contributors may be used to endorse or promote products derived
14 //       from this software without specific prior written permission.
15 //
16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 
28 #ifndef V8_PREPARSE_DATA_H_
29 #define V8_PREPARSE_DATA_H_
30 
31 #include "allocation.h"
32 #include "hashmap.h"
33 #include "utils-inl.h"
34 
35 namespace v8 {
36 namespace internal {
37 
38 // ----------------------------------------------------------------------------
39 // ParserRecorder - Logging of preparser data.
40 
41 // Abstract interface for preparse data recorder.
42 class ParserRecorder {
43  public:
ParserRecorder()44   ParserRecorder() { }
~ParserRecorder()45   virtual ~ParserRecorder() { }
46 
47   // Logs the scope and some details of a function literal in the source.
48   virtual void LogFunction(int start,
49                            int end,
50                            int literals,
51                            int properties,
52                            LanguageMode language_mode) = 0;
53 
54   // Logs a symbol creation of a literal or identifier.
LogAsciiSymbol(int start,Vector<const char> literal)55   virtual void LogAsciiSymbol(int start, Vector<const char> literal) { }
LogUtf16Symbol(int start,Vector<const uc16> literal)56   virtual void LogUtf16Symbol(int start, Vector<const uc16> literal) { }
57 
58   // Logs an error message and marks the log as containing an error.
59   // Further logging will be ignored, and ExtractData will return a vector
60   // representing the error only.
61   virtual void LogMessage(int start,
62                           int end,
63                           const char* message,
64                           const char* argument_opt) = 0;
65 
66   virtual int function_position() = 0;
67 
68   virtual int symbol_position() = 0;
69 
70   virtual int symbol_ids() = 0;
71 
72   virtual Vector<unsigned> ExtractData() = 0;
73 
74   virtual void PauseRecording() = 0;
75 
76   virtual void ResumeRecording() = 0;
77 };
78 
79 
80 // ----------------------------------------------------------------------------
81 // FunctionLoggingParserRecorder - Record only function entries
82 
83 class FunctionLoggingParserRecorder : public ParserRecorder {
84  public:
85   FunctionLoggingParserRecorder();
~FunctionLoggingParserRecorder()86   virtual ~FunctionLoggingParserRecorder() {}
87 
LogFunction(int start,int end,int literals,int properties,LanguageMode language_mode)88   virtual void LogFunction(int start,
89                            int end,
90                            int literals,
91                            int properties,
92                            LanguageMode language_mode) {
93     function_store_.Add(start);
94     function_store_.Add(end);
95     function_store_.Add(literals);
96     function_store_.Add(properties);
97     function_store_.Add(language_mode);
98   }
99 
100   // Logs an error message and marks the log as containing an error.
101   // Further logging will be ignored, and ExtractData will return a vector
102   // representing the error only.
103   virtual void LogMessage(int start,
104                           int end,
105                           const char* message,
106                           const char* argument_opt);
107 
function_position()108   virtual int function_position() { return function_store_.size(); }
109 
110 
111   virtual Vector<unsigned> ExtractData() = 0;
112 
PauseRecording()113   virtual void PauseRecording() {
114     pause_count_++;
115     is_recording_ = false;
116   }
117 
ResumeRecording()118   virtual void ResumeRecording() {
119     ASSERT(pause_count_ > 0);
120     if (--pause_count_ == 0) is_recording_ = !has_error();
121   }
122 
123  protected:
has_error()124   bool has_error() {
125     return static_cast<bool>(preamble_[PreparseDataConstants::kHasErrorOffset]);
126   }
127 
is_recording()128   bool is_recording() {
129     return is_recording_;
130   }
131 
132   void WriteString(Vector<const char> str);
133 
134   Collector<unsigned> function_store_;
135   unsigned preamble_[PreparseDataConstants::kHeaderSize];
136   bool is_recording_;
137   int pause_count_;
138 
139 #ifdef DEBUG
140   int prev_start_;
141 #endif
142 };
143 
144 
145 // ----------------------------------------------------------------------------
146 // PartialParserRecorder - Record only function entries
147 
148 class PartialParserRecorder : public FunctionLoggingParserRecorder {
149  public:
PartialParserRecorder()150   PartialParserRecorder() : FunctionLoggingParserRecorder() { }
LogAsciiSymbol(int start,Vector<const char> literal)151   virtual void LogAsciiSymbol(int start, Vector<const char> literal) { }
LogUtf16Symbol(int start,Vector<const uc16> literal)152   virtual void LogUtf16Symbol(int start, Vector<const uc16> literal) { }
~PartialParserRecorder()153   virtual ~PartialParserRecorder() { }
154   virtual Vector<unsigned> ExtractData();
symbol_position()155   virtual int symbol_position() { return 0; }
symbol_ids()156   virtual int symbol_ids() { return 0; }
157 };
158 
159 
160 // ----------------------------------------------------------------------------
161 // CompleteParserRecorder -  Record both function entries and symbols.
162 
163 class CompleteParserRecorder: public FunctionLoggingParserRecorder {
164  public:
165   CompleteParserRecorder();
~CompleteParserRecorder()166   virtual ~CompleteParserRecorder() { }
167 
LogAsciiSymbol(int start,Vector<const char> literal)168   virtual void LogAsciiSymbol(int start, Vector<const char> literal) {
169     if (!is_recording_) return;
170     int hash = vector_hash(literal);
171     LogSymbol(start, hash, true, Vector<const byte>::cast(literal));
172   }
173 
LogUtf16Symbol(int start,Vector<const uc16> literal)174   virtual void LogUtf16Symbol(int start, Vector<const uc16> literal) {
175     if (!is_recording_) return;
176     int hash = vector_hash(literal);
177     LogSymbol(start, hash, false, Vector<const byte>::cast(literal));
178   }
179 
180   virtual Vector<unsigned> ExtractData();
181 
symbol_position()182   virtual int symbol_position() { return symbol_store_.size(); }
symbol_ids()183   virtual int symbol_ids() { return symbol_id_; }
184 
185  private:
186   struct Key {
187     bool is_ascii;
188     Vector<const byte> literal_bytes;
189   };
190 
191   virtual void LogSymbol(int start,
192                          int hash,
193                          bool is_ascii,
194                          Vector<const byte> literal);
195 
196   template <typename Char>
vector_hash(Vector<const Char> string)197   static int vector_hash(Vector<const Char> string) {
198     int hash = 0;
199     for (int i = 0; i < string.length(); i++) {
200       int c = static_cast<int>(string[i]);
201       hash += c;
202       hash += (hash << 10);
203       hash ^= (hash >> 6);
204     }
205     return hash;
206   }
207 
vector_compare(void * a,void * b)208   static bool vector_compare(void* a, void* b) {
209     Key* string1 = reinterpret_cast<Key*>(a);
210     Key* string2 = reinterpret_cast<Key*>(b);
211     if (string1->is_ascii != string2->is_ascii) return false;
212     int length = string1->literal_bytes.length();
213     if (string2->literal_bytes.length() != length) return false;
214     return memcmp(string1->literal_bytes.start(),
215                   string2->literal_bytes.start(), length) == 0;
216   }
217 
218   // Write a non-negative number to the symbol store.
219   void WriteNumber(int number);
220 
221   Collector<byte> literal_chars_;
222   Collector<byte> symbol_store_;
223   Collector<Key> symbol_keys_;
224   HashMap symbol_table_;
225   int symbol_id_;
226 };
227 
228 
229 } }  // namespace v8::internal.
230 
231 #endif  // V8_PREPARSE_DATA_H_
232