// Copyright 2011 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #ifndef V8_PREPARSE_DATA_H_ #define V8_PREPARSE_DATA_H_ #include "allocation.h" #include "hashmap.h" #include "utils-inl.h" namespace v8 { namespace internal { // ---------------------------------------------------------------------------- // ParserRecorder - Logging of preparser data. // Abstract interface for preparse data recorder. class ParserRecorder { public: ParserRecorder() { } virtual ~ParserRecorder() { } // Logs the scope and some details of a function literal in the source. virtual void LogFunction(int start, int end, int literals, int properties, StrictMode strict_mode) = 0; // Logs a symbol creation of a literal or identifier. virtual void LogAsciiSymbol(int start, Vector literal) { } virtual void LogUtf16Symbol(int start, Vector literal) { } // Logs an error message and marks the log as containing an error. // Further logging will be ignored, and ExtractData will return a vector // representing the error only. virtual void LogMessage(int start, int end, const char* message, const char* argument_opt) = 0; virtual int function_position() = 0; virtual int symbol_position() = 0; virtual int symbol_ids() = 0; virtual Vector ExtractData() = 0; virtual void PauseRecording() = 0; virtual void ResumeRecording() = 0; }; // ---------------------------------------------------------------------------- // FunctionLoggingParserRecorder - Record only function entries class FunctionLoggingParserRecorder : public ParserRecorder { public: FunctionLoggingParserRecorder(); virtual ~FunctionLoggingParserRecorder() {} virtual void LogFunction(int start, int end, int literals, int properties, StrictMode strict_mode) { function_store_.Add(start); function_store_.Add(end); function_store_.Add(literals); function_store_.Add(properties); function_store_.Add(strict_mode); } // Logs an error message and marks the log as containing an error. // Further logging will be ignored, and ExtractData will return a vector // representing the error only. virtual void LogMessage(int start, int end, const char* message, const char* argument_opt); virtual int function_position() { return function_store_.size(); } virtual Vector ExtractData() = 0; virtual void PauseRecording() { pause_count_++; is_recording_ = false; } virtual void ResumeRecording() { ASSERT(pause_count_ > 0); if (--pause_count_ == 0) is_recording_ = !has_error(); } protected: bool has_error() { return static_cast(preamble_[PreparseDataConstants::kHasErrorOffset]); } bool is_recording() { return is_recording_; } void WriteString(Vector str); Collector function_store_; unsigned preamble_[PreparseDataConstants::kHeaderSize]; bool is_recording_; int pause_count_; #ifdef DEBUG int prev_start_; #endif }; // ---------------------------------------------------------------------------- // PartialParserRecorder - Record only function entries class PartialParserRecorder : public FunctionLoggingParserRecorder { public: PartialParserRecorder() : FunctionLoggingParserRecorder() { } virtual void LogAsciiSymbol(int start, Vector literal) { } virtual void LogUtf16Symbol(int start, Vector literal) { } virtual ~PartialParserRecorder() { } virtual Vector ExtractData(); virtual int symbol_position() { return 0; } virtual int symbol_ids() { return 0; } }; // ---------------------------------------------------------------------------- // CompleteParserRecorder - Record both function entries and symbols. class CompleteParserRecorder: public FunctionLoggingParserRecorder { public: CompleteParserRecorder(); virtual ~CompleteParserRecorder() { } virtual void LogAsciiSymbol(int start, Vector literal) { if (!is_recording_) return; int hash = vector_hash(literal); LogSymbol(start, hash, true, Vector::cast(literal)); } virtual void LogUtf16Symbol(int start, Vector literal) { if (!is_recording_) return; int hash = vector_hash(literal); LogSymbol(start, hash, false, Vector::cast(literal)); } virtual Vector ExtractData(); virtual int symbol_position() { return symbol_store_.size(); } virtual int symbol_ids() { return symbol_id_; } private: // For testing. Defined in test-parsing.cc. friend void FakeWritingSymbolIdInPreParseData(CompleteParserRecorder* log, int number); struct Key { bool is_ascii; Vector literal_bytes; }; virtual void LogSymbol(int start, int hash, bool is_ascii, Vector literal); template static int vector_hash(Vector string) { int hash = 0; for (int i = 0; i < string.length(); i++) { int c = static_cast(string[i]); hash += c; hash += (hash << 10); hash ^= (hash >> 6); } return hash; } static bool vector_compare(void* a, void* b) { Key* string1 = reinterpret_cast(a); Key* string2 = reinterpret_cast(b); if (string1->is_ascii != string2->is_ascii) return false; int length = string1->literal_bytes.length(); if (string2->literal_bytes.length() != length) return false; return memcmp(string1->literal_bytes.start(), string2->literal_bytes.start(), length) == 0; } // Write a non-negative number to the symbol store. void WriteNumber(int number); Collector literal_chars_; Collector symbol_store_; Collector symbol_keys_; HashMap string_table_; int symbol_id_; }; } } // namespace v8::internal. #endif // V8_PREPARSE_DATA_H_