2012-01-26 21:47:57 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2008-07-03 15:10:15 +00:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
#include "v8.h"
|
|
|
|
|
|
|
|
#include "api.h"
|
2011-11-09 11:32:54 +00:00
|
|
|
#include "ast.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "bootstrapper.h"
|
2011-09-05 07:39:47 +00:00
|
|
|
#include "char-predicates-inl.h"
|
2010-03-11 09:27:12 +00:00
|
|
|
#include "codegen.h"
|
2009-05-14 13:17:28 +00:00
|
|
|
#include "compiler.h"
|
2010-08-23 13:26:03 +00:00
|
|
|
#include "func-name-inferrer.h"
|
2010-03-09 06:38:33 +00:00
|
|
|
#include "messages.h"
|
2010-05-10 11:32:25 +00:00
|
|
|
#include "parser.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "platform.h"
|
2010-11-02 07:21:37 +00:00
|
|
|
#include "preparser.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "runtime.h"
|
2011-09-08 13:06:44 +00:00
|
|
|
#include "scanner-character-streams.h"
|
2010-07-14 11:18:09 +00:00
|
|
|
#include "scopeinfo.h"
|
2008-11-25 11:07:48 +00:00
|
|
|
#include "string-stream.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
// PositionStack is used for on-stack allocation of token positions for
|
|
|
|
// new expressions. Please look at ParseNewExpression.
|
|
|
|
|
|
|
|
class PositionStack {
|
|
|
|
public:
|
|
|
|
explicit PositionStack(bool* ok) : top_(NULL), ok_(ok) {}
|
2013-01-29 15:28:05 +00:00
|
|
|
~PositionStack() {
|
|
|
|
ASSERT(!*ok_ || is_empty());
|
|
|
|
USE(ok_);
|
|
|
|
}
|
2009-05-15 14:58:02 +00:00
|
|
|
|
|
|
|
class Element {
|
|
|
|
public:
|
|
|
|
Element(PositionStack* stack, int value) {
|
|
|
|
previous_ = stack->top();
|
|
|
|
value_ = value;
|
|
|
|
stack->set_top(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Element* previous() { return previous_; }
|
|
|
|
int value() { return value_; }
|
|
|
|
friend class PositionStack;
|
|
|
|
Element* previous_;
|
|
|
|
int value_;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool is_empty() { return top_ == NULL; }
|
|
|
|
int pop() {
|
|
|
|
ASSERT(!is_empty());
|
|
|
|
int result = top_->value();
|
|
|
|
top_ = top_->previous();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Element* top() { return top_; }
|
|
|
|
void set_top(Element* value) { top_ = value; }
|
|
|
|
Element* top_;
|
|
|
|
bool* ok_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-06-04 14:42:58 +00:00
|
|
|
RegExpBuilder::RegExpBuilder(Zone* zone)
|
|
|
|
: zone_(zone),
|
2011-04-04 06:29:02 +00:00
|
|
|
pending_empty_(false),
|
|
|
|
characters_(NULL),
|
|
|
|
terms_(),
|
|
|
|
alternatives_()
|
2008-11-25 11:07:48 +00:00
|
|
|
#ifdef DEBUG
|
2011-04-04 06:29:02 +00:00
|
|
|
, last_added_(ADD_NONE)
|
2008-11-25 11:07:48 +00:00
|
|
|
#endif
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::FlushCharacters() {
|
|
|
|
pending_empty_ = false;
|
|
|
|
if (characters_ != NULL) {
|
2011-04-04 06:29:02 +00:00
|
|
|
RegExpTree* atom = new(zone()) RegExpAtom(characters_->ToConstVector());
|
2008-11-25 11:07:48 +00:00
|
|
|
characters_ = NULL;
|
2012-06-11 12:42:31 +00:00
|
|
|
text_.Add(atom, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
LAST(ADD_ATOM);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::FlushText() {
|
|
|
|
FlushCharacters();
|
|
|
|
int num_text = text_.length();
|
|
|
|
if (num_text == 0) {
|
|
|
|
return;
|
|
|
|
} else if (num_text == 1) {
|
2012-06-11 12:42:31 +00:00
|
|
|
terms_.Add(text_.last(), zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
} else {
|
2012-06-11 12:42:31 +00:00
|
|
|
RegExpText* text = new(zone()) RegExpText(zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
for (int i = 0; i < num_text; i++)
|
2012-06-11 12:42:31 +00:00
|
|
|
text_.Get(i)->AppendToText(text, zone());
|
|
|
|
terms_.Add(text, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
text_.Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::AddCharacter(uc16 c) {
|
|
|
|
pending_empty_ = false;
|
|
|
|
if (characters_ == NULL) {
|
2012-06-11 12:42:31 +00:00
|
|
|
characters_ = new(zone()) ZoneList<uc16>(4, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
characters_->Add(c, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
LAST(ADD_CHAR);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::AddEmpty() {
|
|
|
|
pending_empty_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::AddAtom(RegExpTree* term) {
|
|
|
|
if (term->IsEmpty()) {
|
|
|
|
AddEmpty();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (term->IsTextElement()) {
|
|
|
|
FlushCharacters();
|
2012-06-11 12:42:31 +00:00
|
|
|
text_.Add(term, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
} else {
|
|
|
|
FlushText();
|
2012-06-11 12:42:31 +00:00
|
|
|
terms_.Add(term, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
LAST(ADD_ATOM);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::AddAssertion(RegExpTree* assert) {
|
|
|
|
FlushText();
|
2012-06-11 12:42:31 +00:00
|
|
|
terms_.Add(assert, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
LAST(ADD_ASSERT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::NewAlternative() {
|
|
|
|
FlushTerms();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpBuilder::FlushTerms() {
|
|
|
|
FlushText();
|
|
|
|
int num_terms = terms_.length();
|
|
|
|
RegExpTree* alternative;
|
|
|
|
if (num_terms == 0) {
|
|
|
|
alternative = RegExpEmpty::GetInstance();
|
|
|
|
} else if (num_terms == 1) {
|
|
|
|
alternative = terms_.last();
|
|
|
|
} else {
|
2012-06-11 12:42:31 +00:00
|
|
|
alternative = new(zone()) RegExpAlternative(terms_.GetList(zone()));
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
alternatives_.Add(alternative, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
terms_.Clear();
|
|
|
|
LAST(ADD_NONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
RegExpTree* RegExpBuilder::ToRegExp() {
|
|
|
|
FlushTerms();
|
|
|
|
int num_alternatives = alternatives_.length();
|
|
|
|
if (num_alternatives == 0) {
|
|
|
|
return RegExpEmpty::GetInstance();
|
|
|
|
}
|
|
|
|
if (num_alternatives == 1) {
|
|
|
|
return alternatives_.last();
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
return new(zone()) RegExpDisjunction(alternatives_.GetList(zone()));
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
void RegExpBuilder::AddQuantifierToAtom(
|
|
|
|
int min, int max, RegExpQuantifier::QuantifierType quantifier_type) {
|
2008-11-25 11:07:48 +00:00
|
|
|
if (pending_empty_) {
|
|
|
|
pending_empty_ = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
RegExpTree* atom;
|
|
|
|
if (characters_ != NULL) {
|
|
|
|
ASSERT(last_added_ == ADD_CHAR);
|
|
|
|
// Last atom was character.
|
|
|
|
Vector<const uc16> char_vector = characters_->ToConstVector();
|
|
|
|
int num_chars = char_vector.length();
|
|
|
|
if (num_chars > 1) {
|
|
|
|
Vector<const uc16> prefix = char_vector.SubVector(0, num_chars - 1);
|
2012-06-11 12:42:31 +00:00
|
|
|
text_.Add(new(zone()) RegExpAtom(prefix), zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
char_vector = char_vector.SubVector(num_chars - 1, num_chars);
|
|
|
|
}
|
|
|
|
characters_ = NULL;
|
2011-04-04 06:29:02 +00:00
|
|
|
atom = new(zone()) RegExpAtom(char_vector);
|
2008-11-25 11:07:48 +00:00
|
|
|
FlushText();
|
|
|
|
} else if (text_.length() > 0) {
|
|
|
|
ASSERT(last_added_ == ADD_ATOM);
|
|
|
|
atom = text_.RemoveLast();
|
|
|
|
FlushText();
|
|
|
|
} else if (terms_.length() > 0) {
|
|
|
|
ASSERT(last_added_ == ADD_ATOM);
|
|
|
|
atom = terms_.RemoveLast();
|
2008-12-17 10:59:14 +00:00
|
|
|
if (atom->max_match() == 0) {
|
|
|
|
// Guaranteed to only match an empty string.
|
2008-11-25 11:07:48 +00:00
|
|
|
LAST(ADD_TERM);
|
|
|
|
if (min == 0) {
|
|
|
|
return;
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
terms_.Add(atom, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Only call immediately after adding an atom or character!
|
|
|
|
UNREACHABLE();
|
|
|
|
return;
|
|
|
|
}
|
2013-06-06 13:28:22 +00:00
|
|
|
terms_.Add(
|
|
|
|
new(zone()) RegExpQuantifier(min, max, quantifier_type, atom), zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
LAST(ADD_TERM);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-22 20:14:19 +00:00
|
|
|
Handle<String> Parser::LookupSymbol(int symbol_id) {
|
2010-11-02 11:45:47 +00:00
|
|
|
// Length of symbol cache is the number of identified symbols.
|
|
|
|
// If we are larger than that, or negative, it's not a cached symbol.
|
|
|
|
// This might also happen if there is no preparser symbol data, even
|
|
|
|
// if there is some preparser data.
|
|
|
|
if (static_cast<unsigned>(symbol_id)
|
|
|
|
>= static_cast<unsigned>(symbol_cache_.length())) {
|
2010-12-22 20:14:19 +00:00
|
|
|
if (scanner().is_literal_ascii()) {
|
2013-02-28 17:03:34 +00:00
|
|
|
return isolate()->factory()->InternalizeOneByteString(
|
2013-01-09 10:30:54 +00:00
|
|
|
Vector<const uint8_t>::cast(scanner().literal_ascii_string()));
|
2010-12-22 20:14:19 +00:00
|
|
|
} else {
|
2013-02-28 17:03:34 +00:00
|
|
|
return isolate()->factory()->InternalizeTwoByteString(
|
2012-03-12 12:35:28 +00:00
|
|
|
scanner().literal_utf16_string());
|
2010-12-22 20:14:19 +00:00
|
|
|
}
|
2010-11-02 11:45:47 +00:00
|
|
|
}
|
2010-12-22 20:14:19 +00:00
|
|
|
return LookupCachedSymbol(symbol_id);
|
2010-11-02 11:45:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-22 20:14:19 +00:00
|
|
|
Handle<String> Parser::LookupCachedSymbol(int symbol_id) {
|
2010-11-02 11:45:47 +00:00
|
|
|
// Make sure the cache is large enough to hold the symbol identifier.
|
|
|
|
if (symbol_cache_.length() <= symbol_id) {
|
|
|
|
// Increase length to index + 1.
|
|
|
|
symbol_cache_.AddBlock(Handle<String>::null(),
|
2012-06-11 12:42:31 +00:00
|
|
|
symbol_id + 1 - symbol_cache_.length(), zone());
|
2010-11-02 11:45:47 +00:00
|
|
|
}
|
|
|
|
Handle<String> result = symbol_cache_.at(symbol_id);
|
|
|
|
if (result.is_null()) {
|
2010-12-22 20:14:19 +00:00
|
|
|
if (scanner().is_literal_ascii()) {
|
2013-02-28 17:03:34 +00:00
|
|
|
result = isolate()->factory()->InternalizeOneByteString(
|
2013-01-09 10:30:54 +00:00
|
|
|
Vector<const uint8_t>::cast(scanner().literal_ascii_string()));
|
2010-12-22 20:14:19 +00:00
|
|
|
} else {
|
2013-02-28 17:03:34 +00:00
|
|
|
result = isolate()->factory()->InternalizeTwoByteString(
|
2012-03-12 12:35:28 +00:00
|
|
|
scanner().literal_utf16_string());
|
2010-12-22 20:14:19 +00:00
|
|
|
}
|
2010-11-02 11:45:47 +00:00
|
|
|
symbol_cache_.at(symbol_id) = result;
|
2010-09-07 12:52:16 +00:00
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-03-23 11:13:07 +00:00
|
|
|
isolate()->counters()->total_preparse_symbols_skipped()->Increment();
|
2010-11-02 11:45:47 +00:00
|
|
|
return result;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2010-08-27 08:26:29 +00:00
|
|
|
FunctionEntry ScriptDataImpl::GetFunctionEntry(int start) {
|
|
|
|
// The current pre-data entry must be a FunctionEntry with the given
|
|
|
|
// start position.
|
2010-09-07 12:52:16 +00:00
|
|
|
if ((function_index_ + FunctionEntry::kSize <= store_.length())
|
|
|
|
&& (static_cast<int>(store_[function_index_]) == start)) {
|
|
|
|
int index = function_index_;
|
|
|
|
function_index_ += FunctionEntry::kSize;
|
2010-08-27 08:26:29 +00:00
|
|
|
return FunctionEntry(store_.SubVector(index,
|
|
|
|
index + FunctionEntry::kSize));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
return FunctionEntry();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-15 10:54:35 +00:00
|
|
|
int ScriptDataImpl::GetSymbolIdentifier() {
|
|
|
|
return ReadNumber(&symbol_data_);
|
2010-09-07 12:52:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ScriptDataImpl::SanityCheck() {
|
|
|
|
// Check that the header data is valid and doesn't specify
|
|
|
|
// point to positions outside the store.
|
2010-11-23 11:46:36 +00:00
|
|
|
if (store_.length() < PreparseDataConstants::kHeaderSize) return false;
|
|
|
|
if (magic() != PreparseDataConstants::kMagicNumber) return false;
|
|
|
|
if (version() != PreparseDataConstants::kCurrentVersion) return false;
|
2010-09-07 12:52:16 +00:00
|
|
|
if (has_error()) {
|
|
|
|
// Extra sane sanity check for error message encoding.
|
2010-11-23 11:46:36 +00:00
|
|
|
if (store_.length() <= PreparseDataConstants::kHeaderSize
|
|
|
|
+ PreparseDataConstants::kMessageTextPos) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Read(PreparseDataConstants::kMessageStartPos) >
|
|
|
|
Read(PreparseDataConstants::kMessageEndPos)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
unsigned arg_count = Read(PreparseDataConstants::kMessageArgCountPos);
|
|
|
|
int pos = PreparseDataConstants::kMessageTextPos;
|
2010-09-07 12:52:16 +00:00
|
|
|
for (unsigned int i = 0; i <= arg_count; i++) {
|
2010-11-23 11:46:36 +00:00
|
|
|
if (store_.length() <= PreparseDataConstants::kHeaderSize + pos) {
|
|
|
|
return false;
|
|
|
|
}
|
2010-09-07 12:52:16 +00:00
|
|
|
int length = static_cast<int>(Read(pos));
|
|
|
|
if (length < 0) return false;
|
|
|
|
pos += 1 + length;
|
|
|
|
}
|
2010-11-23 11:46:36 +00:00
|
|
|
if (store_.length() < PreparseDataConstants::kHeaderSize + pos) {
|
|
|
|
return false;
|
|
|
|
}
|
2010-09-07 12:52:16 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Check that the space allocated for function entries is sane.
|
|
|
|
int functions_size =
|
2010-11-23 11:46:36 +00:00
|
|
|
static_cast<int>(store_[PreparseDataConstants::kFunctionsSizeOffset]);
|
2010-09-07 12:52:16 +00:00
|
|
|
if (functions_size < 0) return false;
|
|
|
|
if (functions_size % FunctionEntry::kSize != 0) return false;
|
|
|
|
// Check that the count of symbols is non-negative.
|
|
|
|
int symbol_count =
|
2010-11-23 11:46:36 +00:00
|
|
|
static_cast<int>(store_[PreparseDataConstants::kSymbolCountOffset]);
|
2010-09-07 12:52:16 +00:00
|
|
|
if (symbol_count < 0) return false;
|
2010-09-15 10:54:35 +00:00
|
|
|
// Check that the total size has room for header and function entries.
|
2010-09-07 12:52:16 +00:00
|
|
|
int minimum_size =
|
2010-11-23 11:46:36 +00:00
|
|
|
PreparseDataConstants::kHeaderSize + functions_size;
|
2010-09-07 12:52:16 +00:00
|
|
|
if (store_.length() < minimum_size) return false;
|
2008-07-03 15:10:15 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-09 11:49:21 +00:00
|
|
|
|
2010-08-25 06:46:53 +00:00
|
|
|
const char* ScriptDataImpl::ReadString(unsigned* start, int* chars) {
|
2008-07-03 15:10:15 +00:00
|
|
|
int length = start[0];
|
|
|
|
char* result = NewArray<char>(length + 1);
|
2010-08-25 06:46:53 +00:00
|
|
|
for (int i = 0; i < length; i++) {
|
2008-07-03 15:10:15 +00:00
|
|
|
result[i] = start[i + 1];
|
2010-08-25 06:46:53 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
result[length] = '\0';
|
|
|
|
if (chars != NULL) *chars = length;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2013-07-05 09:52:11 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Scanner::Location ScriptDataImpl::MessageLocation() {
|
2010-11-23 11:46:36 +00:00
|
|
|
int beg_pos = Read(PreparseDataConstants::kMessageStartPos);
|
|
|
|
int end_pos = Read(PreparseDataConstants::kMessageEndPos);
|
2008-07-03 15:10:15 +00:00
|
|
|
return Scanner::Location(beg_pos, end_pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const char* ScriptDataImpl::BuildMessage() {
|
2010-11-23 11:46:36 +00:00
|
|
|
unsigned* start = ReadAddress(PreparseDataConstants::kMessageTextPos);
|
2010-08-25 06:46:53 +00:00
|
|
|
return ReadString(start, NULL);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Vector<const char*> ScriptDataImpl::BuildArgs() {
|
2010-11-23 11:46:36 +00:00
|
|
|
int arg_count = Read(PreparseDataConstants::kMessageArgCountPos);
|
2008-07-03 15:10:15 +00:00
|
|
|
const char** array = NewArray<const char*>(arg_count);
|
2010-09-15 10:54:35 +00:00
|
|
|
// Position after text found by skipping past length field and
|
|
|
|
// length field content words.
|
2010-11-23 11:46:36 +00:00
|
|
|
int pos = PreparseDataConstants::kMessageTextPos + 1
|
|
|
|
+ Read(PreparseDataConstants::kMessageTextPos);
|
2008-07-03 15:10:15 +00:00
|
|
|
for (int i = 0; i < arg_count; i++) {
|
|
|
|
int count = 0;
|
2010-08-25 06:46:53 +00:00
|
|
|
array[i] = ReadString(ReadAddress(pos), &count);
|
2008-07-03 15:10:15 +00:00
|
|
|
pos += count + 1;
|
|
|
|
}
|
|
|
|
return Vector<const char*>(array, arg_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned ScriptDataImpl::Read(int position) {
|
2010-11-23 11:46:36 +00:00
|
|
|
return store_[PreparseDataConstants::kHeaderSize + position];
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned* ScriptDataImpl::ReadAddress(int position) {
|
2010-11-23 11:46:36 +00:00
|
|
|
return &store_[PreparseDataConstants::kHeaderSize + position];
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
Scope* Parser::NewScope(Scope* parent, ScopeType scope_type) {
|
|
|
|
Scope* result = new(zone()) Scope(parent, scope_type, zone());
|
2011-10-17 09:29:37 +00:00
|
|
|
result->Initialize();
|
2008-07-03 15:10:15 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-06-30 14:37:55 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Target is a support class to facilitate manipulation of the
|
|
|
|
// Parser's target_stack_ (the stack of potential 'break' and
|
|
|
|
// 'continue' statement targets). Upon construction, a new target is
|
|
|
|
// added; it is removed upon destruction.
|
|
|
|
|
|
|
|
class Target BASE_EMBEDDED {
|
|
|
|
public:
|
2010-10-27 12:33:48 +00:00
|
|
|
Target(Target** variable, AstNode* node)
|
|
|
|
: variable_(variable), node_(node), previous_(*variable) {
|
|
|
|
*variable = this;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
~Target() {
|
2010-10-27 12:33:48 +00:00
|
|
|
*variable_ = previous_;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
Target* previous() { return previous_; }
|
2009-07-30 11:53:29 +00:00
|
|
|
AstNode* node() { return node_; }
|
2009-05-15 14:58:02 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
2010-10-27 12:33:48 +00:00
|
|
|
Target** variable_;
|
2009-07-30 11:53:29 +00:00
|
|
|
AstNode* node_;
|
2009-05-15 14:58:02 +00:00
|
|
|
Target* previous_;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class TargetScope BASE_EMBEDDED {
|
|
|
|
public:
|
2010-10-27 12:33:48 +00:00
|
|
|
explicit TargetScope(Target** variable)
|
|
|
|
: variable_(variable), previous_(*variable) {
|
|
|
|
*variable = NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
~TargetScope() {
|
2010-10-27 12:33:48 +00:00
|
|
|
*variable_ = previous_;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2010-10-27 12:33:48 +00:00
|
|
|
Target** variable_;
|
2009-05-15 14:58:02 +00:00
|
|
|
Target* previous_;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2011-11-09 13:54:26 +00:00
|
|
|
// FunctionState and BlockState together implement the parser's scope stack.
|
|
|
|
// The parser's current scope is in top_scope_. The BlockState and
|
|
|
|
// FunctionState constructors push on the scope stack and the destructors
|
|
|
|
// pop. They are also used to hold the parser's per-function and per-block
|
|
|
|
// state.
|
2011-10-17 09:29:37 +00:00
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
class Parser::BlockState BASE_EMBEDDED {
|
2011-10-17 09:29:37 +00:00
|
|
|
public:
|
2011-11-09 13:54:26 +00:00
|
|
|
BlockState(Parser* parser, Scope* scope)
|
2011-10-17 09:29:37 +00:00
|
|
|
: parser_(parser),
|
2011-11-09 13:54:26 +00:00
|
|
|
outer_scope_(parser->top_scope_) {
|
2011-10-17 09:29:37 +00:00
|
|
|
parser->top_scope_ = scope;
|
|
|
|
}
|
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
~BlockState() { parser_->top_scope_ = outer_scope_; }
|
2011-10-17 09:29:37 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
Parser* parser_;
|
2011-11-09 13:54:26 +00:00
|
|
|
Scope* outer_scope_;
|
2011-10-17 09:29:37 +00:00
|
|
|
};
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
Parser::FunctionState::FunctionState(Parser* parser,
|
|
|
|
Scope* scope,
|
|
|
|
Isolate* isolate)
|
|
|
|
: next_materialized_literal_index_(JSFunction::kLiteralsPrefixSize),
|
2011-11-11 13:48:14 +00:00
|
|
|
next_handler_index_(0),
|
2011-11-09 13:54:26 +00:00
|
|
|
expected_property_count_(0),
|
2013-04-15 12:29:44 +00:00
|
|
|
generator_object_variable_(NULL),
|
2011-11-09 13:54:26 +00:00
|
|
|
parser_(parser),
|
|
|
|
outer_function_state_(parser->current_function_state_),
|
|
|
|
outer_scope_(parser->top_scope_),
|
2012-02-08 09:56:33 +00:00
|
|
|
saved_ast_node_id_(isolate->ast_node_id()),
|
2012-06-20 08:58:41 +00:00
|
|
|
factory_(isolate, parser->zone()) {
|
2011-03-22 18:00:03 +00:00
|
|
|
parser->top_scope_ = scope;
|
2011-11-09 13:54:26 +00:00
|
|
|
parser->current_function_state_ = this;
|
2012-08-06 14:13:09 +00:00
|
|
|
isolate->set_ast_node_id(BailoutId::FirstUsable().ToInt());
|
2011-03-22 18:00:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
Parser::FunctionState::~FunctionState() {
|
|
|
|
parser_->top_scope_ = outer_scope_;
|
|
|
|
parser_->current_function_state_ = outer_function_state_;
|
2012-02-14 14:14:51 +00:00
|
|
|
if (outer_function_state_ != NULL) {
|
|
|
|
parser_->isolate()->set_ast_node_id(saved_ast_node_id_);
|
|
|
|
}
|
2011-03-22 18:00:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// The CHECK_OK macro is a convenient macro to enforce error
|
|
|
|
// handling for functions that may fail (by returning !*ok).
|
|
|
|
//
|
|
|
|
// CAUTION: This macro appends extra statements after a call,
|
|
|
|
// thus it must never be used where only a single statement
|
|
|
|
// is correct (e.g. an if statement branch w/o braces)!
|
|
|
|
|
|
|
|
#define CHECK_OK ok); \
|
|
|
|
if (!*ok) return NULL; \
|
|
|
|
((void)0
|
|
|
|
#define DUMMY ) // to make indentation work
|
|
|
|
#undef DUMMY
|
|
|
|
|
2008-12-01 15:42:35 +00:00
|
|
|
#define CHECK_FAILED /**/); \
|
2008-12-01 15:32:20 +00:00
|
|
|
if (failed_) return NULL; \
|
|
|
|
((void)0
|
|
|
|
#define DUMMY ) // to make indentation work
|
|
|
|
#undef DUMMY
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Implementation of Parser
|
|
|
|
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
Parser::Parser(CompilationInfo* info)
|
2012-06-20 10:56:53 +00:00
|
|
|
: isolate_(info->isolate()),
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
symbol_cache_(0, info->zone()),
|
2012-06-20 10:56:53 +00:00
|
|
|
script_(info->script()),
|
2011-04-12 08:27:38 +00:00
|
|
|
scanner_(isolate_->unicode_cache()),
|
2011-11-25 09:36:31 +00:00
|
|
|
reusable_preparser_(NULL),
|
2008-07-03 15:10:15 +00:00
|
|
|
top_scope_(NULL),
|
2013-08-23 09:25:37 +00:00
|
|
|
original_scope_(NULL),
|
2011-11-09 13:54:26 +00:00
|
|
|
current_function_state_(NULL),
|
2008-07-03 15:10:15 +00:00
|
|
|
target_stack_(NULL),
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
extension_(info->extension()),
|
|
|
|
pre_parse_data_(NULL),
|
2010-11-29 13:24:37 +00:00
|
|
|
fni_(NULL),
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
allow_natives_syntax_(false),
|
|
|
|
allow_lazy_(false),
|
|
|
|
allow_generators_(false),
|
2013-06-06 14:38:26 +00:00
|
|
|
allow_for_of_(false),
|
2011-01-14 10:50:13 +00:00
|
|
|
stack_overflow_(false),
|
2012-06-04 14:42:58 +00:00
|
|
|
parenthesized_function_(false),
|
2012-06-20 10:56:53 +00:00
|
|
|
zone_(info->zone()),
|
|
|
|
info_(info) {
|
|
|
|
ASSERT(!script_.is_null());
|
2012-02-14 14:14:51 +00:00
|
|
|
isolate_->set_ast_node_id(0);
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
set_allow_harmony_scoping(!info->is_native() && FLAG_harmony_scoping);
|
|
|
|
set_allow_modules(!info->is_native() && FLAG_harmony_modules);
|
|
|
|
set_allow_natives_syntax(FLAG_allow_natives_syntax || info->is_native());
|
|
|
|
set_allow_lazy(false); // Must be explicitly enabled.
|
|
|
|
set_allow_generators(FLAG_harmony_generators);
|
2013-06-06 14:38:26 +00:00
|
|
|
set_allow_for_of(FLAG_harmony_iteration);
|
2013-07-19 09:57:35 +00:00
|
|
|
set_allow_harmony_numeric_literals(FLAG_harmony_numeric_literals);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-20 10:56:53 +00:00
|
|
|
FunctionLiteral* Parser::ParseProgram() {
|
2013-08-29 09:15:13 +00:00
|
|
|
HistogramTimerScope timer_scope(isolate()->counters()->parse());
|
2011-11-15 13:48:40 +00:00
|
|
|
Handle<String> source(String::cast(script_->source()));
|
2011-03-23 11:13:07 +00:00
|
|
|
isolate()->counters()->total_parse_size()->Increment(source->length());
|
2013-08-29 09:15:13 +00:00
|
|
|
ElapsedTimer timer;
|
|
|
|
if (FLAG_trace_parse) {
|
|
|
|
timer.Start();
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
fni_ = new(zone()) FuncNameInferrer(isolate(), zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Initialize parser state.
|
2010-02-26 14:37:33 +00:00
|
|
|
source->TryFlatten();
|
2012-07-18 11:22:46 +00:00
|
|
|
FunctionLiteral* result;
|
2010-12-07 14:03:59 +00:00
|
|
|
if (source->IsExternalTwoByteString()) {
|
|
|
|
// Notice that the stream is destroyed at the end of the branch block.
|
|
|
|
// The last line of the blocks can't be moved outside, even though they're
|
|
|
|
// identical calls.
|
2012-03-12 12:35:28 +00:00
|
|
|
ExternalTwoByteStringUtf16CharacterStream stream(
|
2010-12-07 14:03:59 +00:00
|
|
|
Handle<ExternalTwoByteString>::cast(source), 0, source->length());
|
2010-12-22 20:14:19 +00:00
|
|
|
scanner_.Initialize(&stream);
|
2013-06-26 08:05:41 +00:00
|
|
|
result = DoParseProgram(info(), source);
|
2010-12-07 14:03:59 +00:00
|
|
|
} else {
|
2012-03-12 12:35:28 +00:00
|
|
|
GenericStringUtf16CharacterStream stream(source, 0, source->length());
|
2010-12-22 20:14:19 +00:00
|
|
|
scanner_.Initialize(&stream);
|
2013-06-26 08:05:41 +00:00
|
|
|
result = DoParseProgram(info(), source);
|
2010-12-07 14:03:59 +00:00
|
|
|
}
|
2012-07-18 11:22:46 +00:00
|
|
|
|
|
|
|
if (FLAG_trace_parse && result != NULL) {
|
2013-08-29 09:15:13 +00:00
|
|
|
double ms = timer.Elapsed().InMillisecondsF();
|
2012-07-18 11:22:46 +00:00
|
|
|
if (info()->is_eval()) {
|
|
|
|
PrintF("[parsing eval");
|
|
|
|
} else if (info()->script()->name()->IsString()) {
|
|
|
|
String* name = String::cast(info()->script()->name());
|
|
|
|
SmartArrayPointer<char> name_chars = name->ToCString();
|
|
|
|
PrintF("[parsing script: %s", *name_chars);
|
|
|
|
} else {
|
|
|
|
PrintF("[parsing script");
|
|
|
|
}
|
|
|
|
PrintF(" - took %0.3f ms]\n", ms);
|
|
|
|
}
|
|
|
|
return result;
|
2010-12-07 14:03:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-15 13:48:40 +00:00
|
|
|
FunctionLiteral* Parser::DoParseProgram(CompilationInfo* info,
|
2013-06-26 08:05:41 +00:00
|
|
|
Handle<String> source) {
|
2011-10-24 07:47:22 +00:00
|
|
|
ASSERT(top_scope_ == NULL);
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(target_stack_ == NULL);
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
if (pre_parse_data_ != NULL) pre_parse_data_->Initialize();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-02-28 17:03:34 +00:00
|
|
|
Handle<String> no_name = isolate()->factory()->empty_string();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
FunctionLiteral* result = NULL;
|
2011-11-15 13:48:40 +00:00
|
|
|
{ Scope* scope = NewScope(top_scope_, GLOBAL_SCOPE);
|
|
|
|
info->SetGlobalScope(scope);
|
2012-08-28 11:25:08 +00:00
|
|
|
if (!info->context().is_null()) {
|
|
|
|
scope = Scope::DeserializeScopeChain(*info->context(), scope, zone());
|
|
|
|
}
|
2013-08-23 09:25:37 +00:00
|
|
|
original_scope_ = scope;
|
2012-03-15 13:02:21 +00:00
|
|
|
if (info->is_eval()) {
|
|
|
|
if (!scope->is_global_scope() || info->language_mode() != CLASSIC_MODE) {
|
|
|
|
scope = NewScope(scope, EVAL_SCOPE);
|
|
|
|
}
|
2012-08-28 11:25:08 +00:00
|
|
|
} else if (info->is_global()) {
|
|
|
|
scope = NewScope(scope, GLOBAL_SCOPE);
|
2011-11-15 13:48:40 +00:00
|
|
|
}
|
2011-10-21 10:26:59 +00:00
|
|
|
scope->set_start_position(0);
|
|
|
|
scope->set_end_position(source->length());
|
2012-08-28 11:25:08 +00:00
|
|
|
|
2012-12-07 10:35:50 +00:00
|
|
|
// Compute the parsing mode.
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
Mode mode = (FLAG_lazy && allow_lazy()) ? PARSE_LAZILY : PARSE_EAGERLY;
|
|
|
|
if (allow_natives_syntax() ||
|
|
|
|
extension_ != NULL ||
|
|
|
|
scope->is_eval_scope()) {
|
2012-12-07 10:35:50 +00:00
|
|
|
mode = PARSE_EAGERLY;
|
|
|
|
}
|
|
|
|
ParsingModeScope parsing_mode(this, mode);
|
|
|
|
|
2013-04-02 17:34:59 +00:00
|
|
|
// Enters 'scope'.
|
2013-04-15 12:29:44 +00:00
|
|
|
FunctionState function_state(this, scope, isolate());
|
2013-04-02 17:34:59 +00:00
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
top_scope_->SetLanguageMode(info->language_mode());
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Statement*>* body = new(zone()) ZoneList<Statement*>(16, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
bool ok = true;
|
2013-10-14 09:24:58 +00:00
|
|
|
int beg_pos = scanner().location().beg_pos;
|
2012-10-05 09:14:08 +00:00
|
|
|
ParseSourceElements(body, Token::EOS, info->is_eval(), true, &ok);
|
2011-11-24 15:17:04 +00:00
|
|
|
if (ok && !top_scope_->is_classic_mode()) {
|
2013-10-14 09:24:58 +00:00
|
|
|
CheckOctalLiteral(beg_pos, scanner().location().end_pos, &ok);
|
2011-01-24 18:13:18 +00:00
|
|
|
}
|
2011-09-01 12:31:18 +00:00
|
|
|
|
2011-11-24 15:58:09 +00:00
|
|
|
if (ok && is_extended_mode()) {
|
2012-03-15 13:02:21 +00:00
|
|
|
CheckConflictingVarDeclarations(top_scope_, &ok);
|
2011-09-01 12:31:18 +00:00
|
|
|
}
|
|
|
|
|
2013-03-07 15:46:14 +00:00
|
|
|
if (ok && info->parse_restriction() == ONLY_SINGLE_FUNCTION_LITERAL) {
|
|
|
|
if (body->length() != 1 ||
|
|
|
|
!body->at(0)->IsExpressionStatement() ||
|
|
|
|
!body->at(0)->AsExpressionStatement()->
|
|
|
|
expression()->IsFunctionLiteral()) {
|
2013-04-26 14:26:54 +00:00
|
|
|
ReportMessage("single_function_literal", Vector<const char*>::empty());
|
2013-03-07 15:46:14 +00:00
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
if (ok) {
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewFunctionLiteral(
|
2009-08-19 07:30:20 +00:00
|
|
|
no_name,
|
|
|
|
top_scope_,
|
2010-11-02 11:45:47 +00:00
|
|
|
body,
|
2011-11-09 13:54:26 +00:00
|
|
|
function_state.materialized_literal_count(),
|
|
|
|
function_state.expected_property_count(),
|
2011-11-11 13:48:14 +00:00
|
|
|
function_state.handler_count(),
|
2009-08-19 07:30:20 +00:00
|
|
|
0,
|
2012-02-14 14:14:51 +00:00
|
|
|
FunctionLiteral::kNoDuplicateParameters,
|
2011-08-09 12:43:08 +00:00
|
|
|
FunctionLiteral::ANONYMOUS_EXPRESSION,
|
2012-08-07 14:47:36 +00:00
|
|
|
FunctionLiteral::kGlobalOrEval,
|
2013-04-02 17:34:59 +00:00
|
|
|
FunctionLiteral::kNotParenthesized,
|
2013-10-14 09:24:58 +00:00
|
|
|
FunctionLiteral::kNotGenerator,
|
|
|
|
0);
|
2012-02-08 09:56:33 +00:00
|
|
|
result->set_ast_properties(factory()->visitor()->ast_properties());
|
2013-09-05 13:20:51 +00:00
|
|
|
result->set_dont_optimize_reason(
|
|
|
|
factory()->visitor()->dont_optimize_reason());
|
2010-11-29 13:24:37 +00:00
|
|
|
} else if (stack_overflow_) {
|
2011-03-18 20:35:07 +00:00
|
|
|
isolate()->StackOverflow();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the target stack is empty.
|
|
|
|
ASSERT(target_stack_ == NULL);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-02-20 14:02:59 +00:00
|
|
|
|
2012-06-20 10:56:53 +00:00
|
|
|
FunctionLiteral* Parser::ParseLazy() {
|
2013-08-29 09:15:13 +00:00
|
|
|
HistogramTimerScope timer_scope(isolate()->counters()->parse_lazy());
|
2010-09-30 09:28:58 +00:00
|
|
|
Handle<String> source(String::cast(script_->source()));
|
2011-03-23 11:13:07 +00:00
|
|
|
isolate()->counters()->total_parse_size()->Increment(source->length());
|
2013-08-29 09:15:13 +00:00
|
|
|
ElapsedTimer timer;
|
|
|
|
if (FLAG_trace_parse) {
|
|
|
|
timer.Start();
|
|
|
|
}
|
2012-06-20 10:56:53 +00:00
|
|
|
Handle<SharedFunctionInfo> shared_info = info()->shared_info();
|
2012-07-18 11:22:46 +00:00
|
|
|
|
2010-12-07 14:03:59 +00:00
|
|
|
// Initialize parser state.
|
|
|
|
source->TryFlatten();
|
2012-07-18 11:22:46 +00:00
|
|
|
FunctionLiteral* result;
|
2010-12-07 14:03:59 +00:00
|
|
|
if (source->IsExternalTwoByteString()) {
|
2012-03-12 12:35:28 +00:00
|
|
|
ExternalTwoByteStringUtf16CharacterStream stream(
|
2010-12-07 14:03:59 +00:00
|
|
|
Handle<ExternalTwoByteString>::cast(source),
|
2011-03-09 16:57:03 +00:00
|
|
|
shared_info->start_position(),
|
|
|
|
shared_info->end_position());
|
2013-06-26 08:05:41 +00:00
|
|
|
result = ParseLazy(&stream);
|
2010-12-07 14:03:59 +00:00
|
|
|
} else {
|
2012-03-12 12:35:28 +00:00
|
|
|
GenericStringUtf16CharacterStream stream(source,
|
|
|
|
shared_info->start_position(),
|
|
|
|
shared_info->end_position());
|
2013-06-26 08:05:41 +00:00
|
|
|
result = ParseLazy(&stream);
|
2010-12-07 14:03:59 +00:00
|
|
|
}
|
2012-07-18 11:22:46 +00:00
|
|
|
|
|
|
|
if (FLAG_trace_parse && result != NULL) {
|
2013-08-29 09:15:13 +00:00
|
|
|
double ms = timer.Elapsed().InMillisecondsF();
|
2012-08-16 11:54:48 +00:00
|
|
|
SmartArrayPointer<char> name_chars = result->debug_name()->ToCString();
|
2012-07-18 11:22:46 +00:00
|
|
|
PrintF("[parsing function: %s - took %0.3f ms]\n", *name_chars, ms);
|
|
|
|
}
|
|
|
|
return result;
|
2010-12-07 14:03:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-26 08:05:41 +00:00
|
|
|
FunctionLiteral* Parser::ParseLazy(Utf16CharacterStream* source) {
|
2012-06-20 10:56:53 +00:00
|
|
|
Handle<SharedFunctionInfo> shared_info = info()->shared_info();
|
2010-12-22 20:14:19 +00:00
|
|
|
scanner_.Initialize(source);
|
2011-10-24 07:47:22 +00:00
|
|
|
ASSERT(top_scope_ == NULL);
|
2010-12-07 14:03:59 +00:00
|
|
|
ASSERT(target_stack_ == NULL);
|
|
|
|
|
2011-03-09 16:57:03 +00:00
|
|
|
Handle<String> name(String::cast(shared_info->name()));
|
2012-06-11 12:42:31 +00:00
|
|
|
fni_ = new(zone()) FuncNameInferrer(isolate(), zone());
|
2010-08-23 13:26:03 +00:00
|
|
|
fni_->PushEnclosingName(name);
|
|
|
|
|
2012-08-07 14:47:36 +00:00
|
|
|
ParsingModeScope parsing_mode(this, PARSE_EAGERLY);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Place holder for the result.
|
|
|
|
FunctionLiteral* result = NULL;
|
|
|
|
|
|
|
|
{
|
|
|
|
// Parse the function literal.
|
2011-10-21 10:26:59 +00:00
|
|
|
Scope* scope = NewScope(top_scope_, GLOBAL_SCOPE);
|
2012-06-20 10:56:53 +00:00
|
|
|
info()->SetGlobalScope(scope);
|
|
|
|
if (!info()->closure().is_null()) {
|
|
|
|
scope = Scope::DeserializeScopeChain(info()->closure()->context(), scope,
|
2012-06-11 12:42:31 +00:00
|
|
|
zone());
|
2011-03-09 16:57:03 +00:00
|
|
|
}
|
2013-08-23 09:25:37 +00:00
|
|
|
original_scope_ = scope;
|
2013-04-15 12:29:44 +00:00
|
|
|
FunctionState function_state(this, scope, isolate());
|
2012-06-20 10:56:53 +00:00
|
|
|
ASSERT(scope->language_mode() != STRICT_MODE || !info()->is_classic_mode());
|
2011-11-24 15:17:04 +00:00
|
|
|
ASSERT(scope->language_mode() != EXTENDED_MODE ||
|
2012-06-20 10:56:53 +00:00
|
|
|
info()->is_extended_mode());
|
|
|
|
ASSERT(info()->language_mode() == shared_info->language_mode());
|
2011-11-24 15:17:04 +00:00
|
|
|
scope->SetLanguageMode(shared_info->language_mode());
|
2013-06-06 13:28:22 +00:00
|
|
|
FunctionLiteral::FunctionType function_type = shared_info->is_expression()
|
2011-08-09 12:43:08 +00:00
|
|
|
? (shared_info->is_anonymous()
|
|
|
|
? FunctionLiteral::ANONYMOUS_EXPRESSION
|
|
|
|
: FunctionLiteral::NAMED_EXPRESSION)
|
|
|
|
: FunctionLiteral::DECLARATION;
|
2008-07-03 15:10:15 +00:00
|
|
|
bool ok = true;
|
2011-08-09 12:43:08 +00:00
|
|
|
result = ParseFunctionLiteral(name,
|
2011-08-08 16:14:46 +00:00
|
|
|
false, // Strict mode name already checked.
|
2013-04-02 17:34:59 +00:00
|
|
|
shared_info->is_generator(),
|
2011-08-08 16:14:46 +00:00
|
|
|
RelocInfo::kNoPosition,
|
2013-06-06 13:28:22 +00:00
|
|
|
function_type,
|
2011-08-08 16:14:46 +00:00
|
|
|
&ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
// Make sure the results agree.
|
|
|
|
ASSERT(ok == (result != NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the target stack is empty.
|
|
|
|
ASSERT(target_stack_ == NULL);
|
|
|
|
|
|
|
|
if (result == NULL) {
|
2011-03-18 20:35:07 +00:00
|
|
|
if (stack_overflow_) isolate()->StackOverflow();
|
2010-12-07 11:31:57 +00:00
|
|
|
} else {
|
2011-03-09 16:57:03 +00:00
|
|
|
Handle<String> inferred_name(shared_info->inferred_name());
|
2010-12-07 11:31:57 +00:00
|
|
|
result->set_inferred_name(inferred_name);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2010-09-30 09:28:58 +00:00
|
|
|
|
2013-05-21 10:45:58 +00:00
|
|
|
Handle<String> Parser::GetSymbol() {
|
2010-09-15 10:54:35 +00:00
|
|
|
int symbol_id = -1;
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
if (pre_parse_data() != NULL) {
|
|
|
|
symbol_id = pre_parse_data()->GetSymbolIdentifier();
|
2010-09-07 12:52:16 +00:00
|
|
|
}
|
2010-12-22 20:14:19 +00:00
|
|
|
return LookupSymbol(symbol_id);
|
2010-11-02 11:45:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
void Parser::ReportMessage(const char* message, Vector<const char*> args) {
|
2010-12-07 14:03:59 +00:00
|
|
|
Scanner::Location source_location = scanner().location();
|
2013-06-06 13:28:22 +00:00
|
|
|
ReportMessageAt(source_location, message, args);
|
2010-09-07 12:52:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
void Parser::ReportMessage(const char* message, Vector<Handle<String> > args) {
|
2012-03-08 13:03:07 +00:00
|
|
|
Scanner::Location source_location = scanner().location();
|
2013-06-06 13:28:22 +00:00
|
|
|
ReportMessageAt(source_location, message, args);
|
2012-03-08 13:03:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-02 11:45:47 +00:00
|
|
|
void Parser::ReportMessageAt(Scanner::Location source_location,
|
2013-06-06 13:28:22 +00:00
|
|
|
const char* message,
|
2010-11-02 11:45:47 +00:00
|
|
|
Vector<const char*> args) {
|
2008-07-03 15:10:15 +00:00
|
|
|
MessageLocation location(script_,
|
2011-01-17 09:36:10 +00:00
|
|
|
source_location.beg_pos,
|
|
|
|
source_location.end_pos);
|
2011-03-18 20:35:07 +00:00
|
|
|
Factory* factory = isolate()->factory();
|
|
|
|
Handle<FixedArray> elements = factory->NewFixedArray(args.length());
|
2008-07-03 15:10:15 +00:00
|
|
|
for (int i = 0; i < args.length(); i++) {
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<String> arg_string = factory->NewStringFromUtf8(CStrVector(args[i]));
|
2011-03-03 10:16:22 +00:00
|
|
|
elements->set(i, *arg_string);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<JSArray> array = factory->NewJSArrayWithElements(elements);
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<Object> result = factory->NewSyntaxError(message, array);
|
2011-03-18 20:35:07 +00:00
|
|
|
isolate()->Throw(*result, &location);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-17 09:36:10 +00:00
|
|
|
void Parser::ReportMessageAt(Scanner::Location source_location,
|
2013-06-06 13:28:22 +00:00
|
|
|
const char* message,
|
2011-01-17 09:36:10 +00:00
|
|
|
Vector<Handle<String> > args) {
|
|
|
|
MessageLocation location(script_,
|
|
|
|
source_location.beg_pos,
|
|
|
|
source_location.end_pos);
|
2011-03-18 20:35:07 +00:00
|
|
|
Factory* factory = isolate()->factory();
|
|
|
|
Handle<FixedArray> elements = factory->NewFixedArray(args.length());
|
2011-01-17 09:36:10 +00:00
|
|
|
for (int i = 0; i < args.length(); i++) {
|
2011-03-03 10:16:22 +00:00
|
|
|
elements->set(i, *args[i]);
|
2011-01-17 09:36:10 +00:00
|
|
|
}
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<JSArray> array = factory->NewJSArrayWithElements(elements);
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<Object> result = factory->NewSyntaxError(message, array);
|
2011-03-18 20:35:07 +00:00
|
|
|
isolate()->Throw(*result, &location);
|
2011-01-17 09:36:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-02 11:45:47 +00:00
|
|
|
void* Parser::ParseSourceElements(ZoneList<Statement*>* processor,
|
2008-07-03 15:10:15 +00:00
|
|
|
int end_token,
|
2012-03-15 13:02:21 +00:00
|
|
|
bool is_eval,
|
2012-10-05 09:14:08 +00:00
|
|
|
bool is_global,
|
2008-07-03 15:10:15 +00:00
|
|
|
bool* ok) {
|
|
|
|
// SourceElements ::
|
2012-02-20 14:02:59 +00:00
|
|
|
// (ModuleElement)* <end_token>
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Allocate a target stack to use for this set of source
|
|
|
|
// elements. This way, all scripts and functions get their own
|
|
|
|
// target stack thus avoiding illegal breaks and continues across
|
|
|
|
// functions.
|
2010-10-27 12:33:48 +00:00
|
|
|
TargetScope scope(&this->target_stack_);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ASSERT(processor != NULL);
|
2011-01-20 18:51:47 +00:00
|
|
|
bool directive_prologue = true; // Parsing directive prologue.
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
while (peek() != end_token) {
|
2011-01-20 18:51:47 +00:00
|
|
|
if (directive_prologue && peek() != Token::STRING) {
|
|
|
|
directive_prologue = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Scanner::Location token_loc = scanner().peek_location();
|
2012-10-05 09:14:08 +00:00
|
|
|
Statement* stat;
|
|
|
|
if (is_global && !is_eval) {
|
|
|
|
stat = ParseModuleElement(NULL, CHECK_OK);
|
|
|
|
} else {
|
|
|
|
stat = ParseBlockElement(NULL, CHECK_OK);
|
|
|
|
}
|
2011-01-20 18:51:47 +00:00
|
|
|
if (stat == NULL || stat->IsEmpty()) {
|
|
|
|
directive_prologue = false; // End of directive prologue.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (directive_prologue) {
|
|
|
|
// A shot at a directive.
|
2012-01-13 13:09:52 +00:00
|
|
|
ExpressionStatement* e_stat;
|
|
|
|
Literal* literal;
|
2011-01-20 18:51:47 +00:00
|
|
|
// Still processing directive prologue?
|
|
|
|
if ((e_stat = stat->AsExpressionStatement()) != NULL &&
|
|
|
|
(literal = e_stat->expression()->AsLiteral()) != NULL &&
|
2013-06-24 10:37:59 +00:00
|
|
|
literal->value()->IsString()) {
|
|
|
|
Handle<String> directive = Handle<String>::cast(literal->value());
|
2011-01-20 18:51:47 +00:00
|
|
|
|
|
|
|
// Check "use strict" directive (ES5 14.1).
|
2011-11-24 15:17:04 +00:00
|
|
|
if (top_scope_->is_classic_mode() &&
|
2013-02-28 17:03:34 +00:00
|
|
|
directive->Equals(isolate()->heap()->use_strict_string()) &&
|
2011-01-20 18:51:47 +00:00
|
|
|
token_loc.end_pos - token_loc.beg_pos ==
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->heap()->use_strict_string()->length() + 2) {
|
2012-03-15 13:02:21 +00:00
|
|
|
// TODO(mstarzinger): Global strict eval calls, need their own scope
|
|
|
|
// as specified in ES5 10.4.2(3). The correct fix would be to always
|
|
|
|
// add this scope in DoParseProgram(), but that requires adaptations
|
|
|
|
// all over the code base, so we go with a quick-fix for now.
|
2012-12-07 10:35:50 +00:00
|
|
|
// In the same manner, we have to patch the parsing mode.
|
2012-03-15 13:02:21 +00:00
|
|
|
if (is_eval && !top_scope_->is_eval_scope()) {
|
|
|
|
ASSERT(top_scope_->is_global_scope());
|
|
|
|
Scope* scope = NewScope(top_scope_, EVAL_SCOPE);
|
|
|
|
scope->set_start_position(top_scope_->start_position());
|
|
|
|
scope->set_end_position(top_scope_->end_position());
|
|
|
|
top_scope_ = scope;
|
2012-12-07 10:35:50 +00:00
|
|
|
mode_ = PARSE_EAGERLY;
|
2012-03-15 13:02:21 +00:00
|
|
|
}
|
2011-11-24 15:58:09 +00:00
|
|
|
// TODO(ES6): Fix entering extended mode, once it is specified.
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
top_scope_->SetLanguageMode(allow_harmony_scoping()
|
2011-11-24 15:17:04 +00:00
|
|
|
? EXTENDED_MODE : STRICT_MODE);
|
2011-01-20 18:51:47 +00:00
|
|
|
// "use strict" is the only directive for now.
|
|
|
|
directive_prologue = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// End of the directive prologue.
|
|
|
|
directive_prologue = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-11 12:42:31 +00:00
|
|
|
processor->Add(stat, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2009-08-19 07:30:20 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-20 14:02:59 +00:00
|
|
|
Statement* Parser::ParseModuleElement(ZoneStringList* labels,
|
|
|
|
bool* ok) {
|
|
|
|
// (Ecma 262 5th Edition, clause 14):
|
|
|
|
// SourceElement:
|
|
|
|
// Statement
|
|
|
|
// FunctionDeclaration
|
|
|
|
//
|
|
|
|
// In harmony mode we allow additionally the following productions
|
|
|
|
// ModuleElement:
|
|
|
|
// LetDeclaration
|
|
|
|
// ConstDeclaration
|
|
|
|
// ModuleDeclaration
|
|
|
|
// ImportDeclaration
|
|
|
|
// ExportDeclaration
|
2013-04-02 17:34:59 +00:00
|
|
|
// GeneratorDeclaration
|
2012-02-20 14:02:59 +00:00
|
|
|
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::FUNCTION:
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseFunctionDeclaration(NULL, ok);
|
2012-02-20 14:02:59 +00:00
|
|
|
case Token::LET:
|
|
|
|
case Token::CONST:
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseVariableStatement(kModuleElement, NULL, ok);
|
2012-02-20 14:02:59 +00:00
|
|
|
case Token::IMPORT:
|
|
|
|
return ParseImportDeclaration(ok);
|
|
|
|
case Token::EXPORT:
|
|
|
|
return ParseExportDeclaration(ok);
|
2012-02-24 15:53:09 +00:00
|
|
|
default: {
|
|
|
|
Statement* stmt = ParseStatement(labels, CHECK_OK);
|
|
|
|
// Handle 'module' as a context-sensitive keyword.
|
|
|
|
if (FLAG_harmony_modules &&
|
|
|
|
peek() == Token::IDENTIFIER &&
|
|
|
|
!scanner().HasAnyLineTerminatorBeforeNext() &&
|
|
|
|
stmt != NULL) {
|
|
|
|
ExpressionStatement* estmt = stmt->AsExpressionStatement();
|
|
|
|
if (estmt != NULL &&
|
|
|
|
estmt->expression()->AsVariableProxy() != NULL &&
|
|
|
|
estmt->expression()->AsVariableProxy()->name()->Equals(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->heap()->module_string()) &&
|
2012-02-24 15:53:09 +00:00
|
|
|
!scanner().literal_contains_escapes()) {
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseModuleDeclaration(NULL, ok);
|
2012-02-24 15:53:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return stmt;
|
|
|
|
}
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-09 08:59:03 +00:00
|
|
|
Statement* Parser::ParseModuleDeclaration(ZoneStringList* names, bool* ok) {
|
2012-02-20 14:02:59 +00:00
|
|
|
// ModuleDeclaration:
|
|
|
|
// 'module' Identifier Module
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-20 14:02:59 +00:00
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
2012-03-08 13:03:07 +00:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Module %s...\n", name->ToAsciiArray());
|
|
|
|
#endif
|
|
|
|
|
2012-02-28 10:12:39 +00:00
|
|
|
Module* module = ParseModule(CHECK_OK);
|
Get rid of static module allocation, do it in code.
Modules now have their own local scope, represented by their own context.
Module instance objects have an accessor for every export that forwards
access to the respective slot from the module's context. (Exports that are
modules themselves, however, are simple data properties.)
All modules have a _hosting_ scope/context, which (currently) is the
(innermost) enclosing global scope. To deal with recursion, nested modules
are hosted by the same scope as global ones.
For every (global or nested) module literal, the hosting context has an
internal slot that points directly to the respective module context. This
enables quick access to (statically resolved) module members by 2-dimensional
access through the hosting context. For example,
module A {
let x;
module B { let y; }
}
module C { let z; }
allocates contexts as follows:
[header| .A | .B | .C | A | C ] (global)
| | |
| | +-- [header| z ] (module)
| |
| +------- [header| y ] (module)
|
+------------ [header| x | B ] (module)
Here, .A, .B, .C are the internal slots pointing to the hosted module
contexts, whereas A, B, C hold the actual instance objects (note that every
module context also points to the respective instance object through its
extension slot in the header).
To deal with arbitrary recursion and aliases between modules,
they are created and initialized in several stages. Each stage applies to
all modules in the hosting global scope, including nested ones.
1. Allocate: for each module _literal_, allocate the module contexts and
respective instance object and wire them up. This happens in the
PushModuleContext runtime function, as generated by AllocateModules
(invoked by VisitDeclarations in the hosting scope).
2. Bind: for each module _declaration_ (i.e. literals as well as aliases),
assign the respective instance object to respective local variables. This
happens in VisitModuleDeclaration, and uses the instance objects created
in the previous stage.
For each module _literal_, this phase also constructs a module descriptor
for the next stage. This happens in VisitModuleLiteral.
3. Populate: invoke the DeclareModules runtime function to populate each
_instance_ object with accessors for it exports. This is generated by
DeclareModules (invoked by VisitDeclarations in the hosting scope again),
and uses the descriptors generated in the previous stage.
4. Initialize: execute the module bodies (and other code) in sequence. This
happens by the separate statements generated for module bodies. To reenter
the module scopes properly, the parser inserted ModuleStatements.
R=mstarzinger@chromium.org,svenpanne@chromium.org
BUG=
Review URL: https://codereview.chromium.org/11093074
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@13033 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-11-22 10:25:22 +00:00
|
|
|
VariableProxy* proxy = NewUnresolved(name, MODULE, module->interface());
|
2012-02-28 10:12:39 +00:00
|
|
|
Declaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewModuleDeclaration(proxy, module, top_scope_, pos);
|
2012-02-28 10:12:39 +00:00
|
|
|
Declare(declaration, true, CHECK_OK);
|
|
|
|
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Module %s.\n", name->ToAsciiArray());
|
|
|
|
|
|
|
|
if (FLAG_print_interfaces) {
|
|
|
|
PrintF("module %s : ", name->ToAsciiArray());
|
|
|
|
module->interface()->Print();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-06-11 12:42:31 +00:00
|
|
|
if (names) names->Add(name, zone());
|
2012-07-09 08:59:03 +00:00
|
|
|
if (module->body() == NULL)
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewEmptyStatement(pos);
|
2012-07-09 08:59:03 +00:00
|
|
|
else
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewModuleStatement(proxy, module->body(), pos);
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module* Parser::ParseModule(bool* ok) {
|
|
|
|
// Module:
|
|
|
|
// '{' ModuleElement '}'
|
2012-02-29 12:12:52 +00:00
|
|
|
// '=' ModulePath ';'
|
|
|
|
// 'at' String ';'
|
2012-02-20 14:02:59 +00:00
|
|
|
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::LBRACE:
|
|
|
|
return ParseModuleLiteral(ok);
|
|
|
|
|
2012-02-29 12:12:52 +00:00
|
|
|
case Token::ASSIGN: {
|
2012-02-20 14:02:59 +00:00
|
|
|
Expect(Token::ASSIGN, CHECK_OK);
|
2012-02-29 12:12:52 +00:00
|
|
|
Module* result = ParseModulePath(CHECK_OK);
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
return result;
|
|
|
|
}
|
2012-02-20 14:02:59 +00:00
|
|
|
|
2012-02-29 12:12:52 +00:00
|
|
|
default: {
|
2013-06-06 14:38:26 +00:00
|
|
|
ExpectContextualKeyword(CStrVector("at"), CHECK_OK);
|
2012-02-29 12:12:52 +00:00
|
|
|
Module* result = ParseModuleUrl(CHECK_OK);
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
return result;
|
|
|
|
}
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module* Parser::ParseModuleLiteral(bool* ok) {
|
|
|
|
// Module:
|
|
|
|
// '{' ModuleElement '}'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-20 14:02:59 +00:00
|
|
|
// Construct block expecting 16 statements.
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* body = factory()->NewBlock(NULL, 16, false, RelocInfo::kNoPosition);
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details) PrintF("# Literal ");
|
|
|
|
#endif
|
2012-02-20 14:02:59 +00:00
|
|
|
Scope* scope = NewScope(top_scope_, MODULE_SCOPE);
|
|
|
|
|
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
|
|
|
scope->set_start_position(scanner().location().beg_pos);
|
|
|
|
scope->SetLanguageMode(EXTENDED_MODE);
|
|
|
|
|
|
|
|
{
|
|
|
|
BlockState block_state(this, scope);
|
2012-06-11 12:42:31 +00:00
|
|
|
TargetCollector collector(zone());
|
2012-02-20 14:02:59 +00:00
|
|
|
Target target(&this->target_stack_, &collector);
|
|
|
|
Target target_body(&this->target_stack_, body);
|
|
|
|
|
|
|
|
while (peek() != Token::RBRACE) {
|
|
|
|
Statement* stat = ParseModuleElement(NULL, CHECK_OK);
|
|
|
|
if (stat && !stat->IsEmpty()) {
|
2012-06-04 14:42:58 +00:00
|
|
|
body->AddStatement(stat, zone());
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
|
|
|
scope->set_end_position(scanner().location().end_pos);
|
2012-04-16 14:43:27 +00:00
|
|
|
body->set_scope(scope);
|
2012-03-08 13:03:07 +00:00
|
|
|
|
2012-07-09 08:59:03 +00:00
|
|
|
// Check that all exports are bound.
|
2012-04-16 14:43:27 +00:00
|
|
|
Interface* interface = scope->interface();
|
2012-07-09 08:59:03 +00:00
|
|
|
for (Interface::Iterator it = interface->iterator();
|
|
|
|
!it.done(); it.Advance()) {
|
|
|
|
if (scope->LocalLookup(it.name()) == NULL) {
|
|
|
|
Handle<String> name(it.name());
|
|
|
|
ReportMessage("module_export_undefined",
|
|
|
|
Vector<Handle<String> >(&name, 1));
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-16 14:43:27 +00:00
|
|
|
interface->MakeModule(ok);
|
2012-07-09 08:59:03 +00:00
|
|
|
ASSERT(*ok);
|
2012-04-16 14:43:27 +00:00
|
|
|
interface->Freeze(ok);
|
2012-07-09 08:59:03 +00:00
|
|
|
ASSERT(*ok);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewModuleLiteral(body, interface, pos);
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module* Parser::ParseModulePath(bool* ok) {
|
|
|
|
// ModulePath:
|
|
|
|
// Identifier
|
|
|
|
// ModulePath '.' Identifier
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-20 14:02:59 +00:00
|
|
|
Module* result = ParseModuleVariable(CHECK_OK);
|
|
|
|
while (Check(Token::PERIOD)) {
|
|
|
|
Handle<String> name = ParseIdentifierName(CHECK_OK);
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Path .%s ", name->ToAsciiArray());
|
|
|
|
#endif
|
2013-10-14 09:24:58 +00:00
|
|
|
Module* member = factory()->NewModulePath(result, name, pos);
|
2012-06-11 12:42:31 +00:00
|
|
|
result->interface()->Add(name, member->interface(), zone(), ok);
|
2012-03-08 13:03:07 +00:00
|
|
|
if (!*ok) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interfaces) {
|
|
|
|
PrintF("PATH TYPE ERROR at '%s'\n", name->ToAsciiArray());
|
|
|
|
PrintF("result: ");
|
|
|
|
result->interface()->Print();
|
|
|
|
PrintF("member: ");
|
|
|
|
member->interface()->Print();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
ReportMessage("invalid_module_path", Vector<Handle<String> >(&name, 1));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
result = member;
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module* Parser::ParseModuleVariable(bool* ok) {
|
|
|
|
// ModulePath:
|
|
|
|
// Identifier
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-20 14:02:59 +00:00
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Module variable %s ", name->ToAsciiArray());
|
|
|
|
#endif
|
2012-02-20 14:02:59 +00:00
|
|
|
VariableProxy* proxy = top_scope_->NewUnresolved(
|
2012-07-13 09:29:43 +00:00
|
|
|
factory(), name, Interface::NewModule(zone()),
|
|
|
|
scanner().location().beg_pos);
|
2012-03-08 13:03:07 +00:00
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewModuleVariable(proxy, pos);
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module* Parser::ParseModuleUrl(bool* ok) {
|
|
|
|
// Module:
|
2012-02-29 12:12:52 +00:00
|
|
|
// String
|
2012-02-20 14:02:59 +00:00
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-20 14:02:59 +00:00
|
|
|
Expect(Token::STRING, CHECK_OK);
|
2013-05-21 10:45:58 +00:00
|
|
|
Handle<String> symbol = GetSymbol();
|
2012-02-20 14:02:59 +00:00
|
|
|
|
2012-03-08 13:03:07 +00:00
|
|
|
// TODO(ES6): Request JS resource from environment...
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details) PrintF("# Url ");
|
|
|
|
#endif
|
2012-04-16 14:43:27 +00:00
|
|
|
|
Get rid of static module allocation, do it in code.
Modules now have their own local scope, represented by their own context.
Module instance objects have an accessor for every export that forwards
access to the respective slot from the module's context. (Exports that are
modules themselves, however, are simple data properties.)
All modules have a _hosting_ scope/context, which (currently) is the
(innermost) enclosing global scope. To deal with recursion, nested modules
are hosted by the same scope as global ones.
For every (global or nested) module literal, the hosting context has an
internal slot that points directly to the respective module context. This
enables quick access to (statically resolved) module members by 2-dimensional
access through the hosting context. For example,
module A {
let x;
module B { let y; }
}
module C { let z; }
allocates contexts as follows:
[header| .A | .B | .C | A | C ] (global)
| | |
| | +-- [header| z ] (module)
| |
| +------- [header| y ] (module)
|
+------------ [header| x | B ] (module)
Here, .A, .B, .C are the internal slots pointing to the hosted module
contexts, whereas A, B, C hold the actual instance objects (note that every
module context also points to the respective instance object through its
extension slot in the header).
To deal with arbitrary recursion and aliases between modules,
they are created and initialized in several stages. Each stage applies to
all modules in the hosting global scope, including nested ones.
1. Allocate: for each module _literal_, allocate the module contexts and
respective instance object and wire them up. This happens in the
PushModuleContext runtime function, as generated by AllocateModules
(invoked by VisitDeclarations in the hosting scope).
2. Bind: for each module _declaration_ (i.e. literals as well as aliases),
assign the respective instance object to respective local variables. This
happens in VisitModuleDeclaration, and uses the instance objects created
in the previous stage.
For each module _literal_, this phase also constructs a module descriptor
for the next stage. This happens in VisitModuleLiteral.
3. Populate: invoke the DeclareModules runtime function to populate each
_instance_ object with accessors for it exports. This is generated by
DeclareModules (invoked by VisitDeclarations in the hosting scope again),
and uses the descriptors generated in the previous stage.
4. Initialize: execute the module bodies (and other code) in sequence. This
happens by the separate statements generated for module bodies. To reenter
the module scopes properly, the parser inserted ModuleStatements.
R=mstarzinger@chromium.org,svenpanne@chromium.org
BUG=
Review URL: https://codereview.chromium.org/11093074
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@13033 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-11-22 10:25:22 +00:00
|
|
|
// Create an empty literal as long as the feature isn't finished.
|
|
|
|
USE(symbol);
|
|
|
|
Scope* scope = NewScope(top_scope_, MODULE_SCOPE);
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* body = factory()->NewBlock(NULL, 1, false, RelocInfo::kNoPosition);
|
Get rid of static module allocation, do it in code.
Modules now have their own local scope, represented by their own context.
Module instance objects have an accessor for every export that forwards
access to the respective slot from the module's context. (Exports that are
modules themselves, however, are simple data properties.)
All modules have a _hosting_ scope/context, which (currently) is the
(innermost) enclosing global scope. To deal with recursion, nested modules
are hosted by the same scope as global ones.
For every (global or nested) module literal, the hosting context has an
internal slot that points directly to the respective module context. This
enables quick access to (statically resolved) module members by 2-dimensional
access through the hosting context. For example,
module A {
let x;
module B { let y; }
}
module C { let z; }
allocates contexts as follows:
[header| .A | .B | .C | A | C ] (global)
| | |
| | +-- [header| z ] (module)
| |
| +------- [header| y ] (module)
|
+------------ [header| x | B ] (module)
Here, .A, .B, .C are the internal slots pointing to the hosted module
contexts, whereas A, B, C hold the actual instance objects (note that every
module context also points to the respective instance object through its
extension slot in the header).
To deal with arbitrary recursion and aliases between modules,
they are created and initialized in several stages. Each stage applies to
all modules in the hosting global scope, including nested ones.
1. Allocate: for each module _literal_, allocate the module contexts and
respective instance object and wire them up. This happens in the
PushModuleContext runtime function, as generated by AllocateModules
(invoked by VisitDeclarations in the hosting scope).
2. Bind: for each module _declaration_ (i.e. literals as well as aliases),
assign the respective instance object to respective local variables. This
happens in VisitModuleDeclaration, and uses the instance objects created
in the previous stage.
For each module _literal_, this phase also constructs a module descriptor
for the next stage. This happens in VisitModuleLiteral.
3. Populate: invoke the DeclareModules runtime function to populate each
_instance_ object with accessors for it exports. This is generated by
DeclareModules (invoked by VisitDeclarations in the hosting scope again),
and uses the descriptors generated in the previous stage.
4. Initialize: execute the module bodies (and other code) in sequence. This
happens by the separate statements generated for module bodies. To reenter
the module scopes properly, the parser inserted ModuleStatements.
R=mstarzinger@chromium.org,svenpanne@chromium.org
BUG=
Review URL: https://codereview.chromium.org/11093074
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@13033 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-11-22 10:25:22 +00:00
|
|
|
body->set_scope(scope);
|
|
|
|
Interface* interface = scope->interface();
|
2013-10-14 09:24:58 +00:00
|
|
|
Module* result = factory()->NewModuleLiteral(body, interface, pos);
|
2012-04-16 14:43:27 +00:00
|
|
|
interface->Freeze(ok);
|
2012-07-09 08:59:03 +00:00
|
|
|
ASSERT(*ok);
|
|
|
|
interface->Unify(scope->interface(), zone(), ok);
|
|
|
|
ASSERT(*ok);
|
2012-04-16 14:43:27 +00:00
|
|
|
return result;
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-29 12:12:52 +00:00
|
|
|
Module* Parser::ParseModuleSpecifier(bool* ok) {
|
|
|
|
// ModuleSpecifier:
|
|
|
|
// String
|
|
|
|
// ModulePath
|
|
|
|
|
|
|
|
if (peek() == Token::STRING) {
|
|
|
|
return ParseModuleUrl(ok);
|
|
|
|
} else {
|
|
|
|
return ParseModulePath(ok);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-20 14:02:59 +00:00
|
|
|
Block* Parser::ParseImportDeclaration(bool* ok) {
|
2012-02-29 12:12:52 +00:00
|
|
|
// ImportDeclaration:
|
|
|
|
// 'import' IdentifierName (',' IdentifierName)* 'from' ModuleSpecifier ';'
|
|
|
|
//
|
|
|
|
// TODO(ES6): implement destructuring ImportSpecifiers
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-02-29 12:12:52 +00:00
|
|
|
Expect(Token::IMPORT, CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneStringList names(1, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
|
|
|
|
Handle<String> name = ParseIdentifierName(CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
names.Add(name, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
while (peek() == Token::COMMA) {
|
|
|
|
Consume(Token::COMMA);
|
|
|
|
name = ParseIdentifierName(CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
names.Add(name, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
}
|
|
|
|
|
2013-06-06 14:38:26 +00:00
|
|
|
ExpectContextualKeyword(CStrVector("from"), CHECK_OK);
|
2012-02-29 12:12:52 +00:00
|
|
|
Module* module = ParseModuleSpecifier(CHECK_OK);
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
|
|
|
|
// Generate a separate declaration for each identifier.
|
|
|
|
// TODO(ES6): once we implement destructuring, make that one declaration.
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* block = factory()->NewBlock(NULL, 1, true, RelocInfo::kNoPosition);
|
2012-02-29 12:12:52 +00:00
|
|
|
for (int i = 0; i < names.length(); ++i) {
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Import %s ", names[i]->ToAsciiArray());
|
|
|
|
#endif
|
2012-06-11 12:42:31 +00:00
|
|
|
Interface* interface = Interface::NewUnknown(zone());
|
|
|
|
module->interface()->Add(names[i], interface, zone(), ok);
|
2012-03-08 13:03:07 +00:00
|
|
|
if (!*ok) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interfaces) {
|
|
|
|
PrintF("IMPORT TYPE ERROR at '%s'\n", names[i]->ToAsciiArray());
|
|
|
|
PrintF("module: ");
|
|
|
|
module->interface()->Print();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
ReportMessage("invalid_module_path", Vector<Handle<String> >(&name, 1));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
VariableProxy* proxy = NewUnresolved(names[i], LET, interface);
|
2012-02-29 12:12:52 +00:00
|
|
|
Declaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewImportDeclaration(proxy, module, top_scope_, pos);
|
2012-02-29 12:12:52 +00:00
|
|
|
Declare(declaration, true, CHECK_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
return block;
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-29 12:12:52 +00:00
|
|
|
Statement* Parser::ParseExportDeclaration(bool* ok) {
|
|
|
|
// ExportDeclaration:
|
|
|
|
// 'export' Identifier (',' Identifier)* ';'
|
|
|
|
// 'export' VariableDeclaration
|
|
|
|
// 'export' FunctionDeclaration
|
2013-04-02 17:34:59 +00:00
|
|
|
// 'export' GeneratorDeclaration
|
2012-02-29 12:12:52 +00:00
|
|
|
// 'export' ModuleDeclaration
|
|
|
|
//
|
|
|
|
// TODO(ES6): implement structuring ExportSpecifiers
|
|
|
|
|
|
|
|
Expect(Token::EXPORT, CHECK_OK);
|
|
|
|
|
|
|
|
Statement* result = NULL;
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneStringList names(1, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
switch (peek()) {
|
|
|
|
case Token::IDENTIFIER: {
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2012-02-29 12:12:52 +00:00
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
|
|
|
// Handle 'module' as a context-sensitive keyword.
|
2013-01-09 10:30:54 +00:00
|
|
|
if (!name->IsOneByteEqualTo(STATIC_ASCII_VECTOR("module"))) {
|
2012-06-11 12:42:31 +00:00
|
|
|
names.Add(name, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
while (peek() == Token::COMMA) {
|
|
|
|
Consume(Token::COMMA);
|
|
|
|
name = ParseIdentifier(CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
names.Add(name, zone());
|
2012-02-29 12:12:52 +00:00
|
|
|
}
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewEmptyStatement(pos);
|
2012-02-29 12:12:52 +00:00
|
|
|
} else {
|
|
|
|
result = ParseModuleDeclaration(&names, CHECK_OK);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::FUNCTION:
|
|
|
|
result = ParseFunctionDeclaration(&names, CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::VAR:
|
|
|
|
case Token::LET:
|
|
|
|
case Token::CONST:
|
|
|
|
result = ParseVariableStatement(kModuleElement, &names, CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
*ok = false;
|
|
|
|
ReportUnexpectedToken(scanner().current_token());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-03-08 13:03:07 +00:00
|
|
|
// Extract declared names into export declarations and interface.
|
|
|
|
Interface* interface = top_scope_->interface();
|
2012-02-29 12:12:52 +00:00
|
|
|
for (int i = 0; i < names.length(); ++i) {
|
2012-03-08 13:03:07 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Export %s ", names[i]->ToAsciiArray());
|
|
|
|
#endif
|
2012-06-11 12:42:31 +00:00
|
|
|
Interface* inner = Interface::NewUnknown(zone());
|
|
|
|
interface->Add(names[i], inner, zone(), CHECK_OK);
|
|
|
|
if (!*ok)
|
|
|
|
return NULL;
|
2012-03-08 13:03:07 +00:00
|
|
|
VariableProxy* proxy = NewUnresolved(names[i], LET, inner);
|
|
|
|
USE(proxy);
|
|
|
|
// TODO(rossberg): Rethink whether we actually need to store export
|
|
|
|
// declarations (for compilation?).
|
|
|
|
// ExportDeclaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
// factory()->NewExportDeclaration(proxy, top_scope_, position);
|
2012-03-08 13:03:07 +00:00
|
|
|
// top_scope_->AddDeclaration(declaration);
|
2012-02-29 12:12:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(result != NULL);
|
|
|
|
return result;
|
2012-02-20 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseBlockElement(ZoneStringList* labels,
|
|
|
|
bool* ok) {
|
|
|
|
// (Ecma 262 5th Edition, clause 14):
|
|
|
|
// SourceElement:
|
|
|
|
// Statement
|
|
|
|
// FunctionDeclaration
|
|
|
|
//
|
|
|
|
// In harmony mode we allow additionally the following productions
|
|
|
|
// BlockElement (aka SourceElement):
|
|
|
|
// LetDeclaration
|
|
|
|
// ConstDeclaration
|
2013-04-02 17:34:59 +00:00
|
|
|
// GeneratorDeclaration
|
2012-02-20 14:02:59 +00:00
|
|
|
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::FUNCTION:
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseFunctionDeclaration(NULL, ok);
|
2012-02-20 14:02:59 +00:00
|
|
|
case Token::LET:
|
|
|
|
case Token::CONST:
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseVariableStatement(kModuleElement, NULL, ok);
|
2012-02-20 14:02:59 +00:00
|
|
|
default:
|
|
|
|
return ParseStatement(labels, ok);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Statement* Parser::ParseStatement(ZoneStringList* labels, bool* ok) {
|
|
|
|
// Statement ::
|
|
|
|
// Block
|
|
|
|
// VariableStatement
|
|
|
|
// EmptyStatement
|
|
|
|
// ExpressionStatement
|
|
|
|
// IfStatement
|
|
|
|
// IterationStatement
|
|
|
|
// ContinueStatement
|
|
|
|
// BreakStatement
|
|
|
|
// ReturnStatement
|
|
|
|
// WithStatement
|
|
|
|
// LabelledStatement
|
|
|
|
// SwitchStatement
|
|
|
|
// ThrowStatement
|
|
|
|
// TryStatement
|
|
|
|
// DebuggerStatement
|
|
|
|
|
|
|
|
// Note: Since labels can only be used by 'break' and 'continue'
|
|
|
|
// statements, which themselves are only valid within blocks,
|
|
|
|
// iterations or 'switch' statements (i.e., BreakableStatements),
|
|
|
|
// labels can be simply ignored in all other cases; except for
|
2009-01-15 19:08:34 +00:00
|
|
|
// trivial labeled break statements 'label: break label' which is
|
2008-07-03 15:10:15 +00:00
|
|
|
// parsed into an empty statement.
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::LBRACE:
|
|
|
|
return ParseBlock(labels, ok);
|
|
|
|
|
|
|
|
case Token::CONST: // fall through
|
2011-11-29 06:38:04 +00:00
|
|
|
case Token::LET:
|
2008-07-03 15:10:15 +00:00
|
|
|
case Token::VAR:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseVariableStatement(kStatement, NULL, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::SEMICOLON:
|
|
|
|
Next();
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewEmptyStatement(RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::IF:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseIfStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::DO:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseDoWhileStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::WHILE:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseWhileStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::FOR:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseForStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::CONTINUE:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseContinueStatement(ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::BREAK:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseBreakStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::RETURN:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseReturnStatement(ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::WITH:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseWithStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::SWITCH:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseSwitchStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::THROW:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseThrowStatement(ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::TRY: {
|
|
|
|
// NOTE: It is somewhat complicated to have labels on
|
|
|
|
// try-statements. When breaking out of a try-finally statement,
|
|
|
|
// one must take great care not to treat it as a
|
|
|
|
// fall-through. It is much easier just to wrap the entire
|
|
|
|
// try-statement in a statement block and put the labels there
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* result =
|
|
|
|
factory()->NewBlock(labels, 1, false, RelocInfo::kNoPosition);
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, result);
|
2008-07-03 15:10:15 +00:00
|
|
|
TryStatement* statement = ParseTryStatement(CHECK_OK);
|
2012-06-04 14:42:58 +00:00
|
|
|
if (result) result->AddStatement(statement, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-02-28 19:07:02 +00:00
|
|
|
case Token::FUNCTION: {
|
2011-09-21 12:27:07 +00:00
|
|
|
// FunctionDeclaration is only allowed in the context of SourceElements
|
|
|
|
// (Ecma 262 5th Edition, clause 14):
|
|
|
|
// SourceElement:
|
|
|
|
// Statement
|
|
|
|
// FunctionDeclaration
|
|
|
|
// Common language extension is to allow function declaration in place
|
|
|
|
// of any statement. This language extension is disabled in strict mode.
|
2013-04-02 17:34:59 +00:00
|
|
|
//
|
|
|
|
// In Harmony mode, this case also handles the extension:
|
|
|
|
// Statement:
|
|
|
|
// GeneratorDeclaration
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-02-28 19:07:02 +00:00
|
|
|
ReportMessageAt(scanner().peek_location(), "strict_function",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-02-29 12:12:52 +00:00
|
|
|
return ParseFunctionDeclaration(NULL, ok);
|
2011-02-28 19:07:02 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
case Token::DEBUGGER:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseDebuggerStatement(ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
default:
|
2013-10-14 09:24:58 +00:00
|
|
|
return ParseExpressionOrLabelledStatement(labels, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-08 13:03:07 +00:00
|
|
|
VariableProxy* Parser::NewUnresolved(
|
|
|
|
Handle<String> name, VariableMode mode, Interface* interface) {
|
2011-08-16 14:24:12 +00:00
|
|
|
// If we are inside a function, a declaration of a var/const variable is a
|
|
|
|
// truly local variable, and the scope of the variable is always the function
|
|
|
|
// scope.
|
2011-10-25 08:33:08 +00:00
|
|
|
// Let/const variables in harmony mode are always added to the immediately
|
|
|
|
// enclosing scope.
|
2012-02-28 10:12:39 +00:00
|
|
|
return DeclarationScope(mode)->NewUnresolved(
|
2013-10-14 09:24:58 +00:00
|
|
|
factory(), name, interface, position());
|
2012-02-28 10:12:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Parser::Declare(Declaration* declaration, bool resolve, bool* ok) {
|
2012-03-08 13:03:07 +00:00
|
|
|
VariableProxy* proxy = declaration->proxy();
|
|
|
|
Handle<String> name = proxy->name();
|
2012-02-28 10:12:39 +00:00
|
|
|
VariableMode mode = declaration->mode();
|
|
|
|
Scope* declaration_scope = DeclarationScope(mode);
|
|
|
|
Variable* var = NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-07-09 08:59:03 +00:00
|
|
|
// If a suitable scope exists, then we can statically declare this
|
2008-07-03 15:10:15 +00:00
|
|
|
// variable and also set its mode. In any case, a Declaration node
|
|
|
|
// will be added to the scope so that the declaration can be added
|
|
|
|
// to the corresponding activation frame at runtime if necessary.
|
|
|
|
// For instance declarations inside an eval scope need to be added
|
|
|
|
// to the calling function context.
|
2011-04-29 15:31:39 +00:00
|
|
|
// Similarly, strict mode eval scope does not leak variable declarations to
|
|
|
|
// the caller's scope so we declare all locals, too.
|
2011-06-30 14:37:55 +00:00
|
|
|
if (declaration_scope->is_function_scope() ||
|
2011-11-24 15:17:04 +00:00
|
|
|
declaration_scope->is_strict_or_extended_eval_scope() ||
|
2012-02-20 14:02:59 +00:00
|
|
|
declaration_scope->is_block_scope() ||
|
2012-03-08 13:03:07 +00:00
|
|
|
declaration_scope->is_module_scope() ||
|
2012-08-28 11:25:08 +00:00
|
|
|
declaration_scope->is_global_scope()) {
|
2012-07-09 08:59:03 +00:00
|
|
|
// Declare the variable in the declaration scope.
|
2012-08-28 11:25:08 +00:00
|
|
|
// For the global scope, we have to check for collisions with earlier
|
|
|
|
// (i.e., enclosing) global scopes, to maintain the illusion of a single
|
|
|
|
// global scope.
|
|
|
|
var = declaration_scope->is_global_scope()
|
|
|
|
? declaration_scope->Lookup(name)
|
|
|
|
: declaration_scope->LocalLookup(name);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (var == NULL) {
|
|
|
|
// Declare the name.
|
2012-02-28 10:12:39 +00:00
|
|
|
var = declaration_scope->DeclareLocal(
|
2012-03-08 13:03:07 +00:00
|
|
|
name, mode, declaration->initialization(), proxy->interface());
|
2012-08-28 11:25:08 +00:00
|
|
|
} else if ((mode != VAR || var->mode() != VAR) &&
|
|
|
|
(!declaration_scope->is_global_scope() ||
|
2012-08-29 09:19:53 +00:00
|
|
|
IsLexicalVariableMode(mode) ||
|
|
|
|
IsLexicalVariableMode(var->mode()))) {
|
2011-09-01 12:31:18 +00:00
|
|
|
// The name was declared in this scope before; check for conflicting
|
|
|
|
// re-declarations. We have a conflict if either of the declarations is
|
2012-08-28 11:25:08 +00:00
|
|
|
// not a var (in the global scope, we also have to ignore legacy const for
|
|
|
|
// compatibility). There is similar code in runtime.cc in the Declare
|
2011-09-01 12:31:18 +00:00
|
|
|
// functions. The function CheckNonConflictingScope checks for conflicting
|
|
|
|
// var and let bindings from different scopes whereas this is a check for
|
|
|
|
// conflicting declarations within the same scope. This check also covers
|
2012-08-28 11:25:08 +00:00
|
|
|
// the special case
|
2011-09-01 12:31:18 +00:00
|
|
|
//
|
|
|
|
// function () { let x; { var x; } }
|
|
|
|
//
|
|
|
|
// because the var declaration is hoisted to the function scope where 'x'
|
|
|
|
// is already bound.
|
2012-08-29 09:19:53 +00:00
|
|
|
ASSERT(IsDeclaredVariableMode(var->mode()));
|
2012-08-28 11:25:08 +00:00
|
|
|
if (is_extended_mode()) {
|
|
|
|
// In harmony mode we treat re-declarations as early errors. See
|
|
|
|
// ES5 16 for a definition of early errors.
|
|
|
|
SmartArrayPointer<char> c_string = name->ToCString(DISALLOW_NULLS);
|
|
|
|
const char* elms[2] = { "Variable", *c_string };
|
|
|
|
Vector<const char*> args(elms, 2);
|
|
|
|
ReportMessage("redeclaration", args);
|
|
|
|
*ok = false;
|
|
|
|
return;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message_string =
|
Get rid of static module allocation, do it in code.
Modules now have their own local scope, represented by their own context.
Module instance objects have an accessor for every export that forwards
access to the respective slot from the module's context. (Exports that are
modules themselves, however, are simple data properties.)
All modules have a _hosting_ scope/context, which (currently) is the
(innermost) enclosing global scope. To deal with recursion, nested modules
are hosted by the same scope as global ones.
For every (global or nested) module literal, the hosting context has an
internal slot that points directly to the respective module context. This
enables quick access to (statically resolved) module members by 2-dimensional
access through the hosting context. For example,
module A {
let x;
module B { let y; }
}
module C { let z; }
allocates contexts as follows:
[header| .A | .B | .C | A | C ] (global)
| | |
| | +-- [header| z ] (module)
| |
| +------- [header| y ] (module)
|
+------------ [header| x | B ] (module)
Here, .A, .B, .C are the internal slots pointing to the hosted module
contexts, whereas A, B, C hold the actual instance objects (note that every
module context also points to the respective instance object through its
extension slot in the header).
To deal with arbitrary recursion and aliases between modules,
they are created and initialized in several stages. Each stage applies to
all modules in the hosting global scope, including nested ones.
1. Allocate: for each module _literal_, allocate the module contexts and
respective instance object and wire them up. This happens in the
PushModuleContext runtime function, as generated by AllocateModules
(invoked by VisitDeclarations in the hosting scope).
2. Bind: for each module _declaration_ (i.e. literals as well as aliases),
assign the respective instance object to respective local variables. This
happens in VisitModuleDeclaration, and uses the instance objects created
in the previous stage.
For each module _literal_, this phase also constructs a module descriptor
for the next stage. This happens in VisitModuleLiteral.
3. Populate: invoke the DeclareModules runtime function to populate each
_instance_ object with accessors for it exports. This is generated by
DeclareModules (invoked by VisitDeclarations in the hosting scope again),
and uses the descriptors generated in the previous stage.
4. Initialize: execute the module bodies (and other code) in sequence. This
happens by the separate statements generated for module bodies. To reenter
the module scopes properly, the parser inserted ModuleStatements.
R=mstarzinger@chromium.org,svenpanne@chromium.org
BUG=
Review URL: https://codereview.chromium.org/11093074
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@13033 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-11-22 10:25:22 +00:00
|
|
|
isolate()->factory()->NewStringFromUtf8(CStrVector("Variable"),
|
|
|
|
TENURED);
|
2012-08-28 11:25:08 +00:00
|
|
|
Expression* expression =
|
2013-02-28 17:03:34 +00:00
|
|
|
NewThrowTypeError(isolate()->factory()->redeclaration_string(),
|
2013-06-06 13:28:22 +00:00
|
|
|
message_string, name);
|
2012-08-28 11:25:08 +00:00
|
|
|
declaration_scope->SetIllegalRedeclaration(expression);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We add a declaration node for every declaration. The compiler
|
|
|
|
// will only generate code if necessary. In particular, declarations
|
|
|
|
// for inner local variables that do not represent functions won't
|
|
|
|
// result in any generated code.
|
|
|
|
//
|
|
|
|
// Note that we always add an unresolved proxy even if it's not
|
|
|
|
// used, simply because we don't know in this method (w/o extra
|
|
|
|
// parameters) if the proxy is needed or not. The proxy will be
|
|
|
|
// bound during variable resolution time unless it was pre-bound
|
|
|
|
// below.
|
|
|
|
//
|
|
|
|
// WARNING: This will lead to multiple declaration nodes for the
|
|
|
|
// same variable if it is declared several times. This is not a
|
|
|
|
// semantic issue as long as we keep the source order, but it may be
|
|
|
|
// a performance issue since it may lead to repeated
|
|
|
|
// Runtime::DeclareContextSlot() calls.
|
2012-02-28 10:12:39 +00:00
|
|
|
declaration_scope->AddDeclaration(declaration);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-08-28 11:25:08 +00:00
|
|
|
if (mode == CONST && declaration_scope->is_global_scope()) {
|
2011-11-15 13:48:40 +00:00
|
|
|
// For global const variables we bind the proxy to a variable.
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(resolve); // should be set by all callers
|
2009-06-24 08:01:38 +00:00
|
|
|
Variable::Kind kind = Variable::NORMAL;
|
2013-07-15 14:12:20 +00:00
|
|
|
var = new(zone()) Variable(
|
|
|
|
declaration_scope, name, mode, true, kind,
|
|
|
|
kNeedsInitialization, proxy->interface());
|
2011-11-15 13:48:40 +00:00
|
|
|
} else if (declaration_scope->is_eval_scope() &&
|
2011-11-24 15:17:04 +00:00
|
|
|
declaration_scope->is_classic_mode()) {
|
2011-11-15 13:48:40 +00:00
|
|
|
// For variable declarations in a non-strict eval scope the proxy is bound
|
|
|
|
// to a lookup variable to force a dynamic declaration using the
|
|
|
|
// DeclareContextSlot runtime function.
|
|
|
|
Variable::Kind kind = Variable::NORMAL;
|
2013-07-15 14:12:20 +00:00
|
|
|
var = new(zone()) Variable(
|
|
|
|
declaration_scope, name, mode, true, kind,
|
|
|
|
declaration->initialization(), proxy->interface());
|
2011-11-15 13:48:40 +00:00
|
|
|
var->AllocateTo(Variable::LOOKUP, -1);
|
|
|
|
resolve = true;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If requested and we have a local variable, bind the proxy to the variable
|
|
|
|
// at parse-time. This is used for functions (and consts) declared inside
|
|
|
|
// statements: the corresponding function (or const) variable must be in the
|
|
|
|
// function scope and not a statement-local scope, e.g. as provided with a
|
|
|
|
// 'with' statement:
|
|
|
|
//
|
|
|
|
// with (obj) {
|
|
|
|
// function f() {}
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// which is translated into:
|
|
|
|
//
|
|
|
|
// with (obj) {
|
|
|
|
// // in this case this is not: 'var f; f = function () {};'
|
|
|
|
// var f = function () {};
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// Note that if 'f' is accessed from inside the 'with' statement, it
|
|
|
|
// will be allocated in the context (because we must be able to look
|
|
|
|
// it up dynamically) but it will also be accessed statically, i.e.,
|
|
|
|
// with a context slot index and a context chain length for this
|
|
|
|
// initialization code. Thus, inside the 'with' statement, we need
|
|
|
|
// both access to the static and the dynamic context chain; the
|
|
|
|
// runtime needs to provide both.
|
2012-03-08 13:03:07 +00:00
|
|
|
if (resolve && var != NULL) {
|
|
|
|
proxy->BindTo(var);
|
|
|
|
|
|
|
|
if (FLAG_harmony_modules) {
|
|
|
|
bool ok;
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Declare %s\n", var->name()->ToAsciiArray());
|
|
|
|
#endif
|
2012-06-11 12:42:31 +00:00
|
|
|
proxy->interface()->Unify(var->interface(), zone(), &ok);
|
2012-03-08 13:03:07 +00:00
|
|
|
if (!ok) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interfaces) {
|
|
|
|
PrintF("DECLARE TYPE ERROR\n");
|
|
|
|
PrintF("proxy: ");
|
|
|
|
proxy->interface()->Print();
|
|
|
|
PrintF("var: ");
|
|
|
|
var->interface()->Print();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
ReportMessage("module_type_error", Vector<Handle<String> >(&name, 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Language extension which is only enabled for source files loaded
|
|
|
|
// through the API's extension mechanism. A native function
|
|
|
|
// declaration is resolved by looking up the function through a
|
|
|
|
// callback provided by the extension.
|
|
|
|
Statement* Parser::ParseNativeDeclaration(bool* ok) {
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::FUNCTION, CHECK_OK);
|
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
bool done = (peek() == Token::RPAREN);
|
|
|
|
while (!done) {
|
|
|
|
ParseIdentifier(CHECK_OK);
|
|
|
|
done = (peek() == Token::RPAREN);
|
2010-11-02 07:21:37 +00:00
|
|
|
if (!done) {
|
|
|
|
Expect(Token::COMMA, CHECK_OK);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
Expect(Token::SEMICOLON, CHECK_OK);
|
|
|
|
|
|
|
|
// Make sure that the function containing the native declaration
|
|
|
|
// isn't lazily compiled. The extension structures are only
|
|
|
|
// accessible while parsing the first time not when reparsing
|
|
|
|
// because of lazy compilation.
|
2012-02-28 10:12:39 +00:00
|
|
|
DeclarationScope(VAR)->ForceEagerCompilation();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// TODO(1240846): It's weird that native function declarations are
|
|
|
|
// introduced dynamically when we meet their declarations, whereas
|
2012-01-13 13:09:52 +00:00
|
|
|
// other functions are set up when entering the surrounding scope.
|
2012-07-13 09:29:43 +00:00
|
|
|
VariableProxy* proxy = NewUnresolved(name, VAR, Interface::NewValue());
|
2012-02-28 10:12:39 +00:00
|
|
|
Declaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewVariableDeclaration(proxy, VAR, top_scope_, pos);
|
2012-02-28 10:12:39 +00:00
|
|
|
Declare(declaration, true, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
NativeFunctionLiteral* lit = factory()->NewNativeFunctionLiteral(
|
|
|
|
name, extension_, RelocInfo::kNoPosition);
|
2012-02-08 09:56:33 +00:00
|
|
|
return factory()->NewExpressionStatement(
|
|
|
|
factory()->NewAssignment(
|
2013-10-14 09:24:58 +00:00
|
|
|
Token::INIT_VAR, proxy, lit, RelocInfo::kNoPosition),
|
|
|
|
pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-29 12:12:52 +00:00
|
|
|
Statement* Parser::ParseFunctionDeclaration(ZoneStringList* names, bool* ok) {
|
2009-10-02 12:47:15 +00:00
|
|
|
// FunctionDeclaration ::
|
|
|
|
// 'function' Identifier '(' FormalParameterListopt ')' '{' FunctionBody '}'
|
2013-04-02 17:34:59 +00:00
|
|
|
// GeneratorDeclaration ::
|
|
|
|
// 'function' '*' Identifier '(' FormalParameterListopt ')'
|
|
|
|
// '{' FunctionBody '}'
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::FUNCTION, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
bool is_generator = allow_generators() && Check(Token::MUL);
|
2011-06-24 14:59:51 +00:00
|
|
|
bool is_strict_reserved = false;
|
|
|
|
Handle<String> name = ParseIdentifierOrStrictReservedWord(
|
|
|
|
&is_strict_reserved, CHECK_OK);
|
2009-10-02 12:47:15 +00:00
|
|
|
FunctionLiteral* fun = ParseFunctionLiteral(name,
|
2011-06-24 14:59:51 +00:00
|
|
|
is_strict_reserved,
|
2013-04-02 17:34:59 +00:00
|
|
|
is_generator,
|
2013-10-14 09:24:58 +00:00
|
|
|
pos,
|
2011-08-09 12:43:08 +00:00
|
|
|
FunctionLiteral::DECLARATION,
|
2009-10-02 12:47:15 +00:00
|
|
|
CHECK_OK);
|
|
|
|
// Even if we're not at the top-level of the global or a function
|
2012-08-28 11:25:08 +00:00
|
|
|
// scope, we treat it as such and introduce the function with its
|
2009-10-02 12:47:15 +00:00
|
|
|
// initial value upon entering the corresponding scope.
|
2012-08-28 11:25:08 +00:00
|
|
|
// In extended mode, a function behaves as a lexical binding, except in the
|
|
|
|
// global scope.
|
|
|
|
VariableMode mode =
|
|
|
|
is_extended_mode() && !top_scope_->is_global_scope() ? LET : VAR;
|
2012-07-13 09:29:43 +00:00
|
|
|
VariableProxy* proxy = NewUnresolved(name, mode, Interface::NewValue());
|
2012-02-28 10:12:39 +00:00
|
|
|
Declaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewFunctionDeclaration(proxy, mode, fun, top_scope_, pos);
|
2012-02-28 10:12:39 +00:00
|
|
|
Declare(declaration, true, CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
if (names) names->Add(name, zone());
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewEmptyStatement(RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Block* Parser::ParseBlock(ZoneStringList* labels, bool* ok) {
|
2011-11-24 15:17:04 +00:00
|
|
|
if (top_scope_->is_extended_mode()) return ParseScopedBlock(labels, ok);
|
2011-08-11 16:29:28 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Block ::
|
|
|
|
// '{' Statement* '}'
|
|
|
|
|
|
|
|
// Note that a Block does not introduce a new execution scope!
|
|
|
|
// (ECMA-262, 3rd, 12.2)
|
|
|
|
//
|
|
|
|
// Construct block expecting 16 statements.
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* result =
|
|
|
|
factory()->NewBlock(labels, 16, false, RelocInfo::kNoPosition);
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, result);
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
|
|
|
while (peek() != Token::RBRACE) {
|
|
|
|
Statement* stat = ParseStatement(NULL, CHECK_OK);
|
2011-07-06 17:21:32 +00:00
|
|
|
if (stat && !stat->IsEmpty()) {
|
2012-06-04 14:42:58 +00:00
|
|
|
result->AddStatement(stat, zone());
|
2011-07-06 17:21:32 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-11 16:29:28 +00:00
|
|
|
Block* Parser::ParseScopedBlock(ZoneStringList* labels, bool* ok) {
|
2012-02-20 14:02:59 +00:00
|
|
|
// The harmony mode uses block elements instead of statements.
|
2011-09-21 12:27:07 +00:00
|
|
|
//
|
|
|
|
// Block ::
|
2012-02-20 14:02:59 +00:00
|
|
|
// '{' BlockElement* '}'
|
2011-09-21 12:27:07 +00:00
|
|
|
|
2011-08-11 16:29:28 +00:00
|
|
|
// Construct block expecting 16 statements.
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* body =
|
|
|
|
factory()->NewBlock(labels, 16, false, RelocInfo::kNoPosition);
|
2011-10-21 10:26:59 +00:00
|
|
|
Scope* block_scope = NewScope(top_scope_, BLOCK_SCOPE);
|
2011-08-11 16:29:28 +00:00
|
|
|
|
|
|
|
// Parse the statements and collect escaping labels.
|
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
block_scope->set_start_position(scanner().location().beg_pos);
|
2011-11-09 13:54:26 +00:00
|
|
|
{ BlockState block_state(this, block_scope);
|
2012-06-11 12:42:31 +00:00
|
|
|
TargetCollector collector(zone());
|
2011-10-17 09:29:37 +00:00
|
|
|
Target target(&this->target_stack_, &collector);
|
2011-08-11 16:29:28 +00:00
|
|
|
Target target_body(&this->target_stack_, body);
|
|
|
|
|
|
|
|
while (peek() != Token::RBRACE) {
|
2012-02-20 14:02:59 +00:00
|
|
|
Statement* stat = ParseBlockElement(NULL, CHECK_OK);
|
2011-08-11 16:29:28 +00:00
|
|
|
if (stat && !stat->IsEmpty()) {
|
2012-06-04 14:42:58 +00:00
|
|
|
body->AddStatement(stat, zone());
|
2011-08-11 16:29:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
block_scope->set_end_position(scanner().location().end_pos);
|
2011-09-02 12:43:28 +00:00
|
|
|
block_scope = block_scope->FinalizeBlockScope();
|
2012-04-16 14:43:27 +00:00
|
|
|
body->set_scope(block_scope);
|
2011-09-06 22:00:59 +00:00
|
|
|
return body;
|
2011-08-11 16:29:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-16 14:24:12 +00:00
|
|
|
Block* Parser::ParseVariableStatement(VariableDeclarationContext var_context,
|
2012-02-29 12:12:52 +00:00
|
|
|
ZoneStringList* names,
|
2011-08-16 14:24:12 +00:00
|
|
|
bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// VariableStatement ::
|
|
|
|
// VariableDeclarations ';'
|
|
|
|
|
2011-06-30 14:37:55 +00:00
|
|
|
Handle<String> ignore;
|
2012-02-20 14:02:59 +00:00
|
|
|
Block* result =
|
2012-02-29 12:12:52 +00:00
|
|
|
ParseVariableDeclarations(var_context, NULL, names, &ignore, CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-03-22 13:20:04 +00:00
|
|
|
|
|
|
|
bool Parser::IsEvalOrArguments(Handle<String> string) {
|
2013-02-28 17:03:34 +00:00
|
|
|
return string.is_identical_to(isolate()->factory()->eval_string()) ||
|
|
|
|
string.is_identical_to(isolate()->factory()->arguments_string());
|
2011-01-20 18:51:47 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-03-22 13:20:04 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// If the variable declaration declares exactly one non-const
|
2012-02-09 13:39:26 +00:00
|
|
|
// variable, then *out is set to that variable. In all other cases,
|
|
|
|
// *out is untouched; in particular, it is the caller's responsibility
|
2008-07-03 15:10:15 +00:00
|
|
|
// to initialize it properly. This mechanism is used for the parsing
|
|
|
|
// of 'for-in' loops.
|
2011-10-17 12:19:06 +00:00
|
|
|
Block* Parser::ParseVariableDeclarations(
|
|
|
|
VariableDeclarationContext var_context,
|
|
|
|
VariableDeclarationProperties* decl_props,
|
2012-02-29 12:12:52 +00:00
|
|
|
ZoneStringList* names,
|
2011-10-17 12:19:06 +00:00
|
|
|
Handle<String>* out,
|
|
|
|
bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// VariableDeclarations ::
|
2011-10-25 08:33:08 +00:00
|
|
|
// ('var' | 'const' | 'let') (Identifier ('=' AssignmentExpression)?)+[',']
|
|
|
|
//
|
|
|
|
// The ES6 Draft Rev3 specifies the following grammar for const declarations
|
|
|
|
//
|
|
|
|
// ConstDeclaration ::
|
|
|
|
// const ConstBinding (',' ConstBinding)* ';'
|
|
|
|
// ConstBinding ::
|
|
|
|
// Identifier '=' AssignmentExpression
|
|
|
|
//
|
|
|
|
// TODO(ES6):
|
|
|
|
// ConstBinding ::
|
|
|
|
// BindingPattern '=' AssignmentExpression
|
2013-10-14 09:24:58 +00:00
|
|
|
|
|
|
|
int pos = peek_position();
|
2011-10-11 08:41:19 +00:00
|
|
|
VariableMode mode = VAR;
|
2011-08-30 11:23:57 +00:00
|
|
|
// True if the binding needs initialization. 'let' and 'const' declared
|
|
|
|
// bindings are created uninitialized by their declaration nodes and
|
|
|
|
// need initialization. 'var' declared bindings are always initialized
|
|
|
|
// immediately by their declaration nodes.
|
|
|
|
bool needs_init = false;
|
2008-07-03 15:10:15 +00:00
|
|
|
bool is_const = false;
|
2011-08-30 11:23:57 +00:00
|
|
|
Token::Value init_op = Token::INIT_VAR;
|
2008-07-03 15:10:15 +00:00
|
|
|
if (peek() == Token::VAR) {
|
|
|
|
Consume(Token::VAR);
|
|
|
|
} else if (peek() == Token::CONST) {
|
2011-11-29 06:38:04 +00:00
|
|
|
// TODO(ES6): The ES6 Draft Rev4 section 12.2.2 reads:
|
|
|
|
//
|
|
|
|
// ConstDeclaration : const ConstBinding (',' ConstBinding)* ';'
|
|
|
|
//
|
|
|
|
// * It is a Syntax Error if the code that matches this production is not
|
|
|
|
// contained in extended code.
|
|
|
|
//
|
|
|
|
// However disallowing const in classic mode will break compatibility with
|
|
|
|
// existing pages. Therefore we keep allowing const with the old
|
|
|
|
// non-harmony semantics in classic mode.
|
2008-07-03 15:10:15 +00:00
|
|
|
Consume(Token::CONST);
|
2011-11-24 15:17:04 +00:00
|
|
|
switch (top_scope_->language_mode()) {
|
|
|
|
case CLASSIC_MODE:
|
|
|
|
mode = CONST;
|
|
|
|
init_op = Token::INIT_CONST;
|
|
|
|
break;
|
|
|
|
case STRICT_MODE:
|
|
|
|
ReportMessage("strict_const", Vector<const char*>::empty());
|
2011-10-25 08:33:08 +00:00
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2011-11-24 15:17:04 +00:00
|
|
|
case EXTENDED_MODE:
|
2012-02-20 14:02:59 +00:00
|
|
|
if (var_context == kStatement) {
|
2011-11-24 15:17:04 +00:00
|
|
|
// In extended mode 'const' declarations are only allowed in source
|
|
|
|
// element positions.
|
|
|
|
ReportMessage("unprotected_const", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
mode = CONST_HARMONY;
|
|
|
|
init_op = Token::INIT_CONST_HARMONY;
|
2011-02-28 18:38:17 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
is_const = true;
|
2011-08-30 11:23:57 +00:00
|
|
|
needs_init = true;
|
2011-08-16 14:24:12 +00:00
|
|
|
} else if (peek() == Token::LET) {
|
2011-11-29 06:38:04 +00:00
|
|
|
// ES6 Draft Rev4 section 12.2.1:
|
|
|
|
//
|
|
|
|
// LetDeclaration : let LetBindingList ;
|
|
|
|
//
|
|
|
|
// * It is a Syntax Error if the code that matches this production is not
|
|
|
|
// contained in extended code.
|
|
|
|
if (!is_extended_mode()) {
|
|
|
|
ReportMessage("illegal_let", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-08-16 14:24:12 +00:00
|
|
|
Consume(Token::LET);
|
2012-02-20 14:02:59 +00:00
|
|
|
if (var_context == kStatement) {
|
2011-10-25 08:33:08 +00:00
|
|
|
// Let declarations are only allowed in source element positions.
|
2011-08-16 14:24:12 +00:00
|
|
|
ReportMessage("unprotected_let", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-10-11 08:41:19 +00:00
|
|
|
mode = LET;
|
2011-08-30 11:23:57 +00:00
|
|
|
needs_init = true;
|
|
|
|
init_op = Token::INIT_LET;
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
UNREACHABLE(); // by current callers
|
|
|
|
}
|
|
|
|
|
2012-02-28 10:12:39 +00:00
|
|
|
Scope* declaration_scope = DeclarationScope(mode);
|
|
|
|
|
2011-08-16 14:24:12 +00:00
|
|
|
// The scope of a var/const declared variable anywhere inside a function
|
2008-07-03 15:10:15 +00:00
|
|
|
// is the entire function (ECMA-262, 3rd, 10.1.3, and 12.2). Thus we can
|
2011-08-16 14:24:12 +00:00
|
|
|
// transform a source-level var/const declaration into a (Function)
|
2008-07-03 15:10:15 +00:00
|
|
|
// Scope declaration, and rewrite the source-level initialization into an
|
|
|
|
// assignment statement. We use a block to collect multiple assignments.
|
|
|
|
//
|
|
|
|
// We mark the block as initializer block because we don't want the
|
|
|
|
// rewriter to add a '.result' assignment to such a block (to get compliant
|
|
|
|
// behavior for code such as print(eval('var x = 7')), and for cosmetic
|
|
|
|
// reasons when pretty-printing. Also, unless an assignment (initialization)
|
|
|
|
// is inside an initializer block, it is ignored.
|
|
|
|
//
|
|
|
|
// Create new block with one expected declaration.
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* block = factory()->NewBlock(NULL, 1, true, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
int nvars = 0; // the number of variables declared
|
2011-06-30 14:37:55 +00:00
|
|
|
Handle<String> name;
|
2008-07-03 15:10:15 +00:00
|
|
|
do {
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->Enter();
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Parse variable name.
|
|
|
|
if (nvars > 0) Consume(Token::COMMA);
|
2011-06-30 14:37:55 +00:00
|
|
|
name = ParseIdentifier(CHECK_OK);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushVariableName(name);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-01-20 18:51:47 +00:00
|
|
|
// Strict mode variables may not be named eval or arguments
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!declaration_scope->is_classic_mode() && IsEvalOrArguments(name)) {
|
2011-01-20 18:51:47 +00:00
|
|
|
ReportMessage("strict_var_name", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Declare variable.
|
|
|
|
// Note that we *always* must treat the initial value via a separate init
|
|
|
|
// assignment for variables and constants because the value must be assigned
|
|
|
|
// when the variable is encountered in the source. But the variable/constant
|
|
|
|
// is declared (and set to 'undefined') upon entering the function within
|
|
|
|
// which the variable or constant is declared. Only function variables have
|
|
|
|
// an initial value in the declaration (because they are initialized upon
|
|
|
|
// entering the function).
|
|
|
|
//
|
|
|
|
// If we have a const declaration, in an inner scope, the proxy is always
|
|
|
|
// bound to the declared variable (independent of possibly surrounding with
|
|
|
|
// statements).
|
2011-10-25 08:33:08 +00:00
|
|
|
// For let/const declarations in harmony mode, we can also immediately
|
|
|
|
// pre-resolve the proxy because it resides in the same scope as the
|
|
|
|
// declaration.
|
2012-07-13 09:29:43 +00:00
|
|
|
Interface* interface =
|
|
|
|
is_const ? Interface::NewConst() : Interface::NewValue();
|
|
|
|
VariableProxy* proxy = NewUnresolved(name, mode, interface);
|
2012-02-28 10:12:39 +00:00
|
|
|
Declaration* declaration =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewVariableDeclaration(proxy, mode, top_scope_, pos);
|
2012-02-28 10:12:39 +00:00
|
|
|
Declare(declaration, mode != VAR, CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
nvars++;
|
2011-06-30 14:37:55 +00:00
|
|
|
if (declaration_scope->num_var_or_const() > kMaxNumFunctionLocals) {
|
2011-05-16 08:27:52 +00:00
|
|
|
ReportMessageAt(scanner().location(), "too_many_variables",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
if (names) names->Add(name, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Parse initialization expression if present and/or needed. A
|
|
|
|
// declaration of the form:
|
|
|
|
//
|
|
|
|
// var v = x;
|
|
|
|
//
|
|
|
|
// is syntactic sugar for:
|
|
|
|
//
|
|
|
|
// var v; v = x;
|
|
|
|
//
|
2011-06-30 14:37:55 +00:00
|
|
|
// In particular, we need to re-lookup 'v' (in top_scope_, not
|
|
|
|
// declaration_scope) as it may be a different 'v' than the 'v' in the
|
|
|
|
// declaration (e.g., if we are inside a 'with' statement or 'catch'
|
|
|
|
// block).
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// However, note that const declarations are different! A const
|
|
|
|
// declaration of the form:
|
|
|
|
//
|
|
|
|
// const c = x;
|
|
|
|
//
|
|
|
|
// is *not* syntactic sugar for:
|
|
|
|
//
|
|
|
|
// const c; c = x;
|
|
|
|
//
|
|
|
|
// The "variable" c initialized to x is the same as the declared
|
|
|
|
// one - there is no re-lookup (see the last parameter of the
|
|
|
|
// Declare() call above).
|
|
|
|
|
2011-06-30 14:37:55 +00:00
|
|
|
Scope* initialization_scope = is_const ? declaration_scope : top_scope_;
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* value = NULL;
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = -1;
|
2011-10-25 08:33:08 +00:00
|
|
|
// Harmony consts have non-optional initializers.
|
|
|
|
if (peek() == Token::ASSIGN || mode == CONST_HARMONY) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::ASSIGN, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
pos = position();
|
2011-08-16 14:24:12 +00:00
|
|
|
value = ParseAssignmentExpression(var_context != kForStatement, CHECK_OK);
|
2010-08-23 13:26:03 +00:00
|
|
|
// Don't infer if it is "a = function(){...}();"-like expression.
|
2011-06-22 20:23:48 +00:00
|
|
|
if (fni_ != NULL &&
|
|
|
|
value->AsCall() == NULL &&
|
|
|
|
value->AsCallNew() == NULL) {
|
|
|
|
fni_->Infer();
|
2011-10-03 19:18:05 +00:00
|
|
|
} else {
|
|
|
|
fni_->RemoveLastFunction();
|
2011-06-22 20:23:48 +00:00
|
|
|
}
|
2011-10-17 12:19:06 +00:00
|
|
|
if (decl_props != NULL) *decl_props = kHasInitializers;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-11-08 13:28:53 +00:00
|
|
|
// Record the end position of the initializer.
|
|
|
|
if (proxy->var() != NULL) {
|
2013-10-14 09:24:58 +00:00
|
|
|
proxy->var()->set_initializer_position(position());
|
2011-11-08 13:28:53 +00:00
|
|
|
}
|
|
|
|
|
2011-08-30 11:23:57 +00:00
|
|
|
// Make sure that 'const x' and 'let x' initialize 'x' to undefined.
|
|
|
|
if (value == NULL && needs_init) {
|
2013-10-14 09:24:58 +00:00
|
|
|
value = GetLiteralUndefined(position());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Global variable declarations must be compiled in a specific
|
|
|
|
// way. When the script containing the global variable declaration
|
|
|
|
// is entered, the global variable must be declared, so that if it
|
2012-04-16 13:20:50 +00:00
|
|
|
// doesn't exist (on the global object itself, see ES5 errata) it
|
2008-07-03 15:10:15 +00:00
|
|
|
// gets created with an initial undefined value. This is handled
|
|
|
|
// by the declarations part of the function representing the
|
|
|
|
// top-level global code; see Runtime::DeclareGlobalVariable. If
|
|
|
|
// it already exists (in the object or in a prototype), it is
|
|
|
|
// *not* touched until the variable declaration statement is
|
|
|
|
// executed.
|
|
|
|
//
|
|
|
|
// Executing the variable declaration statement will always
|
|
|
|
// guarantee to give the global object a "local" variable; a
|
|
|
|
// variable defined in the global object and not in any
|
|
|
|
// prototype. This way, global variable declarations can shadow
|
|
|
|
// properties in the prototype chain, but only after the variable
|
|
|
|
// declaration statement has been executed. This is important in
|
|
|
|
// browsers where the global object (window) has lots of
|
|
|
|
// properties defined in prototype objects.
|
2012-08-28 11:25:08 +00:00
|
|
|
if (initialization_scope->is_global_scope() &&
|
2012-08-29 09:19:53 +00:00
|
|
|
!IsLexicalVariableMode(mode)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Compute the arguments for the runtime call.
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Expression*>* arguments =
|
|
|
|
new(zone()) ZoneList<Expression*>(3, zone());
|
2011-04-04 06:29:02 +00:00
|
|
|
// We have at least 1 parameter.
|
2013-10-14 09:24:58 +00:00
|
|
|
arguments->Add(factory()->NewLiteral(name, pos), zone());
|
2011-03-01 06:10:41 +00:00
|
|
|
CallRuntime* initialize;
|
2011-03-02 04:53:43 +00:00
|
|
|
|
2011-03-01 06:10:41 +00:00
|
|
|
if (is_const) {
|
2012-06-11 12:42:31 +00:00
|
|
|
arguments->Add(value, zone());
|
2011-03-02 04:53:43 +00:00
|
|
|
value = NULL; // zap the value to avoid the unnecessary assignment
|
|
|
|
|
|
|
|
// Construct the call to Runtime_InitializeConstGlobal
|
|
|
|
// and add it to the initialization statement block.
|
|
|
|
// Note that the function does different things depending on
|
|
|
|
// the number of arguments (1 or 2).
|
2012-02-08 09:56:33 +00:00
|
|
|
initialize = factory()->NewCallRuntime(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->InitializeConstGlobal_string(),
|
2012-02-08 09:56:33 +00:00
|
|
|
Runtime::FunctionForId(Runtime::kInitializeConstGlobal),
|
2013-10-14 09:24:58 +00:00
|
|
|
arguments, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
2011-03-02 04:53:43 +00:00
|
|
|
// Add strict mode.
|
|
|
|
// We may want to pass singleton to avoid Literal allocations.
|
2011-11-24 15:17:04 +00:00
|
|
|
LanguageMode language_mode = initialization_scope->language_mode();
|
2013-10-14 09:24:58 +00:00
|
|
|
arguments->Add(factory()->NewNumberLiteral(language_mode, pos), zone());
|
2011-03-02 04:53:43 +00:00
|
|
|
|
|
|
|
// Be careful not to assign a value to the global variable if
|
|
|
|
// we're in a with. The initialization value should not
|
|
|
|
// necessarily be stored in the global object in that case,
|
|
|
|
// which is why we need to generate a separate assignment node.
|
|
|
|
if (value != NULL && !inside_with()) {
|
2012-06-11 12:42:31 +00:00
|
|
|
arguments->Add(value, zone());
|
2011-03-02 04:53:43 +00:00
|
|
|
value = NULL; // zap the value to avoid the unnecessary assignment
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the call to Runtime_InitializeVarGlobal
|
|
|
|
// and add it to the initialization statement block.
|
|
|
|
// Note that the function does different things depending on
|
|
|
|
// the number of arguments (2 or 3).
|
2012-02-08 09:56:33 +00:00
|
|
|
initialize = factory()->NewCallRuntime(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->InitializeVarGlobal_string(),
|
2012-02-08 09:56:33 +00:00
|
|
|
Runtime::FunctionForId(Runtime::kInitializeVarGlobal),
|
2013-10-14 09:24:58 +00:00
|
|
|
arguments, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-03-02 04:53:43 +00:00
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
block->AddStatement(
|
|
|
|
factory()->NewExpressionStatement(initialize, RelocInfo::kNoPosition),
|
|
|
|
zone());
|
2011-11-15 13:48:40 +00:00
|
|
|
} else if (needs_init) {
|
|
|
|
// Constant initializations always assign to the declared constant which
|
|
|
|
// is always at the function scope level. This is only relevant for
|
|
|
|
// dynamically looked-up variables and constants (the start context for
|
|
|
|
// constant lookups is always the function context, while it is the top
|
|
|
|
// context for var declared variables). Sigh...
|
|
|
|
// For 'let' and 'const' declared variables in harmony mode the
|
|
|
|
// initialization also always assigns to the declared variable.
|
|
|
|
ASSERT(proxy != NULL);
|
|
|
|
ASSERT(proxy->var() != NULL);
|
|
|
|
ASSERT(value != NULL);
|
|
|
|
Assignment* assignment =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewAssignment(init_op, proxy, value, pos);
|
|
|
|
block->AddStatement(
|
|
|
|
factory()->NewExpressionStatement(assignment, RelocInfo::kNoPosition),
|
|
|
|
zone());
|
2011-11-15 13:48:40 +00:00
|
|
|
value = NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-09-06 21:22:35 +00:00
|
|
|
// Add an assignment node to the initialization statement block if we still
|
2011-11-15 13:48:40 +00:00
|
|
|
// have a pending initialization value.
|
2008-07-03 15:10:15 +00:00
|
|
|
if (value != NULL) {
|
2011-11-15 13:48:40 +00:00
|
|
|
ASSERT(mode == VAR);
|
|
|
|
// 'var' initializations are simply assignments (with all the consequences
|
|
|
|
// if they are inside a 'with' statement - they may change a 'with' object
|
|
|
|
// property).
|
2012-02-08 09:56:33 +00:00
|
|
|
VariableProxy* proxy =
|
2012-07-13 09:29:43 +00:00
|
|
|
initialization_scope->NewUnresolved(factory(), name, interface);
|
2011-04-04 06:29:02 +00:00
|
|
|
Assignment* assignment =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewAssignment(init_op, proxy, value, pos);
|
|
|
|
block->AddStatement(
|
|
|
|
factory()->NewExpressionStatement(assignment, RelocInfo::kNoPosition),
|
|
|
|
zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2010-08-23 13:26:03 +00:00
|
|
|
|
|
|
|
if (fni_ != NULL) fni_->Leave();
|
2008-07-03 15:10:15 +00:00
|
|
|
} while (peek() == Token::COMMA);
|
|
|
|
|
2011-06-30 14:37:55 +00:00
|
|
|
// If there was a single non-const declaration, return it in the output
|
|
|
|
// parameter for possible use by for/in.
|
|
|
|
if (nvars == 1 && !is_const) {
|
|
|
|
*out = name;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return block;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool ContainsLabel(ZoneStringList* labels, Handle<String> label) {
|
|
|
|
ASSERT(!label.is_null());
|
|
|
|
if (labels != NULL)
|
|
|
|
for (int i = labels->length(); i-- > 0; )
|
|
|
|
if (labels->at(i).is_identical_to(label))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseExpressionOrLabelledStatement(ZoneStringList* labels,
|
|
|
|
bool* ok) {
|
|
|
|
// ExpressionStatement | LabelledStatement ::
|
|
|
|
// Expression ';'
|
|
|
|
// Identifier ':' Statement
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2011-02-04 18:36:37 +00:00
|
|
|
bool starts_with_idenfifier = peek_any_identifier();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* expr = ParseExpression(true, CHECK_OK);
|
2011-06-20 10:20:57 +00:00
|
|
|
if (peek() == Token::COLON && starts_with_idenfifier && expr != NULL &&
|
2008-07-03 15:10:15 +00:00
|
|
|
expr->AsVariableProxy() != NULL &&
|
|
|
|
!expr->AsVariableProxy()->is_this()) {
|
2010-11-16 12:10:48 +00:00
|
|
|
// Expression is a single identifier, and not, e.g., a parenthesized
|
|
|
|
// identifier.
|
2008-07-03 15:10:15 +00:00
|
|
|
VariableProxy* var = expr->AsVariableProxy();
|
|
|
|
Handle<String> label = var->name();
|
|
|
|
// TODO(1240780): We don't check for redeclaration of labels
|
|
|
|
// during preparsing since keeping track of the set of active
|
|
|
|
// labels requires nontrivial changes to the way scopes are
|
|
|
|
// structured. However, these are probably changes we want to
|
|
|
|
// make later anyway so we should go back and fix this then.
|
2010-11-02 11:45:47 +00:00
|
|
|
if (ContainsLabel(labels, label) || TargetStackContainsLabel(label)) {
|
2011-09-09 22:39:47 +00:00
|
|
|
SmartArrayPointer<char> c_string = label->ToCString(DISALLOW_NULLS);
|
2010-11-02 11:45:47 +00:00
|
|
|
const char* elms[2] = { "Label", *c_string };
|
|
|
|
Vector<const char*> args(elms, 2);
|
|
|
|
ReportMessage("redeclaration", args);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
if (labels == NULL) {
|
|
|
|
labels = new(zone()) ZoneStringList(4, zone());
|
|
|
|
}
|
|
|
|
labels->Add(label, zone());
|
2010-11-02 11:45:47 +00:00
|
|
|
// Remove the "ghost" variable that turned out to be a label
|
|
|
|
// from the top scope. This way, we don't try to resolve it
|
|
|
|
// during the scope processing.
|
|
|
|
top_scope_->RemoveUnresolved(var);
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::COLON, CHECK_OK);
|
|
|
|
return ParseStatement(labels, ok);
|
|
|
|
}
|
|
|
|
|
2011-06-20 10:20:57 +00:00
|
|
|
// If we have an extension, we allow a native function declaration.
|
|
|
|
// A native function declaration starts with "native function" with
|
|
|
|
// no line-terminator between the two words.
|
|
|
|
if (extension_ != NULL &&
|
|
|
|
peek() == Token::FUNCTION &&
|
2011-06-21 13:34:16 +00:00
|
|
|
!scanner().HasAnyLineTerminatorBeforeNext() &&
|
2011-06-20 10:20:57 +00:00
|
|
|
expr != NULL &&
|
|
|
|
expr->AsVariableProxy() != NULL &&
|
|
|
|
expr->AsVariableProxy()->name()->Equals(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->heap()->native_string()) &&
|
2011-06-20 11:52:24 +00:00
|
|
|
!scanner().literal_contains_escapes()) {
|
2011-06-20 10:20:57 +00:00
|
|
|
return ParseNativeDeclaration(ok);
|
|
|
|
}
|
|
|
|
|
2012-02-24 15:53:09 +00:00
|
|
|
// Parsed expression statement, or the context-sensitive 'module' keyword.
|
|
|
|
// Only expect semicolon in the former case.
|
|
|
|
if (!FLAG_harmony_modules ||
|
|
|
|
peek() != Token::IDENTIFIER ||
|
|
|
|
scanner().HasAnyLineTerminatorBeforeNext() ||
|
|
|
|
expr->AsVariableProxy() == NULL ||
|
|
|
|
!expr->AsVariableProxy()->name()->Equals(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->heap()->module_string()) ||
|
2012-02-24 15:53:09 +00:00
|
|
|
scanner().literal_contains_escapes()) {
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewExpressionStatement(expr, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IfStatement* Parser::ParseIfStatement(ZoneStringList* labels, bool* ok) {
|
|
|
|
// IfStatement ::
|
|
|
|
// 'if' '(' Expression ')' Statement ('else' Statement)?
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::IF, CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
Expression* condition = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
Statement* then_statement = ParseStatement(labels, CHECK_OK);
|
|
|
|
Statement* else_statement = NULL;
|
|
|
|
if (peek() == Token::ELSE) {
|
|
|
|
Next();
|
|
|
|
else_statement = ParseStatement(labels, CHECK_OK);
|
2010-11-02 11:45:47 +00:00
|
|
|
} else {
|
2013-10-14 09:24:58 +00:00
|
|
|
else_statement = factory()->NewEmptyStatement(RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewIfStatement(
|
|
|
|
condition, then_statement, else_statement, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseContinueStatement(bool* ok) {
|
|
|
|
// ContinueStatement ::
|
|
|
|
// 'continue' Identifier? ';'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::CONTINUE, CHECK_OK);
|
2009-07-07 11:41:21 +00:00
|
|
|
Handle<String> label = Handle<String>::null();
|
2008-07-03 15:10:15 +00:00
|
|
|
Token::Value tok = peek();
|
2011-06-21 13:34:16 +00:00
|
|
|
if (!scanner().HasAnyLineTerminatorBeforeNext() &&
|
2008-09-08 07:58:54 +00:00
|
|
|
tok != Token::SEMICOLON && tok != Token::RBRACE && tok != Token::EOS) {
|
2008-07-03 15:10:15 +00:00
|
|
|
label = ParseIdentifier(CHECK_OK);
|
|
|
|
}
|
|
|
|
IterationStatement* target = NULL;
|
2010-11-02 11:45:47 +00:00
|
|
|
target = LookupContinueTarget(label, CHECK_OK);
|
|
|
|
if (target == NULL) {
|
2011-01-17 09:36:10 +00:00
|
|
|
// Illegal continue statement.
|
|
|
|
const char* message = "illegal_continue";
|
|
|
|
Vector<Handle<String> > args;
|
|
|
|
if (!label.is_null()) {
|
|
|
|
message = "unknown_label";
|
|
|
|
args = Vector<Handle<String> >(&label, 1);
|
|
|
|
}
|
|
|
|
ReportMessageAt(scanner().location(), message, args);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewContinueStatement(target, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseBreakStatement(ZoneStringList* labels, bool* ok) {
|
|
|
|
// BreakStatement ::
|
|
|
|
// 'break' Identifier? ';'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::BREAK, CHECK_OK);
|
|
|
|
Handle<String> label;
|
|
|
|
Token::Value tok = peek();
|
2011-06-21 13:34:16 +00:00
|
|
|
if (!scanner().HasAnyLineTerminatorBeforeNext() &&
|
2008-09-08 07:58:54 +00:00
|
|
|
tok != Token::SEMICOLON && tok != Token::RBRACE && tok != Token::EOS) {
|
2008-07-03 15:10:15 +00:00
|
|
|
label = ParseIdentifier(CHECK_OK);
|
|
|
|
}
|
2009-01-15 19:08:34 +00:00
|
|
|
// Parse labeled break statements that target themselves into
|
2008-07-03 15:10:15 +00:00
|
|
|
// empty statements, e.g. 'l1: l2: l3: break l2;'
|
|
|
|
if (!label.is_null() && ContainsLabel(labels, label)) {
|
2012-02-08 08:40:11 +00:00
|
|
|
ExpectSemicolon(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewEmptyStatement(pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
BreakableStatement* target = NULL;
|
2010-11-02 11:45:47 +00:00
|
|
|
target = LookupBreakTarget(label, CHECK_OK);
|
|
|
|
if (target == NULL) {
|
2011-01-17 09:36:10 +00:00
|
|
|
// Illegal break statement.
|
|
|
|
const char* message = "illegal_break";
|
|
|
|
Vector<Handle<String> > args;
|
|
|
|
if (!label.is_null()) {
|
|
|
|
message = "unknown_label";
|
|
|
|
args = Vector<Handle<String> >(&label, 1);
|
|
|
|
}
|
|
|
|
ReportMessageAt(scanner().location(), message, args);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewBreakStatement(target, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseReturnStatement(bool* ok) {
|
|
|
|
// ReturnStatement ::
|
|
|
|
// 'return' Expression? ';'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
// Consume the return token. It is necessary to do that before
|
2008-07-03 15:10:15 +00:00
|
|
|
// reporting any errors on it, because of the way errors are
|
|
|
|
// reported (underlining).
|
|
|
|
Expect(Token::RETURN, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-12-07 16:03:29 +00:00
|
|
|
Token::Value tok = peek();
|
|
|
|
Statement* result;
|
2013-04-19 14:11:23 +00:00
|
|
|
Expression* return_value;
|
2011-12-07 16:03:29 +00:00
|
|
|
if (scanner().HasAnyLineTerminatorBeforeNext() ||
|
|
|
|
tok == Token::SEMICOLON ||
|
|
|
|
tok == Token::RBRACE ||
|
|
|
|
tok == Token::EOS) {
|
2013-10-14 09:24:58 +00:00
|
|
|
return_value = GetLiteralUndefined(position());
|
2011-12-07 16:03:29 +00:00
|
|
|
} else {
|
2013-04-19 14:11:23 +00:00
|
|
|
return_value = ParseExpression(true, CHECK_OK);
|
|
|
|
}
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
if (is_generator()) {
|
|
|
|
Expression* generator = factory()->NewVariableProxy(
|
|
|
|
current_function_state_->generator_object_variable());
|
|
|
|
Expression* yield = factory()->NewYield(
|
2013-10-14 09:24:58 +00:00
|
|
|
generator, return_value, Yield::FINAL, pos);
|
|
|
|
result = factory()->NewExpressionStatement(yield, pos);
|
2013-04-19 14:11:23 +00:00
|
|
|
} else {
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewReturnStatement(return_value, pos);
|
2011-12-07 16:03:29 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// An ECMAScript program is considered syntactically incorrect if it
|
|
|
|
// contains a return statement that is not within the body of a
|
|
|
|
// function. See ECMA-262, section 12.9, page 67.
|
|
|
|
//
|
|
|
|
// To be consistent with KJS we report the syntax error at runtime.
|
2011-07-04 09:34:47 +00:00
|
|
|
Scope* declaration_scope = top_scope_->DeclarationScope();
|
2011-06-30 14:37:55 +00:00
|
|
|
if (declaration_scope->is_global_scope() ||
|
|
|
|
declaration_scope->is_eval_scope()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message = isolate()->factory()->illegal_return_string();
|
|
|
|
Expression* throw_error =
|
|
|
|
NewThrowSyntaxError(message, Handle<Object>::null());
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewExpressionStatement(throw_error, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-12-07 16:03:29 +00:00
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseWithStatement(ZoneStringList* labels, bool* ok) {
|
|
|
|
// WithStatement ::
|
|
|
|
// 'with' '(' Expression ')' Statement
|
|
|
|
|
|
|
|
Expect(Token::WITH, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2011-01-20 18:51:47 +00:00
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-01-20 18:51:47 +00:00
|
|
|
ReportMessage("strict_mode_with", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
Expression* expr = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
2011-08-12 10:52:49 +00:00
|
|
|
top_scope_->DeclarationScope()->RecordWithStatement();
|
2011-10-21 10:26:59 +00:00
|
|
|
Scope* with_scope = NewScope(top_scope_, WITH_SCOPE);
|
2011-10-17 09:29:37 +00:00
|
|
|
Statement* stmt;
|
2011-11-09 13:54:26 +00:00
|
|
|
{ BlockState block_state(this, with_scope);
|
2011-10-21 10:26:59 +00:00
|
|
|
with_scope->set_start_position(scanner().peek_location().beg_pos);
|
2011-10-17 09:29:37 +00:00
|
|
|
stmt = ParseStatement(labels, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
with_scope->set_end_position(scanner().location().end_pos);
|
2011-10-17 09:29:37 +00:00
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewWithStatement(with_scope, expr, stmt, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CaseClause* Parser::ParseCaseClause(bool* default_seen_ptr, bool* ok) {
|
|
|
|
// CaseClause ::
|
|
|
|
// 'case' Expression ':' Statement*
|
|
|
|
// 'default' ':' Statement*
|
|
|
|
|
|
|
|
Expression* label = NULL; // NULL expression indicates default case
|
|
|
|
if (peek() == Token::CASE) {
|
|
|
|
Expect(Token::CASE, CHECK_OK);
|
|
|
|
label = ParseExpression(true, CHECK_OK);
|
|
|
|
} else {
|
|
|
|
Expect(Token::DEFAULT, CHECK_OK);
|
|
|
|
if (*default_seen_ptr) {
|
|
|
|
ReportMessage("multiple_defaults_in_switch",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*default_seen_ptr = true;
|
|
|
|
}
|
|
|
|
Expect(Token::COLON, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Statement*>* statements =
|
|
|
|
new(zone()) ZoneList<Statement*>(5, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
while (peek() != Token::CASE &&
|
|
|
|
peek() != Token::DEFAULT &&
|
|
|
|
peek() != Token::RBRACE) {
|
|
|
|
Statement* stat = ParseStatement(NULL, CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
statements->Add(stat, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-07-18 17:32:41 +00:00
|
|
|
return new(zone()) CaseClause(isolate(), label, statements, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SwitchStatement* Parser::ParseSwitchStatement(ZoneStringList* labels,
|
|
|
|
bool* ok) {
|
|
|
|
// SwitchStatement ::
|
|
|
|
// 'switch' '(' Expression ')' '{' CaseClause* '}'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
SwitchStatement* statement =
|
|
|
|
factory()->NewSwitchStatement(labels, peek_position());
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, statement);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
Expect(Token::SWITCH, CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
Expression* tag = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
bool default_seen = false;
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<CaseClause*>* cases = new(zone()) ZoneList<CaseClause*>(4, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
|
|
|
while (peek() != Token::RBRACE) {
|
|
|
|
CaseClause* clause = ParseCaseClause(&default_seen, CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
cases->Add(clause, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
|
|
|
|
2010-11-02 11:45:47 +00:00
|
|
|
if (statement) statement->Initialize(tag, cases);
|
2008-07-03 15:10:15 +00:00
|
|
|
return statement;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Statement* Parser::ParseThrowStatement(bool* ok) {
|
|
|
|
// ThrowStatement ::
|
|
|
|
// 'throw' Expression ';'
|
|
|
|
|
|
|
|
Expect(Token::THROW, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2011-06-21 13:34:16 +00:00
|
|
|
if (scanner().HasAnyLineTerminatorBeforeNext()) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ReportMessage("newline_after_throw", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
Expression* exception = ParseExpression(true, CHECK_OK);
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewExpressionStatement(
|
|
|
|
factory()->NewThrow(exception, pos), pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TryStatement* Parser::ParseTryStatement(bool* ok) {
|
|
|
|
// TryStatement ::
|
|
|
|
// 'try' Block Catch
|
|
|
|
// 'try' Block Finally
|
|
|
|
// 'try' Block Catch Finally
|
|
|
|
//
|
|
|
|
// Catch ::
|
|
|
|
// 'catch' '(' Identifier ')' Block
|
|
|
|
//
|
|
|
|
// Finally ::
|
|
|
|
// 'finally' Block
|
|
|
|
|
|
|
|
Expect(Token::TRY, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-06-11 12:42:31 +00:00
|
|
|
TargetCollector try_collector(zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
Block* try_block;
|
|
|
|
|
2011-06-08 13:55:33 +00:00
|
|
|
{ Target target(&this->target_stack_, &try_collector);
|
2008-07-03 15:10:15 +00:00
|
|
|
try_block = ParseBlock(NULL, CHECK_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
Token::Value tok = peek();
|
|
|
|
if (tok != Token::CATCH && tok != Token::FINALLY) {
|
|
|
|
ReportMessage("no_catch_or_finally", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we can break out from the catch block and there is a finally block,
|
2011-06-08 13:55:33 +00:00
|
|
|
// then we will need to collect escaping targets from the catch
|
|
|
|
// block. Since we don't know yet if there will be a finally block, we
|
|
|
|
// always collect the targets.
|
2012-06-11 12:42:31 +00:00
|
|
|
TargetCollector catch_collector(zone());
|
2011-06-30 14:37:55 +00:00
|
|
|
Scope* catch_scope = NULL;
|
|
|
|
Variable* catch_variable = NULL;
|
2011-06-08 13:55:33 +00:00
|
|
|
Block* catch_block = NULL;
|
|
|
|
Handle<String> name;
|
2008-07-03 15:10:15 +00:00
|
|
|
if (tok == Token::CATCH) {
|
|
|
|
Consume(Token::CATCH);
|
|
|
|
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
catch_scope = NewScope(top_scope_, CATCH_SCOPE);
|
|
|
|
catch_scope->set_start_position(scanner().location().beg_pos);
|
2011-06-08 13:55:33 +00:00
|
|
|
name = ParseIdentifier(CHECK_OK);
|
2011-01-20 18:51:47 +00:00
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode() && IsEvalOrArguments(name)) {
|
2011-01-20 18:51:47 +00:00
|
|
|
ReportMessage("strict_catch_variable", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
if (peek() == Token::LBRACE) {
|
2011-08-12 10:52:49 +00:00
|
|
|
Target target(&this->target_stack_, &catch_collector);
|
2011-11-24 15:58:09 +00:00
|
|
|
VariableMode mode = is_extended_mode() ? LET : VAR;
|
2011-11-03 11:59:51 +00:00
|
|
|
catch_variable =
|
|
|
|
catch_scope->DeclareLocal(name, mode, kCreatedInitialized);
|
2011-08-12 10:52:49 +00:00
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
BlockState block_state(this, catch_scope);
|
2011-09-08 08:59:14 +00:00
|
|
|
catch_block = ParseBlock(NULL, CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
|
|
|
}
|
2011-10-21 10:26:59 +00:00
|
|
|
catch_scope->set_end_position(scanner().location().end_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
tok = peek();
|
|
|
|
}
|
|
|
|
|
2011-06-08 13:55:33 +00:00
|
|
|
Block* finally_block = NULL;
|
|
|
|
if (tok == Token::FINALLY || catch_block == NULL) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Consume(Token::FINALLY);
|
|
|
|
finally_block = ParseBlock(NULL, CHECK_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simplify the AST nodes by converting:
|
2011-06-08 13:55:33 +00:00
|
|
|
// 'try B0 catch B1 finally B2'
|
2008-07-03 15:10:15 +00:00
|
|
|
// to:
|
2011-06-08 13:55:33 +00:00
|
|
|
// 'try { try B0 catch B1 } finally B2'
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-11-02 11:45:47 +00:00
|
|
|
if (catch_block != NULL && finally_block != NULL) {
|
2011-06-30 14:37:55 +00:00
|
|
|
// If we have both, create an inner try/catch.
|
|
|
|
ASSERT(catch_scope != NULL && catch_variable != NULL);
|
2011-11-11 13:48:14 +00:00
|
|
|
int index = current_function_state_->NextHandlerIndex();
|
2012-02-08 09:56:33 +00:00
|
|
|
TryCatchStatement* statement = factory()->NewTryCatchStatement(
|
2013-10-14 09:24:58 +00:00
|
|
|
index, try_block, catch_scope, catch_variable, catch_block,
|
|
|
|
RelocInfo::kNoPosition);
|
2011-06-08 13:55:33 +00:00
|
|
|
statement->set_escaping_targets(try_collector.targets());
|
2013-10-14 09:24:58 +00:00
|
|
|
try_block = factory()->NewBlock(NULL, 1, false, RelocInfo::kNoPosition);
|
2012-06-04 14:42:58 +00:00
|
|
|
try_block->AddStatement(statement, zone());
|
2011-06-30 14:37:55 +00:00
|
|
|
catch_block = NULL; // Clear to indicate it's been handled.
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TryStatement* result = NULL;
|
2010-11-02 11:45:47 +00:00
|
|
|
if (catch_block != NULL) {
|
|
|
|
ASSERT(finally_block == NULL);
|
2011-06-30 14:37:55 +00:00
|
|
|
ASSERT(catch_scope != NULL && catch_variable != NULL);
|
2011-11-11 13:48:14 +00:00
|
|
|
int index = current_function_state_->NextHandlerIndex();
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewTryCatchStatement(
|
2013-10-14 09:24:58 +00:00
|
|
|
index, try_block, catch_scope, catch_variable, catch_block, pos);
|
2010-11-02 11:45:47 +00:00
|
|
|
} else {
|
|
|
|
ASSERT(finally_block != NULL);
|
2011-11-11 13:48:14 +00:00
|
|
|
int index = current_function_state_->NextHandlerIndex();
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewTryFinallyStatement(
|
|
|
|
index, try_block, finally_block, pos);
|
2011-06-08 13:55:33 +00:00
|
|
|
// Combine the jump targets of the try block and the possible catch block.
|
2012-06-11 12:42:31 +00:00
|
|
|
try_collector.targets()->AddAll(*catch_collector.targets(), zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-06-08 13:55:33 +00:00
|
|
|
result->set_escaping_targets(try_collector.targets());
|
2008-07-03 15:10:15 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-12 13:14:06 +00:00
|
|
|
DoWhileStatement* Parser::ParseDoWhileStatement(ZoneStringList* labels,
|
|
|
|
bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// DoStatement ::
|
|
|
|
// 'do' Statement 'while' '(' Expression ')' ';'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
DoWhileStatement* loop =
|
|
|
|
factory()->NewDoWhileStatement(labels, peek_position());
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
Expect(Token::DO, CHECK_OK);
|
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
|
|
|
Expect(Token::WHILE, CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
2009-11-16 21:59:31 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* cond = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
// Allow do-statements to be terminated with and without
|
|
|
|
// semi-colons. This allows code such as 'do;while(0)return' to
|
|
|
|
// parse, which would not be the case if we had used the
|
|
|
|
// ExpectSemicolon() functionality here.
|
|
|
|
if (peek() == Token::SEMICOLON) Consume(Token::SEMICOLON);
|
|
|
|
|
2009-10-12 13:14:06 +00:00
|
|
|
if (loop != NULL) loop->Initialize(cond, body);
|
2008-07-03 15:10:15 +00:00
|
|
|
return loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-12 13:14:06 +00:00
|
|
|
WhileStatement* Parser::ParseWhileStatement(ZoneStringList* labels, bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// WhileStatement ::
|
|
|
|
// 'while' '(' Expression ')' Statement
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
WhileStatement* loop = factory()->NewWhileStatement(labels, peek_position());
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
Expect(Token::WHILE, CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
Expression* cond = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
|
|
|
|
2009-10-12 13:14:06 +00:00
|
|
|
if (loop != NULL) loop->Initialize(cond, body);
|
2008-07-03 15:10:15 +00:00
|
|
|
return loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 12:37:44 +00:00
|
|
|
bool Parser::CheckInOrOf(bool accept_OF,
|
|
|
|
ForEachStatement::VisitMode* visit_mode) {
|
2013-06-06 14:38:26 +00:00
|
|
|
if (Check(Token::IN)) {
|
|
|
|
*visit_mode = ForEachStatement::ENUMERATE;
|
|
|
|
return true;
|
2013-06-12 12:37:44 +00:00
|
|
|
} else if (allow_for_of() && accept_OF &&
|
|
|
|
CheckContextualKeyword(CStrVector("of"))) {
|
2013-06-06 14:38:26 +00:00
|
|
|
*visit_mode = ForEachStatement::ITERATE;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-07 11:12:21 +00:00
|
|
|
void Parser::InitializeForEachStatement(ForEachStatement* stmt,
|
|
|
|
Expression* each,
|
|
|
|
Expression* subject,
|
|
|
|
Statement* body) {
|
|
|
|
ForOfStatement* for_of = stmt->AsForOfStatement();
|
|
|
|
|
|
|
|
if (for_of != NULL) {
|
|
|
|
Factory* heap_factory = isolate()->factory();
|
|
|
|
Handle<String> iterator_str = heap_factory->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR(".iterator"));
|
|
|
|
Handle<String> result_str = heap_factory->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR(".result"));
|
|
|
|
Variable* iterator =
|
|
|
|
top_scope_->DeclarationScope()->NewTemporary(iterator_str);
|
|
|
|
Variable* result = top_scope_->DeclarationScope()->NewTemporary(result_str);
|
|
|
|
|
|
|
|
Expression* assign_iterator;
|
|
|
|
Expression* next_result;
|
|
|
|
Expression* result_done;
|
|
|
|
Expression* assign_each;
|
|
|
|
|
|
|
|
// var iterator = iterable;
|
|
|
|
{
|
|
|
|
Expression* iterator_proxy = factory()->NewVariableProxy(iterator);
|
|
|
|
assign_iterator = factory()->NewAssignment(
|
|
|
|
Token::ASSIGN, iterator_proxy, subject, RelocInfo::kNoPosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
// var result = iterator.next();
|
|
|
|
{
|
|
|
|
Expression* iterator_proxy = factory()->NewVariableProxy(iterator);
|
2013-10-14 09:24:58 +00:00
|
|
|
Expression* next_literal = factory()->NewLiteral(
|
|
|
|
heap_factory->next_string(), RelocInfo::kNoPosition);
|
2013-06-07 11:12:21 +00:00
|
|
|
Expression* next_property = factory()->NewProperty(
|
|
|
|
iterator_proxy, next_literal, RelocInfo::kNoPosition);
|
|
|
|
ZoneList<Expression*>* next_arguments =
|
|
|
|
new(zone()) ZoneList<Expression*>(0, zone());
|
|
|
|
Expression* next_call = factory()->NewCall(
|
|
|
|
next_property, next_arguments, RelocInfo::kNoPosition);
|
|
|
|
Expression* result_proxy = factory()->NewVariableProxy(result);
|
|
|
|
next_result = factory()->NewAssignment(
|
|
|
|
Token::ASSIGN, result_proxy, next_call, RelocInfo::kNoPosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
// result.done
|
|
|
|
{
|
2013-10-14 09:24:58 +00:00
|
|
|
Expression* done_literal = factory()->NewLiteral(
|
|
|
|
heap_factory->done_string(), RelocInfo::kNoPosition);
|
2013-06-07 11:12:21 +00:00
|
|
|
Expression* result_proxy = factory()->NewVariableProxy(result);
|
|
|
|
result_done = factory()->NewProperty(
|
|
|
|
result_proxy, done_literal, RelocInfo::kNoPosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
// each = result.value
|
|
|
|
{
|
2013-10-14 09:24:58 +00:00
|
|
|
Expression* value_literal = factory()->NewLiteral(
|
|
|
|
heap_factory->value_string(), RelocInfo::kNoPosition);
|
2013-06-07 11:12:21 +00:00
|
|
|
Expression* result_proxy = factory()->NewVariableProxy(result);
|
|
|
|
Expression* result_value = factory()->NewProperty(
|
|
|
|
result_proxy, value_literal, RelocInfo::kNoPosition);
|
|
|
|
assign_each = factory()->NewAssignment(
|
|
|
|
Token::ASSIGN, each, result_value, RelocInfo::kNoPosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
for_of->Initialize(each, subject, body,
|
|
|
|
assign_iterator, next_result, result_done, assign_each);
|
|
|
|
} else {
|
|
|
|
stmt->Initialize(each, subject, body);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Statement* Parser::ParseForStatement(ZoneStringList* labels, bool* ok) {
|
|
|
|
// ForStatement ::
|
|
|
|
// 'for' '(' Expression? ';' Expression? ';' Expression? ')' Statement
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Statement* init = NULL;
|
|
|
|
|
2011-10-17 12:19:06 +00:00
|
|
|
// Create an in-between scope for let-bound iteration variables.
|
|
|
|
Scope* saved_scope = top_scope_;
|
2011-10-21 10:26:59 +00:00
|
|
|
Scope* for_scope = NewScope(top_scope_, BLOCK_SCOPE);
|
2011-10-17 12:19:06 +00:00
|
|
|
top_scope_ = for_scope;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::FOR, CHECK_OK);
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
for_scope->set_start_position(scanner().location().beg_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (peek() != Token::SEMICOLON) {
|
|
|
|
if (peek() == Token::VAR || peek() == Token::CONST) {
|
2012-07-13 09:29:43 +00:00
|
|
|
bool is_const = peek() == Token::CONST;
|
2011-06-30 14:37:55 +00:00
|
|
|
Handle<String> name;
|
2013-06-12 12:37:44 +00:00
|
|
|
VariableDeclarationProperties decl_props = kHasNoInitializers;
|
2008-07-03 15:10:15 +00:00
|
|
|
Block* variable_statement =
|
2013-06-12 12:37:44 +00:00
|
|
|
ParseVariableDeclarations(kForStatement, &decl_props, NULL, &name,
|
|
|
|
CHECK_OK);
|
|
|
|
bool accept_OF = decl_props == kHasNoInitializers;
|
2013-06-06 14:38:26 +00:00
|
|
|
ForEachStatement::VisitMode mode;
|
2011-06-30 14:37:55 +00:00
|
|
|
|
2013-06-12 12:37:44 +00:00
|
|
|
if (!name.is_null() && CheckInOrOf(accept_OF, &mode)) {
|
2012-07-13 09:29:43 +00:00
|
|
|
Interface* interface =
|
|
|
|
is_const ? Interface::NewConst() : Interface::NewValue();
|
2013-10-14 09:24:58 +00:00
|
|
|
ForEachStatement* loop =
|
|
|
|
factory()->NewForEachStatement(mode, labels, pos);
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
Expression* enumerable = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
2012-10-05 09:07:53 +00:00
|
|
|
VariableProxy* each =
|
|
|
|
top_scope_->NewUnresolved(factory(), name, interface);
|
2008-07-03 15:10:15 +00:00
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
2013-06-07 11:12:21 +00:00
|
|
|
InitializeForEachStatement(loop, each, enumerable, body);
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* result =
|
|
|
|
factory()->NewBlock(NULL, 2, false, RelocInfo::kNoPosition);
|
2012-06-04 14:42:58 +00:00
|
|
|
result->AddStatement(variable_statement, zone());
|
|
|
|
result->AddStatement(loop, zone());
|
2011-10-17 12:19:06 +00:00
|
|
|
top_scope_ = saved_scope;
|
2011-10-21 10:26:59 +00:00
|
|
|
for_scope->set_end_position(scanner().location().end_pos);
|
2011-10-17 12:19:06 +00:00
|
|
|
for_scope = for_scope->FinalizeBlockScope();
|
|
|
|
ASSERT(for_scope == NULL);
|
2010-11-02 11:45:47 +00:00
|
|
|
// Parsed for-in loop w/ variable/const declaration.
|
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
init = variable_statement;
|
|
|
|
}
|
2011-10-17 12:19:06 +00:00
|
|
|
} else if (peek() == Token::LET) {
|
|
|
|
Handle<String> name;
|
|
|
|
VariableDeclarationProperties decl_props = kHasNoInitializers;
|
|
|
|
Block* variable_statement =
|
2012-02-29 12:12:52 +00:00
|
|
|
ParseVariableDeclarations(kForStatement, &decl_props, NULL, &name,
|
|
|
|
CHECK_OK);
|
2011-10-17 12:19:06 +00:00
|
|
|
bool accept_IN = !name.is_null() && decl_props != kHasInitializers;
|
2013-06-12 12:37:44 +00:00
|
|
|
bool accept_OF = decl_props == kHasNoInitializers;
|
2013-06-06 14:38:26 +00:00
|
|
|
ForEachStatement::VisitMode mode;
|
|
|
|
|
2013-06-12 12:37:44 +00:00
|
|
|
if (accept_IN && CheckInOrOf(accept_OF, &mode)) {
|
2011-10-17 12:19:06 +00:00
|
|
|
// Rewrite a for-in statement of the form
|
|
|
|
//
|
|
|
|
// for (let x in e) b
|
|
|
|
//
|
|
|
|
// into
|
|
|
|
//
|
|
|
|
// <let x' be a temporary variable>
|
|
|
|
// for (x' in e) {
|
|
|
|
// let x;
|
|
|
|
// x = x';
|
|
|
|
// b;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// TODO(keuchel): Move the temporary variable to the block scope, after
|
|
|
|
// implementing stack allocated block scoped variables.
|
2012-10-05 12:47:34 +00:00
|
|
|
Factory* heap_factory = isolate()->factory();
|
2012-10-08 11:39:08 +00:00
|
|
|
Handle<String> tempstr =
|
2013-02-28 17:03:34 +00:00
|
|
|
heap_factory->NewConsString(heap_factory->dot_for_string(), name);
|
|
|
|
Handle<String> tempname = heap_factory->InternalizeString(tempstr);
|
2012-10-05 12:47:34 +00:00
|
|
|
Variable* temp = top_scope_->DeclarationScope()->NewTemporary(tempname);
|
2012-02-08 09:56:33 +00:00
|
|
|
VariableProxy* temp_proxy = factory()->NewVariableProxy(temp);
|
2013-10-14 09:24:58 +00:00
|
|
|
ForEachStatement* loop =
|
|
|
|
factory()->NewForEachStatement(mode, labels, pos);
|
2011-10-17 12:19:06 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
|
|
|
|
2012-10-05 09:07:53 +00:00
|
|
|
// The expression does not see the loop variable.
|
|
|
|
top_scope_ = saved_scope;
|
2011-10-17 12:19:06 +00:00
|
|
|
Expression* enumerable = ParseExpression(true, CHECK_OK);
|
2012-10-05 09:07:53 +00:00
|
|
|
top_scope_ = for_scope;
|
2011-10-17 12:19:06 +00:00
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
2012-10-05 09:07:53 +00:00
|
|
|
VariableProxy* each =
|
|
|
|
top_scope_->NewUnresolved(factory(), name, Interface::NewValue());
|
2011-10-17 12:19:06 +00:00
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* body_block =
|
|
|
|
factory()->NewBlock(NULL, 3, false, RelocInfo::kNoPosition);
|
2012-02-08 09:56:33 +00:00
|
|
|
Assignment* assignment = factory()->NewAssignment(
|
|
|
|
Token::ASSIGN, each, temp_proxy, RelocInfo::kNoPosition);
|
2013-10-14 09:24:58 +00:00
|
|
|
Statement* assignment_statement = factory()->NewExpressionStatement(
|
|
|
|
assignment, RelocInfo::kNoPosition);
|
2012-06-04 14:42:58 +00:00
|
|
|
body_block->AddStatement(variable_statement, zone());
|
|
|
|
body_block->AddStatement(assignment_statement, zone());
|
|
|
|
body_block->AddStatement(body, zone());
|
2013-06-07 11:12:21 +00:00
|
|
|
InitializeForEachStatement(loop, temp_proxy, enumerable, body_block);
|
2011-10-17 12:19:06 +00:00
|
|
|
top_scope_ = saved_scope;
|
2011-10-21 10:26:59 +00:00
|
|
|
for_scope->set_end_position(scanner().location().end_pos);
|
2011-10-17 12:19:06 +00:00
|
|
|
for_scope = for_scope->FinalizeBlockScope();
|
2012-04-16 14:43:27 +00:00
|
|
|
body_block->set_scope(for_scope);
|
2011-10-17 12:19:06 +00:00
|
|
|
// Parsed for-in loop w/ let declaration.
|
|
|
|
return loop;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-10-17 12:19:06 +00:00
|
|
|
} else {
|
|
|
|
init = variable_statement;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
Expression* expression = ParseExpression(false, CHECK_OK);
|
2013-06-06 14:38:26 +00:00
|
|
|
ForEachStatement::VisitMode mode;
|
2013-06-12 12:37:44 +00:00
|
|
|
bool accept_OF = expression->AsVariableProxy();
|
2013-06-06 14:38:26 +00:00
|
|
|
|
2013-06-12 12:37:44 +00:00
|
|
|
if (CheckInOrOf(accept_OF, &mode)) {
|
2009-09-30 09:49:36 +00:00
|
|
|
// Signal a reference error if the expression is an invalid
|
|
|
|
// left-hand side expression. We could report this as a syntax
|
|
|
|
// error here but for compatibility with JSC we choose to report
|
|
|
|
// the error at runtime.
|
2008-07-03 15:10:15 +00:00
|
|
|
if (expression == NULL || !expression->IsValidLeftHandSide()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message =
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->invalid_lhs_in_for_in_string();
|
2013-06-06 13:28:22 +00:00
|
|
|
expression = NewThrowReferenceError(message);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
ForEachStatement* loop =
|
|
|
|
factory()->NewForEachStatement(mode, labels, pos);
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
Expression* enumerable = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
2013-06-07 11:12:21 +00:00
|
|
|
InitializeForEachStatement(loop, expression, enumerable, body);
|
2011-10-17 12:19:06 +00:00
|
|
|
top_scope_ = saved_scope;
|
2011-10-21 10:26:59 +00:00
|
|
|
for_scope->set_end_position(scanner().location().end_pos);
|
2011-10-17 12:19:06 +00:00
|
|
|
for_scope = for_scope->FinalizeBlockScope();
|
|
|
|
ASSERT(for_scope == NULL);
|
2008-07-03 15:10:15 +00:00
|
|
|
// Parsed for-in loop.
|
|
|
|
return loop;
|
|
|
|
|
|
|
|
} else {
|
2013-10-14 09:24:58 +00:00
|
|
|
init = factory()->NewExpressionStatement(
|
|
|
|
expression, RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Standard 'for' loop
|
2013-10-14 09:24:58 +00:00
|
|
|
ForStatement* loop = factory()->NewForStatement(labels, pos);
|
2010-10-27 12:33:48 +00:00
|
|
|
Target target(&this->target_stack_, loop);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Parsed initializer at this point.
|
|
|
|
Expect(Token::SEMICOLON, CHECK_OK);
|
|
|
|
|
|
|
|
Expression* cond = NULL;
|
|
|
|
if (peek() != Token::SEMICOLON) {
|
|
|
|
cond = ParseExpression(true, CHECK_OK);
|
|
|
|
}
|
|
|
|
Expect(Token::SEMICOLON, CHECK_OK);
|
|
|
|
|
|
|
|
Statement* next = NULL;
|
|
|
|
if (peek() != Token::RPAREN) {
|
|
|
|
Expression* exp = ParseExpression(true, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
next = factory()->NewExpressionStatement(exp, RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
Statement* body = ParseStatement(NULL, CHECK_OK);
|
2011-10-17 12:19:06 +00:00
|
|
|
top_scope_ = saved_scope;
|
2011-10-21 10:26:59 +00:00
|
|
|
for_scope->set_end_position(scanner().location().end_pos);
|
2011-10-17 12:19:06 +00:00
|
|
|
for_scope = for_scope->FinalizeBlockScope();
|
|
|
|
if (for_scope != NULL) {
|
|
|
|
// Rewrite a for statement of the form
|
|
|
|
//
|
|
|
|
// for (let x = i; c; n) b
|
|
|
|
//
|
|
|
|
// into
|
|
|
|
//
|
|
|
|
// {
|
|
|
|
// let x = i;
|
|
|
|
// for (; c; n) b
|
|
|
|
// }
|
|
|
|
ASSERT(init != NULL);
|
2013-10-14 09:24:58 +00:00
|
|
|
Block* result = factory()->NewBlock(NULL, 2, false, RelocInfo::kNoPosition);
|
2012-06-04 14:42:58 +00:00
|
|
|
result->AddStatement(init, zone());
|
|
|
|
result->AddStatement(loop, zone());
|
2012-04-16 14:43:27 +00:00
|
|
|
result->set_scope(for_scope);
|
2013-06-06 14:38:26 +00:00
|
|
|
loop->Initialize(NULL, cond, next, body);
|
2011-10-17 12:19:06 +00:00
|
|
|
return result;
|
|
|
|
} else {
|
2013-06-06 14:38:26 +00:00
|
|
|
loop->Initialize(init, cond, next, body);
|
2011-10-17 12:19:06 +00:00
|
|
|
return loop;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Precedence = 1
|
|
|
|
Expression* Parser::ParseExpression(bool accept_IN, bool* ok) {
|
|
|
|
// Expression ::
|
|
|
|
// AssignmentExpression
|
|
|
|
// Expression ',' AssignmentExpression
|
|
|
|
|
|
|
|
Expression* result = ParseAssignmentExpression(accept_IN, CHECK_OK);
|
|
|
|
while (peek() == Token::COMMA) {
|
|
|
|
Expect(Token::COMMA, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* right = ParseAssignmentExpression(accept_IN, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewBinaryOperation(Token::COMMA, result, right, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Precedence = 2
|
|
|
|
Expression* Parser::ParseAssignmentExpression(bool accept_IN, bool* ok) {
|
|
|
|
// AssignmentExpression ::
|
|
|
|
// ConditionalExpression
|
2013-04-02 17:34:59 +00:00
|
|
|
// YieldExpression
|
2008-07-03 15:10:15 +00:00
|
|
|
// LeftHandSideExpression AssignmentOperator AssignmentExpression
|
|
|
|
|
2013-04-02 17:34:59 +00:00
|
|
|
if (peek() == Token::YIELD && is_generator()) {
|
|
|
|
return ParseYieldExpression(ok);
|
|
|
|
}
|
|
|
|
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->Enter();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* expression = ParseConditionalExpression(accept_IN, CHECK_OK);
|
|
|
|
|
|
|
|
if (!Token::IsAssignmentOp(peek())) {
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->Leave();
|
2008-07-03 15:10:15 +00:00
|
|
|
// Parsed conditional expression only (no assignment).
|
|
|
|
return expression;
|
|
|
|
}
|
|
|
|
|
2009-09-30 09:49:36 +00:00
|
|
|
// Signal a reference error if the expression is an invalid left-hand
|
|
|
|
// side expression. We could report this as a syntax error here but
|
|
|
|
// for compatibility with JSC we choose to report the error at
|
|
|
|
// runtime.
|
2012-12-18 12:00:50 +00:00
|
|
|
// TODO(ES5): Should change parsing for spec conformance.
|
2008-07-03 15:10:15 +00:00
|
|
|
if (expression == NULL || !expression->IsValidLeftHandSide()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message =
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->invalid_lhs_in_assignment_string();
|
2013-06-06 13:28:22 +00:00
|
|
|
expression = NewThrowReferenceError(message);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-01-26 19:21:46 +00:00
|
|
|
// Assignment to eval or arguments is disallowed in strict mode.
|
|
|
|
CheckStrictModeLValue(expression, "strict_lhs_assignment", CHECK_OK);
|
|
|
|
}
|
2011-12-05 14:43:28 +00:00
|
|
|
MarkAsLValue(expression);
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Token::Value op = Next(); // Get assignment operator.
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* right = ParseAssignmentExpression(accept_IN, CHECK_OK);
|
|
|
|
|
|
|
|
// TODO(1231235): We try to estimate the set of properties set by
|
|
|
|
// constructors. We define a new property whenever there is an
|
|
|
|
// assignment to a property of 'this'. We should probably only add
|
|
|
|
// properties if we haven't seen them before. Otherwise we'll
|
|
|
|
// probably overestimate the number of properties.
|
|
|
|
Property* property = expression ? expression->AsProperty() : NULL;
|
|
|
|
if (op == Token::ASSIGN &&
|
|
|
|
property != NULL &&
|
|
|
|
property->obj()->AsVariableProxy() != NULL &&
|
|
|
|
property->obj()->AsVariableProxy()->is_this()) {
|
2011-11-09 13:54:26 +00:00
|
|
|
current_function_state_->AddProperty();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2010-11-22 09:57:21 +00:00
|
|
|
// If we assign a function literal to a property we pretenure the
|
|
|
|
// literal so it can be added as a constant function property.
|
|
|
|
if (property != NULL && right->AsFunctionLiteral() != NULL) {
|
2011-11-09 13:54:26 +00:00
|
|
|
right->AsFunctionLiteral()->set_pretenure();
|
2010-11-22 09:57:21 +00:00
|
|
|
}
|
|
|
|
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) {
|
|
|
|
// Check if the right hand side is a call to avoid inferring a
|
|
|
|
// name if we're dealing with "a = function(){...}();"-like
|
|
|
|
// expression.
|
|
|
|
if ((op == Token::INIT_VAR
|
|
|
|
|| op == Token::INIT_CONST
|
|
|
|
|| op == Token::ASSIGN)
|
2011-06-22 20:23:48 +00:00
|
|
|
&& (right->AsCall() == NULL && right->AsCallNew() == NULL)) {
|
2010-08-23 13:26:03 +00:00
|
|
|
fni_->Infer();
|
2011-10-03 19:18:05 +00:00
|
|
|
} else {
|
|
|
|
fni_->RemoveLastFunction();
|
2010-08-23 13:26:03 +00:00
|
|
|
}
|
|
|
|
fni_->Leave();
|
|
|
|
}
|
|
|
|
|
2012-02-08 09:56:33 +00:00
|
|
|
return factory()->NewAssignment(op, expression, right, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-02 17:34:59 +00:00
|
|
|
Expression* Parser::ParseYieldExpression(bool* ok) {
|
|
|
|
// YieldExpression ::
|
|
|
|
// 'yield' '*'? AssignmentExpression
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2013-04-02 17:34:59 +00:00
|
|
|
Expect(Token::YIELD, CHECK_OK);
|
2013-04-19 14:11:23 +00:00
|
|
|
Yield::Kind kind =
|
2013-08-20 06:39:04 +00:00
|
|
|
Check(Token::MUL) ? Yield::DELEGATING : Yield::SUSPEND;
|
2013-04-15 12:29:44 +00:00
|
|
|
Expression* generator_object = factory()->NewVariableProxy(
|
|
|
|
current_function_state_->generator_object_variable());
|
2013-04-02 17:34:59 +00:00
|
|
|
Expression* expression = ParseAssignmentExpression(false, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
Yield* yield = factory()->NewYield(generator_object, expression, kind, pos);
|
2013-08-20 06:39:04 +00:00
|
|
|
if (kind == Yield::DELEGATING) {
|
2013-05-14 16:26:56 +00:00
|
|
|
yield->set_index(current_function_state_->NextHandlerIndex());
|
|
|
|
}
|
|
|
|
return yield;
|
2013-04-02 17:34:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Precedence = 3
|
|
|
|
Expression* Parser::ParseConditionalExpression(bool accept_IN, bool* ok) {
|
|
|
|
// ConditionalExpression ::
|
|
|
|
// LogicalOrExpression
|
|
|
|
// LogicalOrExpression '?' AssignmentExpression ':' AssignmentExpression
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
// We start using the binary expression parser for prec >= 4 only!
|
|
|
|
Expression* expression = ParseBinaryExpression(4, accept_IN, CHECK_OK);
|
|
|
|
if (peek() != Token::CONDITIONAL) return expression;
|
|
|
|
Consume(Token::CONDITIONAL);
|
|
|
|
// In parsing the first assignment expression in conditional
|
|
|
|
// expressions we always accept the 'in' keyword; see ECMA-262,
|
|
|
|
// section 11.12, page 58.
|
|
|
|
Expression* left = ParseAssignmentExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::COLON, CHECK_OK);
|
|
|
|
Expression* right = ParseAssignmentExpression(accept_IN, CHECK_OK);
|
2013-10-14 09:41:41 +00:00
|
|
|
return factory()->NewConditional(expression, left, right, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int Precedence(Token::Value tok, bool accept_IN) {
|
|
|
|
if (tok == Token::IN && !accept_IN)
|
|
|
|
return 0; // 0 precedence will terminate binary expression parsing
|
|
|
|
|
|
|
|
return Token::Precedence(tok);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Precedence >= 4
|
|
|
|
Expression* Parser::ParseBinaryExpression(int prec, bool accept_IN, bool* ok) {
|
|
|
|
ASSERT(prec >= 4);
|
|
|
|
Expression* x = ParseUnaryExpression(CHECK_OK);
|
|
|
|
for (int prec1 = Precedence(peek(), accept_IN); prec1 >= prec; prec1--) {
|
|
|
|
// prec1 >= 4
|
|
|
|
while (Precedence(peek(), accept_IN) == prec1) {
|
|
|
|
Token::Value op = Next();
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* y = ParseBinaryExpression(prec1 + 1, accept_IN, CHECK_OK);
|
|
|
|
|
|
|
|
// Compute some expressions involving only number literals.
|
2013-06-24 10:37:59 +00:00
|
|
|
if (x && x->AsLiteral() && x->AsLiteral()->value()->IsNumber() &&
|
|
|
|
y && y->AsLiteral() && y->AsLiteral()->value()->IsNumber()) {
|
|
|
|
double x_val = x->AsLiteral()->value()->Number();
|
|
|
|
double y_val = y->AsLiteral()->value()->Number();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
switch (op) {
|
|
|
|
case Token::ADD:
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(x_val + y_val, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
case Token::SUB:
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(x_val - y_val, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
case Token::MUL:
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(x_val * y_val, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
case Token::DIV:
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(x_val / y_val, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
2012-02-08 09:56:33 +00:00
|
|
|
case Token::BIT_OR: {
|
|
|
|
int value = DoubleToInt32(x_val) | DoubleToInt32(y_val);
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
2012-02-08 09:56:33 +00:00
|
|
|
}
|
|
|
|
case Token::BIT_AND: {
|
|
|
|
int value = DoubleToInt32(x_val) & DoubleToInt32(y_val);
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
2012-02-08 09:56:33 +00:00
|
|
|
}
|
|
|
|
case Token::BIT_XOR: {
|
|
|
|
int value = DoubleToInt32(x_val) ^ DoubleToInt32(y_val);
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
2012-02-08 09:56:33 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
case Token::SHL: {
|
|
|
|
int value = DoubleToInt32(x_val) << (DoubleToInt32(y_val) & 0x1f);
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case Token::SHR: {
|
|
|
|
uint32_t shift = DoubleToInt32(y_val) & 0x1f;
|
|
|
|
uint32_t value = DoubleToUint32(x_val) >> shift;
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case Token::SAR: {
|
|
|
|
uint32_t shift = DoubleToInt32(y_val) & 0x1f;
|
|
|
|
int value = ArithmeticShiftRight(DoubleToInt32(x_val), shift);
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// For now we distinguish between comparisons and other binary
|
|
|
|
// operations. (We could combine the two and get rid of this
|
2010-08-24 07:26:49 +00:00
|
|
|
// code and AST node eventually.)
|
2008-07-03 15:10:15 +00:00
|
|
|
if (Token::IsCompareOp(op)) {
|
|
|
|
// We have a comparison.
|
|
|
|
Token::Value cmp = op;
|
|
|
|
switch (op) {
|
|
|
|
case Token::NE: cmp = Token::EQ; break;
|
|
|
|
case Token::NE_STRICT: cmp = Token::EQ_STRICT; break;
|
|
|
|
default: break;
|
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewCompareOperation(cmp, x, y, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (cmp != op) {
|
|
|
|
// The comparison was negated - add a NOT.
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewUnaryOperation(Token::NOT, x, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// We have a "normal" binary operation.
|
2013-10-14 09:24:58 +00:00
|
|
|
x = factory()->NewBinaryOperation(op, x, y, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::ParseUnaryExpression(bool* ok) {
|
|
|
|
// UnaryExpression ::
|
|
|
|
// PostfixExpression
|
|
|
|
// 'delete' UnaryExpression
|
|
|
|
// 'void' UnaryExpression
|
|
|
|
// 'typeof' UnaryExpression
|
|
|
|
// '++' UnaryExpression
|
|
|
|
// '--' UnaryExpression
|
|
|
|
// '+' UnaryExpression
|
|
|
|
// '-' UnaryExpression
|
|
|
|
// '~' UnaryExpression
|
|
|
|
// '!' UnaryExpression
|
|
|
|
|
|
|
|
Token::Value op = peek();
|
|
|
|
if (Token::IsUnaryOp(op)) {
|
|
|
|
op = Next();
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2009-09-30 09:49:36 +00:00
|
|
|
Expression* expression = ParseUnaryExpression(CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-06-20 07:40:42 +00:00
|
|
|
if (expression != NULL && (expression->AsLiteral() != NULL)) {
|
2013-06-24 10:37:59 +00:00
|
|
|
Handle<Object> literal = expression->AsLiteral()->value();
|
2011-06-20 07:40:42 +00:00
|
|
|
if (op == Token::NOT) {
|
|
|
|
// Convert the literal to a boolean condition and negate it.
|
2013-03-07 16:22:19 +00:00
|
|
|
bool condition = literal->BooleanValue();
|
2013-10-01 09:27:03 +00:00
|
|
|
Handle<Object> result = isolate()->factory()->ToBoolean(!condition);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewLiteral(result, pos);
|
2011-06-20 07:40:42 +00:00
|
|
|
} else if (literal->IsNumber()) {
|
|
|
|
// Compute some expressions involving only number literals.
|
|
|
|
double value = literal->Number();
|
|
|
|
switch (op) {
|
|
|
|
case Token::ADD:
|
|
|
|
return expression;
|
|
|
|
case Token::SUB:
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewNumberLiteral(-value, pos);
|
2011-06-20 07:40:42 +00:00
|
|
|
case Token::BIT_NOT:
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewNumberLiteral(~DoubleToInt32(value), pos);
|
2011-06-20 07:40:42 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-14 18:44:26 +00:00
|
|
|
// "delete identifier" is a syntax error in strict mode.
|
2011-11-24 15:17:04 +00:00
|
|
|
if (op == Token::DELETE && !top_scope_->is_classic_mode()) {
|
2011-02-14 18:44:26 +00:00
|
|
|
VariableProxy* operand = expression->AsVariableProxy();
|
|
|
|
if (operand != NULL && !operand->is_this()) {
|
|
|
|
ReportMessage("strict_delete", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-17 12:47:15 +00:00
|
|
|
// Desugar '+foo' into 'foo*1', this enables the collection of type feedback
|
|
|
|
// without any special stub and the multiplication is removed later in
|
|
|
|
// Crankshaft's canonicalization pass.
|
|
|
|
if (op == Token::ADD) {
|
|
|
|
return factory()->NewBinaryOperation(Token::MUL,
|
|
|
|
expression,
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewNumberLiteral(1, pos),
|
|
|
|
pos);
|
2013-04-17 12:47:15 +00:00
|
|
|
}
|
2013-08-02 11:56:35 +00:00
|
|
|
// The same idea for '-foo' => 'foo*(-1)'.
|
|
|
|
if (op == Token::SUB) {
|
|
|
|
return factory()->NewBinaryOperation(Token::MUL,
|
|
|
|
expression,
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewNumberLiteral(-1, pos),
|
|
|
|
pos);
|
2013-08-02 11:56:35 +00:00
|
|
|
}
|
2013-08-06 13:34:51 +00:00
|
|
|
// ...and one more time for '~foo' => 'foo^(~0)'.
|
|
|
|
if (op == Token::BIT_NOT) {
|
|
|
|
return factory()->NewBinaryOperation(Token::BIT_XOR,
|
|
|
|
expression,
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewNumberLiteral(~0, pos),
|
|
|
|
pos);
|
2013-08-06 13:34:51 +00:00
|
|
|
}
|
2013-04-17 12:47:15 +00:00
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewUnaryOperation(op, expression, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
} else if (Token::IsCountOp(op)) {
|
|
|
|
op = Next();
|
2009-09-30 09:49:36 +00:00
|
|
|
Expression* expression = ParseUnaryExpression(CHECK_OK);
|
|
|
|
// Signal a reference error if the expression is an invalid
|
|
|
|
// left-hand side expression. We could report this as a syntax
|
|
|
|
// error here but for compatibility with JSC we choose to report the
|
|
|
|
// error at runtime.
|
|
|
|
if (expression == NULL || !expression->IsValidLeftHandSide()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message =
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->invalid_lhs_in_prefix_op_string();
|
2013-06-06 13:28:22 +00:00
|
|
|
expression = NewThrowReferenceError(message);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-01-26 19:21:46 +00:00
|
|
|
// Prefix expression operand in strict mode may not be eval or arguments.
|
|
|
|
CheckStrictModeLValue(expression, "strict_lhs_prefix", CHECK_OK);
|
|
|
|
}
|
2011-12-05 14:43:28 +00:00
|
|
|
MarkAsLValue(expression);
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2012-02-08 09:56:33 +00:00
|
|
|
return factory()->NewCountOperation(op,
|
|
|
|
true /* prefix */,
|
|
|
|
expression,
|
2013-10-14 09:24:58 +00:00
|
|
|
position()); // TODO(rossberg): ???
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
return ParsePostfixExpression(ok);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::ParsePostfixExpression(bool* ok) {
|
|
|
|
// PostfixExpression ::
|
|
|
|
// LeftHandSideExpression ('++' | '--')?
|
|
|
|
|
2009-09-30 09:49:36 +00:00
|
|
|
Expression* expression = ParseLeftHandSideExpression(CHECK_OK);
|
2011-06-21 13:34:16 +00:00
|
|
|
if (!scanner().HasAnyLineTerminatorBeforeNext() &&
|
2010-12-07 14:03:59 +00:00
|
|
|
Token::IsCountOp(peek())) {
|
2009-09-30 09:49:36 +00:00
|
|
|
// Signal a reference error if the expression is an invalid
|
|
|
|
// left-hand side expression. We could report this as a syntax
|
|
|
|
// error here but for compatibility with JSC we choose to report the
|
|
|
|
// error at runtime.
|
|
|
|
if (expression == NULL || !expression->IsValidLeftHandSide()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message =
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->invalid_lhs_in_postfix_op_string();
|
2013-06-06 13:28:22 +00:00
|
|
|
expression = NewThrowReferenceError(message);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-01-26 19:21:46 +00:00
|
|
|
// Postfix expression operand in strict mode may not be eval or arguments.
|
|
|
|
CheckStrictModeLValue(expression, "strict_lhs_prefix", CHECK_OK);
|
|
|
|
}
|
2011-12-05 14:43:28 +00:00
|
|
|
MarkAsLValue(expression);
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Token::Value next = Next();
|
2011-04-04 06:29:02 +00:00
|
|
|
expression =
|
2012-02-08 09:56:33 +00:00
|
|
|
factory()->NewCountOperation(next,
|
|
|
|
false /* postfix */,
|
|
|
|
expression,
|
2013-10-14 09:24:58 +00:00
|
|
|
position()); // TODO(rossberg): ???
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2009-09-30 09:49:36 +00:00
|
|
|
return expression;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::ParseLeftHandSideExpression(bool* ok) {
|
|
|
|
// LeftHandSideExpression ::
|
|
|
|
// (NewExpression | MemberExpression) ...
|
|
|
|
|
|
|
|
Expression* result;
|
|
|
|
if (peek() == Token::NEW) {
|
|
|
|
result = ParseNewExpression(CHECK_OK);
|
|
|
|
} else {
|
|
|
|
result = ParseMemberExpression(CHECK_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::LBRACK: {
|
|
|
|
Consume(Token::LBRACK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* index = ParseExpression(true, CHECK_OK);
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewProperty(result, index, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::RBRACK, CHECK_OK);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::LPAREN: {
|
2012-01-06 10:26:17 +00:00
|
|
|
int pos;
|
|
|
|
if (scanner().current_token() == Token::IDENTIFIER) {
|
|
|
|
// For call of an identifier we want to report position of
|
|
|
|
// the identifier as position of the call in the stack trace.
|
2013-10-14 09:24:58 +00:00
|
|
|
pos = position();
|
2012-01-06 10:26:17 +00:00
|
|
|
} else {
|
|
|
|
// For other kinds of calls we record position of the parenthesis as
|
|
|
|
// position of the call. Note that this is extremely important for
|
|
|
|
// expressions of the form function(){...}() for which call position
|
|
|
|
// should not point to the closing brace otherwise it will intersect
|
|
|
|
// with positions recorded for function literal and confuse debugger.
|
2013-10-14 09:24:58 +00:00
|
|
|
pos = peek_position();
|
2012-08-16 11:54:48 +00:00
|
|
|
// Also the trailing parenthesis are a hint that the function will
|
|
|
|
// be called immediately. If we happen to have parsed a preceding
|
|
|
|
// function literal eagerly, we can also compile it eagerly.
|
|
|
|
if (result->IsFunctionLiteral() && mode() == PARSE_EAGERLY) {
|
|
|
|
result->AsFunctionLiteral()->set_parenthesized();
|
|
|
|
}
|
2012-01-06 10:26:17 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
ZoneList<Expression*>* args = ParseArguments(CHECK_OK);
|
|
|
|
|
|
|
|
// Keep track of eval() calls since they disable all local variable
|
2008-11-27 13:55:06 +00:00
|
|
|
// optimizations.
|
|
|
|
// The calls that need special treatment are the
|
2011-10-31 09:38:52 +00:00
|
|
|
// direct eval calls. These calls are all of the form eval(...), with
|
|
|
|
// no explicit receiver.
|
2011-01-14 10:50:13 +00:00
|
|
|
// These calls are marked as potentially direct eval calls. Whether
|
|
|
|
// they are actually direct calls to eval is determined at run time.
|
2010-11-02 11:45:47 +00:00
|
|
|
VariableProxy* callee = result->AsVariableProxy();
|
2011-03-18 20:35:07 +00:00
|
|
|
if (callee != NULL &&
|
2013-02-28 17:03:34 +00:00
|
|
|
callee->IsVariable(isolate()->factory()->eval_string())) {
|
2011-10-31 09:38:52 +00:00
|
|
|
top_scope_->DeclarationScope()->RecordEvalCall();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewCall(result, args, pos);
|
2013-05-31 12:52:28 +00:00
|
|
|
if (fni_ != NULL) fni_->RemoveLastFunction();
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::PERIOD: {
|
|
|
|
Consume(Token::PERIOD);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2010-08-06 08:03:44 +00:00
|
|
|
Handle<String> name = ParseIdentifierName(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewProperty(
|
|
|
|
result, factory()->NewLiteral(name, pos), pos);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushLiteralName(name);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
Expression* Parser::ParseNewPrefix(PositionStack* stack, bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// NewExpression ::
|
|
|
|
// ('new')+ MemberExpression
|
|
|
|
|
|
|
|
// The grammar for new expressions is pretty warped. The keyword
|
|
|
|
// 'new' can either be a part of the new expression (where it isn't
|
|
|
|
// followed by an argument list) or a part of the member expression,
|
|
|
|
// where it must be followed by an argument list. To accommodate
|
|
|
|
// this, we parse the 'new' keywords greedily and keep track of how
|
|
|
|
// many we have parsed. This information is then passed on to the
|
|
|
|
// member expression parser, which is only allowed to match argument
|
|
|
|
// lists as long as it has 'new' prefixes left
|
2009-05-15 14:58:02 +00:00
|
|
|
Expect(Token::NEW, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
PositionStack::Element pos(stack, position());
|
2009-05-15 14:58:02 +00:00
|
|
|
|
|
|
|
Expression* result;
|
|
|
|
if (peek() == Token::NEW) {
|
|
|
|
result = ParseNewPrefix(stack, CHECK_OK);
|
|
|
|
} else {
|
|
|
|
result = ParseMemberWithNewPrefixesExpression(stack, CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
if (!stack->is_empty()) {
|
|
|
|
int last = stack->pop();
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewCallNew(
|
2012-06-11 12:42:31 +00:00
|
|
|
result, new(zone()) ZoneList<Expression*>(0, zone()), last);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
Expression* Parser::ParseNewExpression(bool* ok) {
|
|
|
|
PositionStack stack(ok);
|
|
|
|
return ParseNewPrefix(&stack, ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* Parser::ParseMemberExpression(bool* ok) {
|
2009-05-15 14:58:02 +00:00
|
|
|
return ParseMemberWithNewPrefixesExpression(NULL, ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-15 14:58:02 +00:00
|
|
|
Expression* Parser::ParseMemberWithNewPrefixesExpression(PositionStack* stack,
|
|
|
|
bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// MemberExpression ::
|
|
|
|
// (PrimaryExpression | FunctionLiteral)
|
|
|
|
// ('[' Expression ']' | '.' Identifier | Arguments)*
|
|
|
|
|
|
|
|
// Parse the initial primary or function expression.
|
|
|
|
Expression* result = NULL;
|
|
|
|
if (peek() == Token::FUNCTION) {
|
|
|
|
Expect(Token::FUNCTION, CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int function_token_position = position();
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
bool is_generator = allow_generators() && Check(Token::MUL);
|
2008-07-03 15:10:15 +00:00
|
|
|
Handle<String> name;
|
2011-06-24 14:59:51 +00:00
|
|
|
bool is_strict_reserved_name = false;
|
2011-02-04 18:36:37 +00:00
|
|
|
if (peek_any_identifier()) {
|
2011-06-24 14:59:51 +00:00
|
|
|
name = ParseIdentifierOrStrictReservedWord(&is_strict_reserved_name,
|
|
|
|
CHECK_OK);
|
2011-02-04 18:36:37 +00:00
|
|
|
}
|
2013-06-06 13:28:22 +00:00
|
|
|
FunctionLiteral::FunctionType function_type = name.is_null()
|
2011-08-09 12:43:08 +00:00
|
|
|
? FunctionLiteral::ANONYMOUS_EXPRESSION
|
|
|
|
: FunctionLiteral::NAMED_EXPRESSION;
|
2011-08-08 16:14:46 +00:00
|
|
|
result = ParseFunctionLiteral(name,
|
|
|
|
is_strict_reserved_name,
|
2013-04-02 17:34:59 +00:00
|
|
|
is_generator,
|
2011-08-08 16:14:46 +00:00
|
|
|
function_token_position,
|
2013-06-06 13:28:22 +00:00
|
|
|
function_type,
|
2011-08-08 16:14:46 +00:00
|
|
|
CHECK_OK);
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
result = ParsePrimaryExpression(CHECK_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::LBRACK: {
|
|
|
|
Consume(Token::LBRACK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* index = ParseExpression(true, CHECK_OK);
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewProperty(result, index, pos);
|
2011-06-22 20:23:48 +00:00
|
|
|
if (fni_ != NULL) {
|
|
|
|
if (index->IsPropertyName()) {
|
|
|
|
fni_->PushLiteralName(index->AsLiteral()->AsPropertyName());
|
|
|
|
} else {
|
|
|
|
fni_->PushLiteralName(
|
2013-02-28 17:03:34 +00:00
|
|
|
isolate()->factory()->anonymous_function_string());
|
2011-06-22 20:23:48 +00:00
|
|
|
}
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::RBRACK, CHECK_OK);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Token::PERIOD: {
|
|
|
|
Consume(Token::PERIOD);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2010-08-06 08:03:44 +00:00
|
|
|
Handle<String> name = ParseIdentifierName(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewProperty(
|
|
|
|
result, factory()->NewLiteral(name, pos), pos);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushLiteralName(name);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Token::LPAREN: {
|
2009-05-15 14:58:02 +00:00
|
|
|
if ((stack == NULL) || stack->is_empty()) return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
// Consume one of the new prefixes (already parsed).
|
|
|
|
ZoneList<Expression*>* args = ParseArguments(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = stack->pop();
|
|
|
|
result = factory()->NewCallNew(result, args, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DebuggerStatement* Parser::ParseDebuggerStatement(bool* ok) {
|
|
|
|
// In ECMA-262 'debugger' is defined as a reserved keyword. In some browser
|
|
|
|
// contexts this is used as a statement which invokes the debugger as i a
|
|
|
|
// break point is present.
|
|
|
|
// DebuggerStatement ::
|
|
|
|
// 'debugger' ';'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::DEBUGGER, CHECK_OK);
|
|
|
|
ExpectSemicolon(CHECK_OK);
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewDebuggerStatement(pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Parser::ReportUnexpectedToken(Token::Value token) {
|
|
|
|
// We don't report stack overflows here, to avoid increasing the
|
|
|
|
// stack depth even further. Instead we report it after parsing is
|
2010-02-01 10:31:55 +00:00
|
|
|
// over, in ParseProgram/ParseJson.
|
2010-11-29 13:24:37 +00:00
|
|
|
if (token == Token::ILLEGAL && stack_overflow_) return;
|
2008-07-03 15:10:15 +00:00
|
|
|
// Four of the tokens are treated specially
|
|
|
|
switch (token) {
|
2010-11-29 13:24:37 +00:00
|
|
|
case Token::EOS:
|
|
|
|
return ReportMessage("unexpected_eos", Vector<const char*>::empty());
|
|
|
|
case Token::NUMBER:
|
|
|
|
return ReportMessage("unexpected_token_number",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
case Token::STRING:
|
|
|
|
return ReportMessage("unexpected_token_string",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
case Token::IDENTIFIER:
|
|
|
|
return ReportMessage("unexpected_token_identifier",
|
|
|
|
Vector<const char*>::empty());
|
2011-02-04 18:36:37 +00:00
|
|
|
case Token::FUTURE_RESERVED_WORD:
|
2011-06-24 14:59:51 +00:00
|
|
|
return ReportMessage("unexpected_reserved",
|
|
|
|
Vector<const char*>::empty());
|
2013-04-03 16:14:56 +00:00
|
|
|
case Token::YIELD:
|
2011-06-24 14:59:51 +00:00
|
|
|
case Token::FUTURE_STRICT_RESERVED_WORD:
|
2011-11-24 15:17:04 +00:00
|
|
|
return ReportMessage(top_scope_->is_classic_mode() ?
|
|
|
|
"unexpected_token_identifier" :
|
|
|
|
"unexpected_strict_reserved",
|
2011-02-04 18:36:37 +00:00
|
|
|
Vector<const char*>::empty());
|
2010-11-29 13:24:37 +00:00
|
|
|
default:
|
|
|
|
const char* name = Token::String(token);
|
|
|
|
ASSERT(name != NULL);
|
|
|
|
ReportMessage("unexpected_token", Vector<const char*>(&name, 1));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-07 10:28:22 +00:00
|
|
|
void Parser::ReportInvalidPreparseData(Handle<String> name, bool* ok) {
|
2011-09-09 22:39:47 +00:00
|
|
|
SmartArrayPointer<char> name_string = name->ToCString(DISALLOW_NULLS);
|
2010-07-07 10:28:22 +00:00
|
|
|
const char* element[1] = { *name_string };
|
|
|
|
ReportMessage("invalid_preparser_data",
|
|
|
|
Vector<const char*>(element, 1));
|
|
|
|
*ok = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* Parser::ParsePrimaryExpression(bool* ok) {
|
|
|
|
// PrimaryExpression ::
|
|
|
|
// 'this'
|
|
|
|
// 'null'
|
|
|
|
// 'true'
|
|
|
|
// 'false'
|
|
|
|
// Identifier
|
|
|
|
// Number
|
|
|
|
// String
|
|
|
|
// ArrayLiteral
|
|
|
|
// ObjectLiteral
|
|
|
|
// RegExpLiteral
|
|
|
|
// '(' Expression ')'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* result = NULL;
|
|
|
|
switch (peek()) {
|
|
|
|
case Token::THIS: {
|
|
|
|
Consume(Token::THIS);
|
2012-02-08 09:56:33 +00:00
|
|
|
result = factory()->NewVariableProxy(top_scope_->receiver());
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::NULL_LITERAL:
|
|
|
|
Consume(Token::NULL_LITERAL);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewLiteral(isolate()->factory()->null_value(), pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::TRUE_LITERAL:
|
|
|
|
Consume(Token::TRUE_LITERAL);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewLiteral(isolate()->factory()->true_value(), pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::FALSE_LITERAL:
|
|
|
|
Consume(Token::FALSE_LITERAL);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewLiteral(isolate()->factory()->false_value(), pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
|
2011-02-04 18:36:37 +00:00
|
|
|
case Token::IDENTIFIER:
|
2013-04-02 17:34:59 +00:00
|
|
|
case Token::YIELD:
|
2011-06-24 14:59:51 +00:00
|
|
|
case Token::FUTURE_STRICT_RESERVED_WORD: {
|
2008-07-03 15:10:15 +00:00
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushVariableName(name);
|
2012-03-08 13:03:07 +00:00
|
|
|
// The name may refer to a module instance object, so its type is unknown.
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (FLAG_print_interface_details)
|
|
|
|
PrintF("# Variable %s ", name->ToAsciiArray());
|
|
|
|
#endif
|
2012-06-11 12:42:31 +00:00
|
|
|
Interface* interface = Interface::NewUnknown(zone());
|
2013-10-14 09:24:58 +00:00
|
|
|
result = top_scope_->NewUnresolved(factory(), name, interface, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::NUMBER: {
|
|
|
|
Consume(Token::NUMBER);
|
2010-12-22 20:14:19 +00:00
|
|
|
ASSERT(scanner().is_literal_ascii());
|
2011-04-12 08:27:38 +00:00
|
|
|
double value = StringToDouble(isolate()->unicode_cache(),
|
|
|
|
scanner().literal_ascii_string(),
|
2013-07-19 09:57:35 +00:00
|
|
|
ALLOW_HEX | ALLOW_OCTAL |
|
|
|
|
ALLOW_IMPLICIT_OCTAL | ALLOW_BINARY);
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewNumberLiteral(value, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::STRING: {
|
|
|
|
Consume(Token::STRING);
|
2013-05-21 10:45:58 +00:00
|
|
|
Handle<String> symbol = GetSymbol();
|
2013-10-14 09:24:58 +00:00
|
|
|
result = factory()->NewLiteral(symbol, pos);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushLiteralName(symbol);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Token::ASSIGN_DIV:
|
|
|
|
result = ParseRegExpLiteral(true, CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::DIV:
|
|
|
|
result = ParseRegExpLiteral(false, CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::LBRACK:
|
|
|
|
result = ParseArrayLiteral(CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::LBRACE:
|
|
|
|
result = ParseObjectLiteral(CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::LPAREN:
|
|
|
|
Consume(Token::LPAREN);
|
2011-01-14 10:50:13 +00:00
|
|
|
// Heuristically try to detect immediately called functions before
|
|
|
|
// seeing the call parentheses.
|
|
|
|
parenthesized_function_ = (peek() == Token::FUNCTION);
|
2008-07-03 15:10:15 +00:00
|
|
|
result = ParseExpression(true, CHECK_OK);
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Token::MOD:
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
if (allow_natives_syntax() || extension_ != NULL) {
|
2008-07-03 15:10:15 +00:00
|
|
|
result = ParseV8Intrinsic(CHECK_OK);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// If we're not allowing special syntax we fall-through to the
|
|
|
|
// default case.
|
|
|
|
|
|
|
|
default: {
|
2010-08-09 08:54:29 +00:00
|
|
|
Token::Value tok = Next();
|
2008-07-03 15:10:15 +00:00
|
|
|
ReportUnexpectedToken(tok);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::ParseArrayLiteral(bool* ok) {
|
|
|
|
// ArrayLiteral ::
|
|
|
|
// '[' Expression? (',' Expression?)* ']'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Expression*>* values = new(zone()) ZoneList<Expression*>(4, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LBRACK, CHECK_OK);
|
|
|
|
while (peek() != Token::RBRACK) {
|
|
|
|
Expression* elem;
|
|
|
|
if (peek() == Token::COMMA) {
|
2013-10-14 09:24:58 +00:00
|
|
|
elem = GetLiteralTheHole(peek_position());
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
|
|
|
elem = ParseAssignmentExpression(true, CHECK_OK);
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
values->Add(elem, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
if (peek() != Token::RBRACK) {
|
|
|
|
Expect(Token::COMMA, CHECK_OK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expect(Token::RBRACK, CHECK_OK);
|
2008-08-22 13:33:59 +00:00
|
|
|
|
|
|
|
// Update the scope information before the pre-parsing bailout.
|
2011-11-09 13:54:26 +00:00
|
|
|
int literal_index = current_function_state_->NextMaterializedLiteralIndex();
|
2008-08-22 13:33:59 +00:00
|
|
|
|
2011-10-19 11:36:55 +00:00
|
|
|
// Allocate a fixed array to hold all the object literals.
|
2012-11-29 15:55:27 +00:00
|
|
|
Handle<JSArray> array =
|
|
|
|
isolate()->factory()->NewJSArray(0, FAST_HOLEY_SMI_ELEMENTS);
|
|
|
|
isolate()->factory()->SetElementsCapacityAndLength(
|
|
|
|
array, values->length(), values->length());
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Fill in the literals.
|
2012-05-23 14:24:29 +00:00
|
|
|
Heap* heap = isolate()->heap();
|
2009-03-23 07:27:47 +00:00
|
|
|
bool is_simple = true;
|
|
|
|
int depth = 1;
|
2012-11-29 15:55:27 +00:00
|
|
|
bool is_holey = false;
|
2010-11-02 11:45:47 +00:00
|
|
|
for (int i = 0, n = values->length(); i < n; i++) {
|
|
|
|
MaterializedLiteral* m_literal = values->at(i)->AsMaterializedLiteral();
|
2009-03-23 07:27:47 +00:00
|
|
|
if (m_literal != NULL && m_literal->depth() + 1 > depth) {
|
|
|
|
depth = m_literal->depth() + 1;
|
|
|
|
}
|
2010-11-02 11:45:47 +00:00
|
|
|
Handle<Object> boilerplate_value = GetBoilerplateValue(values->at(i));
|
2012-05-23 14:24:29 +00:00
|
|
|
if (boilerplate_value->IsTheHole()) {
|
2012-11-29 15:55:27 +00:00
|
|
|
is_holey = true;
|
2013-06-06 14:21:35 +00:00
|
|
|
} else if (boilerplate_value->IsUninitialized()) {
|
2009-03-23 07:27:47 +00:00
|
|
|
is_simple = false;
|
2012-11-29 15:55:27 +00:00
|
|
|
JSObject::SetOwnElement(
|
|
|
|
array, i, handle(Smi::FromInt(0), isolate()), kNonStrictMode);
|
2008-07-03 15:10:15 +00:00
|
|
|
} else {
|
2012-11-29 15:55:27 +00:00
|
|
|
JSObject::SetOwnElement(array, i, boilerplate_value, kNonStrictMode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-29 15:55:27 +00:00
|
|
|
Handle<FixedArrayBase> element_values(array->elements());
|
|
|
|
|
2010-08-16 16:06:46 +00:00
|
|
|
// Simple and shallow arrays can be lazily copied, we transform the
|
|
|
|
// elements array to a copy-on-write array.
|
2011-10-19 11:36:55 +00:00
|
|
|
if (is_simple && depth == 1 && values->length() > 0 &&
|
2012-11-29 15:55:27 +00:00
|
|
|
array->HasFastSmiOrObjectElements()) {
|
|
|
|
element_values->set_map(heap->fixed_cow_array_map());
|
2010-08-16 16:06:46 +00:00
|
|
|
}
|
|
|
|
|
2011-10-19 11:36:55 +00:00
|
|
|
// Remember both the literal's constant values as well as the ElementsKind
|
|
|
|
// in a 2-element FixedArray.
|
2012-11-29 15:55:27 +00:00
|
|
|
Handle<FixedArray> literals = isolate()->factory()->NewFixedArray(2, TENURED);
|
2011-10-19 11:36:55 +00:00
|
|
|
|
2012-11-29 15:55:27 +00:00
|
|
|
ElementsKind kind = array->GetElementsKind();
|
|
|
|
kind = is_holey ? GetHoleyElementsKind(kind) : GetPackedElementsKind(kind);
|
2012-05-23 14:24:29 +00:00
|
|
|
|
2012-11-29 15:55:27 +00:00
|
|
|
literals->set(0, Smi::FromInt(kind));
|
2011-10-19 11:36:55 +00:00
|
|
|
literals->set(1, *element_values);
|
|
|
|
|
2012-02-08 09:56:33 +00:00
|
|
|
return factory()->NewArrayLiteral(
|
2013-10-14 09:24:58 +00:00
|
|
|
literals, values, literal_index, is_simple, depth, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-05 16:47:42 +00:00
|
|
|
bool Parser::IsBoilerplateProperty(ObjectLiteral::Property* property) {
|
|
|
|
return property != NULL &&
|
|
|
|
property->kind() != ObjectLiteral::Property::PROTOTYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-23 07:27:47 +00:00
|
|
|
bool CompileTimeValue::IsCompileTimeValue(Expression* expression) {
|
2010-12-07 11:31:57 +00:00
|
|
|
if (expression->AsLiteral() != NULL) return true;
|
2009-03-23 07:27:47 +00:00
|
|
|
MaterializedLiteral* lit = expression->AsMaterializedLiteral();
|
|
|
|
return lit != NULL && lit->is_simple();
|
|
|
|
}
|
|
|
|
|
2010-08-16 16:06:46 +00:00
|
|
|
|
2013-09-04 07:05:11 +00:00
|
|
|
Handle<FixedArray> CompileTimeValue::GetValue(Isolate* isolate,
|
|
|
|
Expression* expression) {
|
|
|
|
Factory* factory = isolate->factory();
|
2009-03-23 07:27:47 +00:00
|
|
|
ASSERT(IsCompileTimeValue(expression));
|
2013-06-04 10:30:05 +00:00
|
|
|
Handle<FixedArray> result = factory->NewFixedArray(2, TENURED);
|
2009-03-23 07:27:47 +00:00
|
|
|
ObjectLiteral* object_literal = expression->AsObjectLiteral();
|
|
|
|
if (object_literal != NULL) {
|
|
|
|
ASSERT(object_literal->is_simple());
|
2010-03-11 10:34:29 +00:00
|
|
|
if (object_literal->fast_elements()) {
|
2013-06-06 13:28:22 +00:00
|
|
|
result->set(kLiteralTypeSlot, Smi::FromInt(OBJECT_LITERAL_FAST_ELEMENTS));
|
2010-03-11 10:34:29 +00:00
|
|
|
} else {
|
2013-06-06 13:28:22 +00:00
|
|
|
result->set(kLiteralTypeSlot, Smi::FromInt(OBJECT_LITERAL_SLOW_ELEMENTS));
|
2010-03-11 10:34:29 +00:00
|
|
|
}
|
2009-03-23 07:27:47 +00:00
|
|
|
result->set(kElementsSlot, *object_literal->constant_properties());
|
|
|
|
} else {
|
|
|
|
ArrayLiteral* array_literal = expression->AsArrayLiteral();
|
|
|
|
ASSERT(array_literal != NULL && array_literal->is_simple());
|
2013-06-06 13:28:22 +00:00
|
|
|
result->set(kLiteralTypeSlot, Smi::FromInt(ARRAY_LITERAL));
|
2009-12-22 12:41:45 +00:00
|
|
|
result->set(kElementsSlot, *array_literal->constant_elements());
|
2009-03-23 07:27:47 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
CompileTimeValue::LiteralType CompileTimeValue::GetLiteralType(
|
|
|
|
Handle<FixedArray> value) {
|
|
|
|
Smi* literal_type = Smi::cast(value->get(kLiteralTypeSlot));
|
|
|
|
return static_cast<LiteralType>(literal_type->value());
|
2009-03-23 07:27:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<FixedArray> CompileTimeValue::GetElements(Handle<FixedArray> value) {
|
|
|
|
return Handle<FixedArray>(FixedArray::cast(value->get(kElementsSlot)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Parser::GetBoilerplateValue(Expression* expression) {
|
|
|
|
if (expression->AsLiteral() != NULL) {
|
2013-06-24 10:37:59 +00:00
|
|
|
return expression->AsLiteral()->value();
|
2009-03-23 07:27:47 +00:00
|
|
|
}
|
|
|
|
if (CompileTimeValue::IsCompileTimeValue(expression)) {
|
2013-09-04 07:05:11 +00:00
|
|
|
return CompileTimeValue::GetValue(isolate(), expression);
|
2009-03-23 07:27:47 +00:00
|
|
|
}
|
2013-06-06 14:21:35 +00:00
|
|
|
return isolate()->factory()->uninitialized_value();
|
2008-09-05 16:47:42 +00:00
|
|
|
}
|
|
|
|
|
2013-07-05 09:52:11 +00:00
|
|
|
|
2010-02-01 10:31:55 +00:00
|
|
|
void Parser::BuildObjectLiteralConstantProperties(
|
|
|
|
ZoneList<ObjectLiteral::Property*>* properties,
|
|
|
|
Handle<FixedArray> constant_properties,
|
|
|
|
bool* is_simple,
|
2010-03-11 10:34:29 +00:00
|
|
|
bool* fast_elements,
|
2013-05-08 15:02:08 +00:00
|
|
|
int* depth,
|
|
|
|
bool* may_store_doubles) {
|
2010-02-01 10:31:55 +00:00
|
|
|
int position = 0;
|
|
|
|
// Accumulate the value in local variables and store it at the end.
|
|
|
|
bool is_simple_acc = true;
|
|
|
|
int depth_acc = 1;
|
2010-03-11 10:34:29 +00:00
|
|
|
uint32_t max_element_index = 0;
|
|
|
|
uint32_t elements = 0;
|
2010-02-01 10:31:55 +00:00
|
|
|
for (int i = 0; i < properties->length(); i++) {
|
|
|
|
ObjectLiteral::Property* property = properties->at(i);
|
|
|
|
if (!IsBoilerplateProperty(property)) {
|
|
|
|
is_simple_acc = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
MaterializedLiteral* m_literal = property->value()->AsMaterializedLiteral();
|
|
|
|
if (m_literal != NULL && m_literal->depth() >= depth_acc) {
|
|
|
|
depth_acc = m_literal->depth() + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add CONSTANT and COMPUTED properties to boilerplate. Use undefined
|
|
|
|
// value for COMPUTED properties, the real value is filled in at
|
|
|
|
// runtime. The enumeration order is maintained.
|
2013-06-24 10:37:59 +00:00
|
|
|
Handle<Object> key = property->key()->value();
|
2010-02-01 10:31:55 +00:00
|
|
|
Handle<Object> value = GetBoilerplateValue(property->value());
|
2013-05-08 15:02:08 +00:00
|
|
|
|
2013-06-06 14:21:35 +00:00
|
|
|
// Ensure objects that may, at any point in time, contain fields with double
|
|
|
|
// representation are always treated as nested objects. This is true for
|
|
|
|
// computed fields (value is undefined), and smi and double literals
|
|
|
|
// (value->IsNumber()).
|
2013-05-08 15:02:08 +00:00
|
|
|
// TODO(verwaest): Remove once we can store them inline.
|
2013-06-06 14:21:35 +00:00
|
|
|
if (FLAG_track_double_fields &&
|
|
|
|
(value->IsNumber() || value->IsUninitialized())) {
|
2013-05-08 15:02:08 +00:00
|
|
|
*may_store_doubles = true;
|
|
|
|
}
|
|
|
|
|
2013-06-06 14:21:35 +00:00
|
|
|
is_simple_acc = is_simple_acc && !value->IsUninitialized();
|
2010-02-01 10:31:55 +00:00
|
|
|
|
2010-03-11 10:34:29 +00:00
|
|
|
// Keep track of the number of elements in the object literal and
|
|
|
|
// the largest element index. If the largest element index is
|
|
|
|
// much larger than the number of elements, creating an object
|
|
|
|
// literal with fast elements will be a waste of space.
|
|
|
|
uint32_t element_index = 0;
|
|
|
|
if (key->IsString()
|
|
|
|
&& Handle<String>::cast(key)->AsArrayIndex(&element_index)
|
|
|
|
&& element_index > max_element_index) {
|
|
|
|
max_element_index = element_index;
|
|
|
|
elements++;
|
|
|
|
} else if (key->IsSmi()) {
|
|
|
|
int key_value = Smi::cast(*key)->value();
|
|
|
|
if (key_value > 0
|
|
|
|
&& static_cast<uint32_t>(key_value) > max_element_index) {
|
|
|
|
max_element_index = key_value;
|
|
|
|
}
|
|
|
|
elements++;
|
|
|
|
}
|
|
|
|
|
2010-02-01 10:31:55 +00:00
|
|
|
// Add name, value pair to the fixed array.
|
|
|
|
constant_properties->set(position++, *key);
|
|
|
|
constant_properties->set(position++, *value);
|
|
|
|
}
|
2010-03-11 10:34:29 +00:00
|
|
|
*fast_elements =
|
|
|
|
(max_element_index <= 32) || ((2 * elements) >= max_element_index);
|
2010-02-01 10:31:55 +00:00
|
|
|
*is_simple = is_simple_acc;
|
|
|
|
*depth = depth_acc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-10 11:58:16 +00:00
|
|
|
// Force instantiation of template instances class.
|
|
|
|
template void ObjectLiteralChecker<Parser>::CheckProperty(
|
|
|
|
Token::Value property, PropertyKind type, bool* ok);
|
2010-08-09 08:54:29 +00:00
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* Parser::ParseObjectLiteral(bool* ok) {
|
|
|
|
// ObjectLiteral ::
|
|
|
|
// '{' (
|
2010-08-06 08:03:44 +00:00
|
|
|
// ((IdentifierName | String | Number) ':' AssignmentExpression)
|
|
|
|
// | (('get' | 'set') (IdentifierName | String | Number) FunctionLiteral)
|
2008-07-03 15:10:15 +00:00
|
|
|
// )*[','] '}'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2010-11-02 11:45:47 +00:00
|
|
|
ZoneList<ObjectLiteral::Property*>* properties =
|
2012-06-11 12:42:31 +00:00
|
|
|
new(zone()) ZoneList<ObjectLiteral::Property*>(4, zone());
|
2008-09-04 20:49:18 +00:00
|
|
|
int number_of_boilerplate_properties = 0;
|
2011-03-21 12:25:31 +00:00
|
|
|
bool has_function = false;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-10-10 11:58:16 +00:00
|
|
|
ObjectLiteralChecker<Parser> checker(this, &scanner_,
|
|
|
|
top_scope_->language_mode());
|
2011-01-25 18:42:35 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
2011-01-25 18:42:35 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
while (peek() != Token::RBRACE) {
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->Enter();
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Literal* key = NULL;
|
2010-08-06 08:03:44 +00:00
|
|
|
Token::Value next = peek();
|
2013-10-14 09:24:58 +00:00
|
|
|
int next_pos = peek_position();
|
2011-01-25 18:42:35 +00:00
|
|
|
|
2010-08-06 08:03:44 +00:00
|
|
|
switch (next) {
|
2011-02-04 18:36:37 +00:00
|
|
|
case Token::FUTURE_RESERVED_WORD:
|
2011-06-24 14:59:51 +00:00
|
|
|
case Token::FUTURE_STRICT_RESERVED_WORD:
|
2008-07-03 15:10:15 +00:00
|
|
|
case Token::IDENTIFIER: {
|
|
|
|
bool is_getter = false;
|
|
|
|
bool is_setter = false;
|
|
|
|
Handle<String> id =
|
2011-06-24 14:59:51 +00:00
|
|
|
ParseIdentifierNameOrGetOrSet(&is_getter, &is_setter, CHECK_OK);
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushLiteralName(id);
|
|
|
|
|
2010-08-09 08:54:29 +00:00
|
|
|
if ((is_getter || is_setter) && peek() != Token::COLON) {
|
2013-10-10 11:58:16 +00:00
|
|
|
// Special handling of getter and setter syntax:
|
|
|
|
// { ... , get foo() { ... }, ... , set foo(v) { ... v ... } , ... }
|
|
|
|
// We have already read the "get" or "set" keyword.
|
|
|
|
Token::Value next = Next();
|
|
|
|
bool is_keyword = Token::IsKeyword(next);
|
|
|
|
if (next != i::Token::IDENTIFIER &&
|
|
|
|
next != i::Token::FUTURE_RESERVED_WORD &&
|
|
|
|
next != i::Token::FUTURE_STRICT_RESERVED_WORD &&
|
|
|
|
next != i::Token::NUMBER &&
|
|
|
|
next != i::Token::STRING &&
|
|
|
|
!is_keyword) {
|
|
|
|
// Unexpected token.
|
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
// Validate the property.
|
|
|
|
PropertyKind type = is_getter ? kGetterProperty : kSetterProperty;
|
|
|
|
checker.CheckProperty(next, type, CHECK_OK);
|
|
|
|
Handle<String> name = is_keyword
|
|
|
|
? isolate_->factory()->InternalizeUtf8String(Token::String(next))
|
|
|
|
: GetSymbol();
|
|
|
|
FunctionLiteral* value =
|
|
|
|
ParseFunctionLiteral(name,
|
|
|
|
false, // reserved words are allowed here
|
|
|
|
false, // not a generator
|
|
|
|
RelocInfo::kNoPosition,
|
|
|
|
FunctionLiteral::ANONYMOUS_EXPRESSION,
|
|
|
|
CHECK_OK);
|
|
|
|
// Allow any number of parameters for compatibilty with JSC.
|
|
|
|
// Specification only allows zero parameters for get and one for set.
|
|
|
|
ObjectLiteral::Property* property =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewObjectLiteralProperty(is_getter, value, next_pos);
|
2013-10-10 11:58:16 +00:00
|
|
|
if (IsBoilerplateProperty(property)) {
|
|
|
|
number_of_boilerplate_properties++;
|
|
|
|
}
|
|
|
|
properties->Add(property, zone());
|
|
|
|
if (peek() != Token::RBRACE) Expect(Token::COMMA, CHECK_OK);
|
|
|
|
|
|
|
|
if (fni_ != NULL) {
|
|
|
|
fni_->Infer();
|
|
|
|
fni_->Leave();
|
|
|
|
}
|
|
|
|
continue; // restart the while
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2010-08-09 08:54:29 +00:00
|
|
|
// Failed to parse as get/set property, so it's just a property
|
|
|
|
// called "get" or "set".
|
2013-10-14 09:24:58 +00:00
|
|
|
key = factory()->NewLiteral(id, next_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Token::STRING: {
|
2010-08-09 08:54:29 +00:00
|
|
|
Consume(Token::STRING);
|
2013-05-21 10:45:58 +00:00
|
|
|
Handle<String> string = GetSymbol();
|
2010-08-23 13:26:03 +00:00
|
|
|
if (fni_ != NULL) fni_->PushLiteralName(string);
|
2008-07-03 15:10:15 +00:00
|
|
|
uint32_t index;
|
2010-08-10 12:44:13 +00:00
|
|
|
if (!string.is_null() && string->AsArrayIndex(&index)) {
|
2013-10-14 09:24:58 +00:00
|
|
|
key = factory()->NewNumberLiteral(index, next_pos);
|
2010-08-09 08:54:29 +00:00
|
|
|
break;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-10-14 09:24:58 +00:00
|
|
|
key = factory()->NewLiteral(string, next_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Token::NUMBER: {
|
|
|
|
Consume(Token::NUMBER);
|
2010-12-22 20:14:19 +00:00
|
|
|
ASSERT(scanner().is_literal_ascii());
|
2011-04-12 08:27:38 +00:00
|
|
|
double value = StringToDouble(isolate()->unicode_cache(),
|
|
|
|
scanner().literal_ascii_string(),
|
2013-07-19 09:57:35 +00:00
|
|
|
ALLOW_HEX | ALLOW_OCTAL |
|
|
|
|
ALLOW_IMPLICIT_OCTAL | ALLOW_BINARY);
|
2013-10-14 09:24:58 +00:00
|
|
|
key = factory()->NewNumberLiteral(value, next_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2010-08-09 08:54:29 +00:00
|
|
|
if (Token::IsKeyword(next)) {
|
|
|
|
Consume(next);
|
2013-05-21 10:45:58 +00:00
|
|
|
Handle<String> string = GetSymbol();
|
2013-10-14 09:24:58 +00:00
|
|
|
key = factory()->NewLiteral(string, next_pos);
|
2010-08-09 08:54:29 +00:00
|
|
|
} else {
|
|
|
|
// Unexpected token.
|
|
|
|
Token::Value next = Next();
|
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2013-10-10 11:58:16 +00:00
|
|
|
// Validate the property
|
|
|
|
checker.CheckProperty(next, kValueProperty, CHECK_OK);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::COLON, CHECK_OK);
|
|
|
|
Expression* value = ParseAssignmentExpression(true, CHECK_OK);
|
|
|
|
|
|
|
|
ObjectLiteral::Property* property =
|
2012-02-28 15:32:06 +00:00
|
|
|
new(zone()) ObjectLiteral::Property(key, value, isolate());
|
2008-09-05 09:41:30 +00:00
|
|
|
|
2011-11-30 16:00:47 +00:00
|
|
|
// Mark top-level object literals that contain function literals and
|
|
|
|
// pretenure the literal so it can be added as a constant function
|
|
|
|
// property.
|
|
|
|
if (top_scope_->DeclarationScope()->is_global_scope() &&
|
|
|
|
value->AsFunctionLiteral() != NULL) {
|
2011-03-21 12:25:31 +00:00
|
|
|
has_function = true;
|
2011-11-09 13:54:26 +00:00
|
|
|
value->AsFunctionLiteral()->set_pretenure();
|
2011-03-21 12:25:31 +00:00
|
|
|
}
|
|
|
|
|
2008-09-04 20:49:18 +00:00
|
|
|
// Count CONSTANT or COMPUTED properties to maintain the enumeration order.
|
2008-09-25 07:46:07 +00:00
|
|
|
if (IsBoilerplateProperty(property)) number_of_boilerplate_properties++;
|
2012-06-11 12:42:31 +00:00
|
|
|
properties->Add(property, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// TODO(1240767): Consider allowing trailing comma.
|
|
|
|
if (peek() != Token::RBRACE) Expect(Token::COMMA, CHECK_OK);
|
2010-08-23 13:26:03 +00:00
|
|
|
|
|
|
|
if (fni_ != NULL) {
|
|
|
|
fni_->Infer();
|
|
|
|
fni_->Leave();
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
2011-01-25 18:42:35 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Computation of literal_index must happen before pre parse bailout.
|
2011-11-09 13:54:26 +00:00
|
|
|
int literal_index = current_function_state_->NextMaterializedLiteralIndex();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<FixedArray> constant_properties = isolate()->factory()->NewFixedArray(
|
|
|
|
number_of_boilerplate_properties * 2, TENURED);
|
2010-02-01 10:31:55 +00:00
|
|
|
|
2009-03-23 07:27:47 +00:00
|
|
|
bool is_simple = true;
|
2010-03-11 10:34:29 +00:00
|
|
|
bool fast_elements = true;
|
2009-03-23 07:27:47 +00:00
|
|
|
int depth = 1;
|
2013-05-08 15:02:08 +00:00
|
|
|
bool may_store_doubles = false;
|
2010-11-02 11:45:47 +00:00
|
|
|
BuildObjectLiteralConstantProperties(properties,
|
2010-02-01 10:31:55 +00:00
|
|
|
constant_properties,
|
|
|
|
&is_simple,
|
2010-03-11 10:34:29 +00:00
|
|
|
&fast_elements,
|
2013-05-08 15:02:08 +00:00
|
|
|
&depth,
|
|
|
|
&may_store_doubles);
|
2012-02-08 09:56:33 +00:00
|
|
|
return factory()->NewObjectLiteral(constant_properties,
|
|
|
|
properties,
|
|
|
|
literal_index,
|
|
|
|
is_simple,
|
|
|
|
fast_elements,
|
|
|
|
depth,
|
2013-05-08 15:02:08 +00:00
|
|
|
may_store_doubles,
|
2013-10-14 09:24:58 +00:00
|
|
|
has_function,
|
|
|
|
pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::ParseRegExpLiteral(bool seen_equal, bool* ok) {
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2010-12-07 14:03:59 +00:00
|
|
|
if (!scanner().ScanRegExpPattern(seen_equal)) {
|
2008-07-03 15:10:15 +00:00
|
|
|
Next();
|
|
|
|
ReportMessage("unterminated_regexp", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
int literal_index = current_function_state_->NextMaterializedLiteralIndex();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-12-22 20:14:19 +00:00
|
|
|
Handle<String> js_pattern = NextLiteralString(TENURED);
|
2010-12-07 14:03:59 +00:00
|
|
|
scanner().ScanRegExpFlags();
|
2010-12-22 20:14:19 +00:00
|
|
|
Handle<String> js_flags = NextLiteralString(TENURED);
|
2008-07-03 15:10:15 +00:00
|
|
|
Next();
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewRegExpLiteral(js_pattern, js_flags, literal_index, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ZoneList<Expression*>* Parser::ParseArguments(bool* ok) {
|
|
|
|
// Arguments ::
|
|
|
|
// '(' (AssignmentExpression)*[','] ')'
|
|
|
|
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Expression*>* result = new(zone()) ZoneList<Expression*>(4, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
|
|
|
bool done = (peek() == Token::RPAREN);
|
|
|
|
while (!done) {
|
|
|
|
Expression* argument = ParseAssignmentExpression(true, CHECK_OK);
|
2012-06-11 12:42:31 +00:00
|
|
|
result->Add(argument, zone());
|
2013-03-18 13:35:17 +00:00
|
|
|
if (result->length() > Code::kMaxArguments) {
|
2011-06-07 08:15:47 +00:00
|
|
|
ReportMessageAt(scanner().location(), "too_many_arguments",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
done = (peek() == Token::RPAREN);
|
|
|
|
if (!done) Expect(Token::COMMA, CHECK_OK);
|
|
|
|
}
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
2010-11-02 11:45:47 +00:00
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-25 09:36:31 +00:00
|
|
|
class SingletonLogger : public ParserRecorder {
|
|
|
|
public:
|
|
|
|
SingletonLogger() : has_error_(false), start_(-1), end_(-1) { }
|
2012-02-08 09:56:33 +00:00
|
|
|
virtual ~SingletonLogger() { }
|
2011-11-25 09:36:31 +00:00
|
|
|
|
|
|
|
void Reset() { has_error_ = false; }
|
|
|
|
|
|
|
|
virtual void LogFunction(int start,
|
|
|
|
int end,
|
|
|
|
int literals,
|
|
|
|
int properties,
|
|
|
|
LanguageMode mode) {
|
|
|
|
ASSERT(!has_error_);
|
|
|
|
start_ = start;
|
|
|
|
end_ = end;
|
|
|
|
literals_ = literals;
|
|
|
|
properties_ = properties;
|
|
|
|
mode_ = mode;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Logs a symbol creation of a literal or identifier.
|
|
|
|
virtual void LogAsciiSymbol(int start, Vector<const char> literal) { }
|
2012-03-12 12:35:28 +00:00
|
|
|
virtual void LogUtf16Symbol(int start, Vector<const uc16> literal) { }
|
2011-11-25 09:36:31 +00:00
|
|
|
|
|
|
|
// 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) {
|
2012-07-10 12:24:17 +00:00
|
|
|
if (has_error_) return;
|
2011-11-25 09:36:31 +00:00
|
|
|
has_error_ = true;
|
|
|
|
start_ = start;
|
|
|
|
end_ = end;
|
|
|
|
message_ = message;
|
|
|
|
argument_opt_ = argument_opt;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual int function_position() { return 0; }
|
|
|
|
|
|
|
|
virtual int symbol_position() { return 0; }
|
|
|
|
|
|
|
|
virtual int symbol_ids() { return -1; }
|
|
|
|
|
|
|
|
virtual Vector<unsigned> ExtractData() {
|
|
|
|
UNREACHABLE();
|
|
|
|
return Vector<unsigned>();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void PauseRecording() { }
|
|
|
|
|
|
|
|
virtual void ResumeRecording() { }
|
|
|
|
|
|
|
|
bool has_error() { return has_error_; }
|
|
|
|
|
|
|
|
int start() { return start_; }
|
|
|
|
int end() { return end_; }
|
|
|
|
int literals() {
|
|
|
|
ASSERT(!has_error_);
|
|
|
|
return literals_;
|
|
|
|
}
|
|
|
|
int properties() {
|
|
|
|
ASSERT(!has_error_);
|
|
|
|
return properties_;
|
|
|
|
}
|
|
|
|
LanguageMode language_mode() {
|
|
|
|
ASSERT(!has_error_);
|
|
|
|
return mode_;
|
|
|
|
}
|
|
|
|
const char* message() {
|
|
|
|
ASSERT(has_error_);
|
|
|
|
return message_;
|
|
|
|
}
|
|
|
|
const char* argument_opt() {
|
|
|
|
ASSERT(has_error_);
|
|
|
|
return argument_opt_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool has_error_;
|
|
|
|
int start_;
|
|
|
|
int end_;
|
|
|
|
// For function entries.
|
|
|
|
int literals_;
|
|
|
|
int properties_;
|
|
|
|
LanguageMode mode_;
|
|
|
|
// For error messages.
|
|
|
|
const char* message_;
|
|
|
|
const char* argument_opt_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
FunctionLiteral* Parser::ParseFunctionLiteral(
|
|
|
|
Handle<String> function_name,
|
|
|
|
bool name_is_strict_reserved,
|
|
|
|
bool is_generator,
|
2013-10-14 09:24:58 +00:00
|
|
|
int function_token_pos,
|
2013-06-06 13:28:22 +00:00
|
|
|
FunctionLiteral::FunctionType function_type,
|
|
|
|
bool* ok) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Function ::
|
|
|
|
// '(' FormalParameterList? ')' '{' FunctionBody '}'
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = function_token_pos == RelocInfo::kNoPosition
|
|
|
|
? peek_position() : function_token_pos;
|
|
|
|
|
2011-08-08 16:14:46 +00:00
|
|
|
// Anonymous functions were passed either the empty symbol or a null
|
|
|
|
// handle as the function name. Remember if we were passed a non-empty
|
|
|
|
// handle to decide whether to invoke function name inference.
|
|
|
|
bool should_infer_name = function_name.is_null();
|
|
|
|
|
|
|
|
// We want a non-null handle as the function name.
|
|
|
|
if (should_infer_name) {
|
2013-02-28 17:03:34 +00:00
|
|
|
function_name = isolate()->factory()->empty_string();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int num_parameters = 0;
|
2011-08-16 14:24:12 +00:00
|
|
|
// Function declarations are function scoped in normal mode, so they are
|
|
|
|
// hoisted. In harmony block scoping mode they are block scoped, so they
|
|
|
|
// are not hoisted.
|
2013-08-23 09:25:37 +00:00
|
|
|
//
|
|
|
|
// One tricky case are function declarations in a local sloppy-mode eval:
|
|
|
|
// their declaration is hoisted, but they still see the local scope. E.g.,
|
|
|
|
//
|
|
|
|
// function() {
|
|
|
|
// var x = 0
|
|
|
|
// try { throw 1 } catch (x) { eval("function g() { return x }") }
|
|
|
|
// return g()
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// needs to return 1. To distinguish such cases, we need to detect
|
|
|
|
// (1) whether a function stems from a sloppy eval, and
|
|
|
|
// (2) whether it actually hoists across the eval.
|
|
|
|
// Unfortunately, we do not represent sloppy eval scopes, so we do not have
|
|
|
|
// either information available directly, especially not when lazily compiling
|
|
|
|
// a function like 'g'. We hence rely on the following invariants:
|
|
|
|
// - (1) is the case iff the innermost scope of the deserialized scope chain
|
|
|
|
// under which we compile is _not_ a declaration scope. This holds because
|
|
|
|
// in all normal cases, function declarations are fully hoisted to a
|
|
|
|
// declaration scope and compiled relative to that.
|
|
|
|
// - (2) is the case iff the current declaration scope is still the original
|
|
|
|
// one relative to the deserialized scope chain. Otherwise we must be
|
|
|
|
// compiling a function in an inner declaration scope in the eval, e.g. a
|
|
|
|
// nested function, and hoisting works normally relative to that.
|
|
|
|
Scope* declaration_scope = top_scope_->DeclarationScope();
|
|
|
|
Scope* original_declaration_scope = original_scope_->DeclarationScope();
|
2013-06-06 13:28:22 +00:00
|
|
|
Scope* scope =
|
2013-08-23 09:25:37 +00:00
|
|
|
function_type == FunctionLiteral::DECLARATION && !is_extended_mode() &&
|
|
|
|
(original_scope_ == original_declaration_scope ||
|
|
|
|
declaration_scope != original_declaration_scope)
|
|
|
|
? NewScope(declaration_scope, FUNCTION_SCOPE)
|
|
|
|
: NewScope(top_scope_, FUNCTION_SCOPE);
|
2011-11-11 13:48:14 +00:00
|
|
|
ZoneList<Statement*>* body = NULL;
|
2011-11-25 09:36:31 +00:00
|
|
|
int materialized_literal_count = -1;
|
|
|
|
int expected_property_count = -1;
|
2011-11-11 13:48:14 +00:00
|
|
|
int handler_count = 0;
|
2012-02-14 14:14:51 +00:00
|
|
|
FunctionLiteral::ParameterFlag duplicate_parameters =
|
|
|
|
FunctionLiteral::kNoDuplicateParameters;
|
2012-08-07 14:47:36 +00:00
|
|
|
FunctionLiteral::IsParenthesizedFlag parenthesized = parenthesized_function_
|
|
|
|
? FunctionLiteral::kIsParenthesized
|
|
|
|
: FunctionLiteral::kNotParenthesized;
|
2013-04-02 17:34:59 +00:00
|
|
|
FunctionLiteral::IsGeneratorFlag generator = is_generator
|
|
|
|
? FunctionLiteral::kIsGenerator
|
|
|
|
: FunctionLiteral::kNotGenerator;
|
2012-02-08 09:56:33 +00:00
|
|
|
AstProperties ast_properties;
|
2013-09-05 13:20:51 +00:00
|
|
|
BailoutReason dont_optimize_reason = kNoReason;
|
2008-07-03 15:10:15 +00:00
|
|
|
// Parse function body.
|
2013-04-15 12:29:44 +00:00
|
|
|
{ FunctionState function_state(this, scope, isolate());
|
2011-08-08 16:14:46 +00:00
|
|
|
top_scope_->SetScopeName(function_name);
|
2013-04-15 12:29:44 +00:00
|
|
|
|
|
|
|
if (is_generator) {
|
|
|
|
// For generators, allocating variables in contexts is currently a win
|
|
|
|
// because it minimizes the work needed to suspend and resume an
|
|
|
|
// activation.
|
|
|
|
top_scope_->ForceContextAllocation();
|
|
|
|
|
|
|
|
// Calling a generator returns a generator object. That object is stored
|
|
|
|
// in a temporary variable, a definition that is used by "yield"
|
|
|
|
// expressions. Presence of a variable for the generator object in the
|
|
|
|
// FunctionState indicates that this function is a generator.
|
|
|
|
Handle<String> tempname = isolate()->factory()->InternalizeOneByteString(
|
|
|
|
STATIC_ASCII_VECTOR(".generator_object"));
|
|
|
|
Variable* temp = top_scope_->DeclarationScope()->NewTemporary(tempname);
|
|
|
|
function_state.set_generator_object_variable(temp);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// FormalParameterList ::
|
|
|
|
// '(' (Identifier)*[','] ')'
|
|
|
|
Expect(Token::LPAREN, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
scope->set_start_position(scanner().location().beg_pos);
|
2011-05-06 11:41:15 +00:00
|
|
|
Scanner::Location name_loc = Scanner::Location::invalid();
|
|
|
|
Scanner::Location dupe_loc = Scanner::Location::invalid();
|
|
|
|
Scanner::Location reserved_loc = Scanner::Location::invalid();
|
2011-01-20 18:51:47 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool done = (peek() == Token::RPAREN);
|
|
|
|
while (!done) {
|
2011-06-24 14:59:51 +00:00
|
|
|
bool is_strict_reserved = false;
|
2011-02-04 18:36:37 +00:00
|
|
|
Handle<String> param_name =
|
2011-06-24 14:59:51 +00:00
|
|
|
ParseIdentifierOrStrictReservedWord(&is_strict_reserved,
|
|
|
|
CHECK_OK);
|
2011-01-25 17:21:45 +00:00
|
|
|
|
|
|
|
// Store locations for possible future error reports.
|
|
|
|
if (!name_loc.IsValid() && IsEvalOrArguments(param_name)) {
|
|
|
|
name_loc = scanner().location();
|
|
|
|
}
|
|
|
|
if (!dupe_loc.IsValid() && top_scope_->IsDeclared(param_name)) {
|
2012-02-14 14:14:51 +00:00
|
|
|
duplicate_parameters = FunctionLiteral::kHasDuplicateParameters;
|
2011-01-25 17:21:45 +00:00
|
|
|
dupe_loc = scanner().location();
|
|
|
|
}
|
2011-06-24 14:59:51 +00:00
|
|
|
if (!reserved_loc.IsValid() && is_strict_reserved) {
|
2011-02-04 18:36:37 +00:00
|
|
|
reserved_loc = scanner().location();
|
|
|
|
}
|
2011-01-25 17:21:45 +00:00
|
|
|
|
2012-02-14 13:47:54 +00:00
|
|
|
top_scope_->DeclareParameter(param_name, VAR);
|
2010-11-02 11:45:47 +00:00
|
|
|
num_parameters++;
|
2013-03-18 13:35:17 +00:00
|
|
|
if (num_parameters > Code::kMaxArguments) {
|
2011-02-09 12:46:22 +00:00
|
|
|
ReportMessageAt(scanner().location(), "too_many_parameters",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
done = (peek() == Token::RPAREN);
|
|
|
|
if (!done) Expect(Token::COMMA, CHECK_OK);
|
|
|
|
}
|
|
|
|
Expect(Token::RPAREN, CHECK_OK);
|
|
|
|
|
|
|
|
Expect(Token::LBRACE, CHECK_OK);
|
|
|
|
|
|
|
|
// If we have a named function expression, we add a local variable
|
|
|
|
// declaration to the body of the function with the name of the
|
|
|
|
// function and let it refer to the function itself (closure).
|
|
|
|
// NOTE: We create a proxy and resolve it here so that in the
|
|
|
|
// future we can change the AST to only refer to VariableProxies
|
|
|
|
// instead of Variables and Proxis as is the case now.
|
2011-11-11 13:48:14 +00:00
|
|
|
Variable* fvar = NULL;
|
|
|
|
Token::Value fvar_init_op = Token::INIT_CONST;
|
2013-06-06 13:28:22 +00:00
|
|
|
if (function_type == FunctionLiteral::NAMED_EXPRESSION) {
|
2012-04-16 11:48:20 +00:00
|
|
|
if (is_extended_mode()) fvar_init_op = Token::INIT_CONST_HARMONY;
|
|
|
|
VariableMode fvar_mode = is_extended_mode() ? CONST_HARMONY : CONST;
|
|
|
|
fvar = new(zone()) Variable(top_scope_,
|
|
|
|
function_name, fvar_mode, true /* is valid LHS */,
|
2012-07-13 09:29:43 +00:00
|
|
|
Variable::NORMAL, kCreatedInitialized, Interface::NewConst());
|
2012-04-16 11:48:20 +00:00
|
|
|
VariableProxy* proxy = factory()->NewVariableProxy(fvar);
|
2013-10-14 09:24:58 +00:00
|
|
|
VariableDeclaration* fvar_declaration = factory()->NewVariableDeclaration(
|
|
|
|
proxy, fvar_mode, top_scope_, RelocInfo::kNoPosition);
|
2012-04-16 11:48:20 +00:00
|
|
|
top_scope_->DeclareFunctionVar(fvar_declaration);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-11-25 09:36:31 +00:00
|
|
|
// Determine whether the function will be lazily compiled.
|
|
|
|
// The heuristics are:
|
|
|
|
// - It must not have been prohibited by the caller to Parse (some callers
|
|
|
|
// need a full AST).
|
2012-06-28 14:56:28 +00:00
|
|
|
// - The outer scope must allow lazy compilation of inner functions.
|
2011-11-25 09:36:31 +00:00
|
|
|
// - The function mustn't be a function expression with an open parenthesis
|
|
|
|
// before; we consider that a hint that the function will be called
|
|
|
|
// immediately, and it would be a waste of time to make it lazily
|
|
|
|
// compiled.
|
|
|
|
// These are all things we can know at this point, without looking at the
|
|
|
|
// function itself.
|
2011-01-14 10:50:13 +00:00
|
|
|
bool is_lazily_compiled = (mode() == PARSE_LAZILY &&
|
2012-06-28 14:56:28 +00:00
|
|
|
top_scope_->AllowsLazyCompilation() &&
|
2011-11-25 09:36:31 +00:00
|
|
|
!parenthesized_function_);
|
2011-01-14 10:50:13 +00:00
|
|
|
parenthesized_function_ = false; // The bit was set for this function only.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-08-03 09:10:35 +00:00
|
|
|
if (is_lazily_compiled) {
|
2013-10-14 09:24:58 +00:00
|
|
|
int function_block_pos = position();
|
2011-11-25 09:36:31 +00:00
|
|
|
FunctionEntry entry;
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
if (pre_parse_data_ != NULL) {
|
|
|
|
// If we have pre_parse_data_, we use it to skip parsing the function
|
|
|
|
// body. The preparser data contains the information we need to
|
|
|
|
// construct the lazy function.
|
|
|
|
entry = pre_parse_data()->GetFunctionEntry(function_block_pos);
|
2011-11-25 09:36:31 +00:00
|
|
|
if (entry.is_valid()) {
|
|
|
|
if (entry.end_pos() <= function_block_pos) {
|
|
|
|
// End position greater than end of stream is safe, and hard
|
|
|
|
// to check.
|
|
|
|
ReportInvalidPreparseData(function_name, CHECK_OK);
|
|
|
|
}
|
|
|
|
scanner().SeekForward(entry.end_pos() - 1);
|
|
|
|
|
|
|
|
scope->set_end_position(entry.end_pos());
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
|
|
|
isolate()->counters()->total_preparse_skipped()->Increment(
|
|
|
|
scope->end_position() - function_block_pos);
|
|
|
|
materialized_literal_count = entry.literal_count();
|
|
|
|
expected_property_count = entry.property_count();
|
|
|
|
top_scope_->SetLanguageMode(entry.language_mode());
|
|
|
|
} else {
|
|
|
|
is_lazily_compiled = false;
|
|
|
|
}
|
2011-08-03 09:10:35 +00:00
|
|
|
} else {
|
2011-11-25 09:36:31 +00:00
|
|
|
// With no preparser data, we partially parse the function, without
|
|
|
|
// building an AST. This gathers the data needed to build a lazy
|
|
|
|
// function.
|
|
|
|
SingletonLogger logger;
|
|
|
|
preparser::PreParser::PreParseResult result =
|
|
|
|
LazyParseFunctionLiteral(&logger);
|
|
|
|
if (result == preparser::PreParser::kPreParseStackOverflow) {
|
|
|
|
// Propagate stack overflow.
|
|
|
|
stack_overflow_ = true;
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (logger.has_error()) {
|
|
|
|
const char* arg = logger.argument_opt();
|
|
|
|
Vector<const char*> args;
|
|
|
|
if (arg != NULL) {
|
|
|
|
args = Vector<const char*>(&arg, 1);
|
|
|
|
}
|
|
|
|
ReportMessageAt(Scanner::Location(logger.start(), logger.end()),
|
|
|
|
logger.message(), args);
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2011-08-03 09:10:35 +00:00
|
|
|
}
|
2011-11-25 09:36:31 +00:00
|
|
|
scope->set_end_position(logger.end());
|
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
2011-08-03 09:10:35 +00:00
|
|
|
isolate()->counters()->total_preparse_skipped()->Increment(
|
2011-10-21 10:26:59 +00:00
|
|
|
scope->end_position() - function_block_pos);
|
2011-11-25 09:36:31 +00:00
|
|
|
materialized_literal_count = logger.literals();
|
|
|
|
expected_property_count = logger.properties();
|
|
|
|
top_scope_->SetLanguageMode(logger.language_mode());
|
2010-07-07 10:28:22 +00:00
|
|
|
}
|
2011-08-03 09:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_lazily_compiled) {
|
2012-08-07 14:47:36 +00:00
|
|
|
ParsingModeScope parsing_mode(this, PARSE_EAGERLY);
|
2012-06-11 12:42:31 +00:00
|
|
|
body = new(zone()) ZoneList<Statement*>(8, zone());
|
2011-11-11 13:48:14 +00:00
|
|
|
if (fvar != NULL) {
|
2012-07-13 09:29:43 +00:00
|
|
|
VariableProxy* fproxy = top_scope_->NewUnresolved(
|
|
|
|
factory(), function_name, Interface::NewConst());
|
2011-11-11 13:48:14 +00:00
|
|
|
fproxy->BindTo(fvar);
|
2012-02-08 09:56:33 +00:00
|
|
|
body->Add(factory()->NewExpressionStatement(
|
|
|
|
factory()->NewAssignment(fvar_init_op,
|
|
|
|
fproxy,
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewThisFunction(pos),
|
|
|
|
RelocInfo::kNoPosition),
|
|
|
|
RelocInfo::kNoPosition), zone());
|
2011-11-11 13:48:14 +00:00
|
|
|
}
|
2013-04-15 12:29:44 +00:00
|
|
|
|
|
|
|
// For generators, allocate and yield an iterator on function entry.
|
|
|
|
if (is_generator) {
|
|
|
|
ZoneList<Expression*>* arguments =
|
|
|
|
new(zone()) ZoneList<Expression*>(0, zone());
|
|
|
|
CallRuntime* allocation = factory()->NewCallRuntime(
|
|
|
|
isolate()->factory()->empty_string(),
|
|
|
|
Runtime::FunctionForId(Runtime::kCreateJSGeneratorObject),
|
2013-10-14 09:24:58 +00:00
|
|
|
arguments, pos);
|
2013-04-15 12:29:44 +00:00
|
|
|
VariableProxy* init_proxy = factory()->NewVariableProxy(
|
|
|
|
current_function_state_->generator_object_variable());
|
|
|
|
Assignment* assignment = factory()->NewAssignment(
|
|
|
|
Token::INIT_VAR, init_proxy, allocation, RelocInfo::kNoPosition);
|
|
|
|
VariableProxy* get_proxy = factory()->NewVariableProxy(
|
|
|
|
current_function_state_->generator_object_variable());
|
|
|
|
Yield* yield = factory()->NewYield(
|
2013-08-20 06:39:04 +00:00
|
|
|
get_proxy, assignment, Yield::INITIAL, RelocInfo::kNoPosition);
|
2013-10-14 09:24:58 +00:00
|
|
|
body->Add(factory()->NewExpressionStatement(
|
|
|
|
yield, RelocInfo::kNoPosition), zone());
|
2013-04-15 12:29:44 +00:00
|
|
|
}
|
|
|
|
|
2012-10-05 09:14:08 +00:00
|
|
|
ParseSourceElements(body, Token::RBRACE, false, false, CHECK_OK);
|
2010-11-02 11:45:47 +00:00
|
|
|
|
2013-04-19 14:11:23 +00:00
|
|
|
if (is_generator) {
|
|
|
|
VariableProxy* get_proxy = factory()->NewVariableProxy(
|
|
|
|
current_function_state_->generator_object_variable());
|
|
|
|
Expression *undefined = factory()->NewLiteral(
|
2013-10-14 09:24:58 +00:00
|
|
|
isolate()->factory()->undefined_value(), RelocInfo::kNoPosition);
|
2013-04-19 14:11:23 +00:00
|
|
|
Yield* yield = factory()->NewYield(
|
2013-08-20 06:39:04 +00:00
|
|
|
get_proxy, undefined, Yield::FINAL, RelocInfo::kNoPosition);
|
2013-10-14 09:24:58 +00:00
|
|
|
body->Add(factory()->NewExpressionStatement(
|
|
|
|
yield, RelocInfo::kNoPosition), zone());
|
2013-04-19 14:11:23 +00:00
|
|
|
}
|
|
|
|
|
2011-11-09 13:54:26 +00:00
|
|
|
materialized_literal_count = function_state.materialized_literal_count();
|
|
|
|
expected_property_count = function_state.expected_property_count();
|
2011-11-11 13:48:14 +00:00
|
|
|
handler_count = function_state.handler_count();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-08-27 08:26:29 +00:00
|
|
|
Expect(Token::RBRACE, CHECK_OK);
|
2011-10-21 10:26:59 +00:00
|
|
|
scope->set_end_position(scanner().location().end_pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-01-20 18:51:47 +00:00
|
|
|
// Validate strict mode.
|
2011-11-24 15:17:04 +00:00
|
|
|
if (!top_scope_->is_classic_mode()) {
|
2011-08-08 16:14:46 +00:00
|
|
|
if (IsEvalOrArguments(function_name)) {
|
2011-10-21 10:26:59 +00:00
|
|
|
int start_pos = scope->start_position();
|
2013-10-14 09:24:58 +00:00
|
|
|
int position = function_token_pos != RelocInfo::kNoPosition
|
|
|
|
? function_token_pos : (start_pos > 0 ? start_pos - 1 : start_pos);
|
2011-02-04 18:36:37 +00:00
|
|
|
Scanner::Location location = Scanner::Location(position, start_pos);
|
|
|
|
ReportMessageAt(location,
|
2011-01-20 18:51:47 +00:00
|
|
|
"strict_function_name", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-01-25 17:21:45 +00:00
|
|
|
if (name_loc.IsValid()) {
|
|
|
|
ReportMessageAt(name_loc, "strict_param_name",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (dupe_loc.IsValid()) {
|
|
|
|
ReportMessageAt(dupe_loc, "strict_param_dupe",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-06-24 14:59:51 +00:00
|
|
|
if (name_is_strict_reserved) {
|
2011-10-21 10:26:59 +00:00
|
|
|
int start_pos = scope->start_position();
|
2013-10-14 09:24:58 +00:00
|
|
|
int position = function_token_pos != RelocInfo::kNoPosition
|
|
|
|
? function_token_pos : (start_pos > 0 ? start_pos - 1 : start_pos);
|
2011-02-04 18:36:37 +00:00
|
|
|
Scanner::Location location = Scanner::Location(position, start_pos);
|
|
|
|
ReportMessageAt(location, "strict_reserved_word",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (reserved_loc.IsValid()) {
|
|
|
|
ReportMessageAt(reserved_loc, "strict_reserved_word",
|
|
|
|
Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-10-21 10:26:59 +00:00
|
|
|
CheckOctalLiteral(scope->start_position(),
|
|
|
|
scope->end_position(),
|
|
|
|
CHECK_OK);
|
2011-01-20 18:51:47 +00:00
|
|
|
}
|
2012-02-08 09:56:33 +00:00
|
|
|
ast_properties = *factory()->visitor()->ast_properties();
|
2013-09-05 13:20:51 +00:00
|
|
|
dont_optimize_reason = factory()->visitor()->dont_optimize_reason();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-04-07 14:45:34 +00:00
|
|
|
|
2011-11-24 15:58:09 +00:00
|
|
|
if (is_extended_mode()) {
|
2011-09-01 12:31:18 +00:00
|
|
|
CheckConflictingVarDeclarations(scope, CHECK_OK);
|
|
|
|
}
|
|
|
|
|
2011-04-07 14:45:34 +00:00
|
|
|
FunctionLiteral* function_literal =
|
2012-02-08 09:56:33 +00:00
|
|
|
factory()->NewFunctionLiteral(function_name,
|
|
|
|
scope,
|
|
|
|
body,
|
|
|
|
materialized_literal_count,
|
|
|
|
expected_property_count,
|
|
|
|
handler_count,
|
|
|
|
num_parameters,
|
2012-02-14 14:14:51 +00:00
|
|
|
duplicate_parameters,
|
2013-06-06 13:28:22 +00:00
|
|
|
function_type,
|
2012-08-07 14:47:36 +00:00
|
|
|
FunctionLiteral::kIsFunction,
|
2013-04-02 17:34:59 +00:00
|
|
|
parenthesized,
|
2013-10-14 09:24:58 +00:00
|
|
|
generator,
|
|
|
|
pos);
|
|
|
|
function_literal->set_function_token_position(function_token_pos);
|
2012-02-08 09:56:33 +00:00
|
|
|
function_literal->set_ast_properties(&ast_properties);
|
2013-09-05 13:20:51 +00:00
|
|
|
function_literal->set_dont_optimize_reason(dont_optimize_reason);
|
2011-04-07 14:45:34 +00:00
|
|
|
|
2011-08-08 16:14:46 +00:00
|
|
|
if (fni_ != NULL && should_infer_name) fni_->AddFunction(function_literal);
|
2011-04-07 14:45:34 +00:00
|
|
|
return function_literal;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-25 09:36:31 +00:00
|
|
|
preparser::PreParser::PreParseResult Parser::LazyParseFunctionLiteral(
|
|
|
|
SingletonLogger* logger) {
|
|
|
|
HistogramTimerScope preparse_scope(isolate()->counters()->pre_parse());
|
|
|
|
ASSERT_EQ(Token::LBRACE, scanner().current_token());
|
|
|
|
|
|
|
|
if (reusable_preparser_ == NULL) {
|
|
|
|
intptr_t stack_limit = isolate()->stack_guard()->real_climit();
|
|
|
|
reusable_preparser_ = new preparser::PreParser(&scanner_,
|
|
|
|
NULL,
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
stack_limit);
|
|
|
|
reusable_preparser_->set_allow_harmony_scoping(allow_harmony_scoping());
|
|
|
|
reusable_preparser_->set_allow_modules(allow_modules());
|
|
|
|
reusable_preparser_->set_allow_natives_syntax(allow_natives_syntax());
|
|
|
|
reusable_preparser_->set_allow_lazy(true);
|
|
|
|
reusable_preparser_->set_allow_generators(allow_generators());
|
2013-06-06 14:38:26 +00:00
|
|
|
reusable_preparser_->set_allow_for_of(allow_for_of());
|
2013-07-19 09:57:35 +00:00
|
|
|
reusable_preparser_->set_allow_harmony_numeric_literals(
|
|
|
|
allow_harmony_numeric_literals());
|
2011-11-25 09:36:31 +00:00
|
|
|
}
|
|
|
|
preparser::PreParser::PreParseResult result =
|
|
|
|
reusable_preparser_->PreParseLazyFunction(top_scope_->language_mode(),
|
2013-04-02 17:34:59 +00:00
|
|
|
is_generator(),
|
2011-11-25 09:36:31 +00:00
|
|
|
logger);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Expression* Parser::ParseV8Intrinsic(bool* ok) {
|
|
|
|
// CallRuntime ::
|
|
|
|
// '%' Identifier Arguments
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = peek_position();
|
2008-07-03 15:10:15 +00:00
|
|
|
Expect(Token::MOD, CHECK_OK);
|
|
|
|
Handle<String> name = ParseIdentifier(CHECK_OK);
|
|
|
|
ZoneList<Expression*>* args = ParseArguments(CHECK_OK);
|
2010-09-14 14:52:53 +00:00
|
|
|
|
|
|
|
if (extension_ != NULL) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// The extension structures are only accessible while parsing the
|
|
|
|
// very first time not when reparsing because of lazy compilation.
|
2011-07-04 09:34:47 +00:00
|
|
|
top_scope_->DeclarationScope()->ForceEagerCompilation();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2013-02-28 17:03:34 +00:00
|
|
|
const Runtime::Function* function = Runtime::FunctionForName(name);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-09-14 14:52:53 +00:00
|
|
|
// Check for built-in IS_VAR macro.
|
|
|
|
if (function != NULL &&
|
|
|
|
function->intrinsic_type == Runtime::RUNTIME &&
|
|
|
|
function->function_id == Runtime::kIS_VAR) {
|
|
|
|
// %IS_VAR(x) evaluates to x if x is a variable,
|
|
|
|
// leads to a parse error otherwise. Could be implemented as an
|
|
|
|
// inline function %_IS_VAR(x) to eliminate this special case.
|
|
|
|
if (args->length() == 1 && args->at(0)->AsVariableProxy() != NULL) {
|
|
|
|
return args->at(0);
|
|
|
|
} else {
|
2013-04-26 14:26:54 +00:00
|
|
|
ReportMessage("not_isvar", Vector<const char*>::empty());
|
2010-09-14 14:52:53 +00:00
|
|
|
*ok = false;
|
2008-07-03 15:10:15 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-14 14:52:53 +00:00
|
|
|
// Check that the expected number of arguments are being passed.
|
|
|
|
if (function != NULL &&
|
|
|
|
function->nargs != -1 &&
|
|
|
|
function->nargs != args->length()) {
|
|
|
|
ReportMessage("illegal_access", Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2012-08-14 10:06:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the function is defined if it's an inline runtime call.
|
|
|
|
if (function == NULL && name->Get(0) == '_') {
|
|
|
|
ReportMessage("not_defined", Vector<Handle<String> >(&name, 1));
|
|
|
|
*ok = false;
|
|
|
|
return NULL;
|
2010-03-11 09:27:12 +00:00
|
|
|
}
|
|
|
|
|
2010-09-14 14:52:53 +00:00
|
|
|
// We have a valid intrinsics call or a call to a builtin.
|
2013-10-14 09:24:58 +00:00
|
|
|
return factory()->NewCallRuntime(name, function, args, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-04 18:36:37 +00:00
|
|
|
bool Parser::peek_any_identifier() {
|
|
|
|
Token::Value next = peek();
|
|
|
|
return next == Token::IDENTIFIER ||
|
2011-06-24 14:59:51 +00:00
|
|
|
next == Token::FUTURE_RESERVED_WORD ||
|
2013-04-02 17:34:59 +00:00
|
|
|
next == Token::FUTURE_STRICT_RESERVED_WORD ||
|
|
|
|
next == Token::YIELD;
|
2011-02-04 18:36:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void Parser::Consume(Token::Value token) {
|
|
|
|
Token::Value next = Next();
|
|
|
|
USE(next);
|
|
|
|
USE(token);
|
|
|
|
ASSERT(next == token);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Parser::Expect(Token::Value token, bool* ok) {
|
|
|
|
Token::Value next = Next();
|
|
|
|
if (next == token) return;
|
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-01 10:31:55 +00:00
|
|
|
bool Parser::Check(Token::Value token) {
|
|
|
|
Token::Value next = peek();
|
|
|
|
if (next == token) {
|
|
|
|
Consume(next);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 14:38:26 +00:00
|
|
|
bool Parser::CheckContextualKeyword(Vector<const char> keyword) {
|
|
|
|
if (peek() == Token::IDENTIFIER &&
|
|
|
|
scanner().is_next_contextual_keyword(keyword)) {
|
|
|
|
Consume(Token::IDENTIFIER);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void Parser::ExpectSemicolon(bool* ok) {
|
|
|
|
// Check for automatic semicolon insertion according to
|
|
|
|
// the rules given in ECMA-262, section 7.9, page 21.
|
|
|
|
Token::Value tok = peek();
|
|
|
|
if (tok == Token::SEMICOLON) {
|
|
|
|
Next();
|
|
|
|
return;
|
|
|
|
}
|
2011-06-21 13:34:16 +00:00
|
|
|
if (scanner().HasAnyLineTerminatorBeforeNext() ||
|
2008-07-03 15:10:15 +00:00
|
|
|
tok == Token::RBRACE ||
|
|
|
|
tok == Token::EOS) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Expect(Token::SEMICOLON, ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 14:38:26 +00:00
|
|
|
void Parser::ExpectContextualKeyword(Vector<const char> keyword, bool* ok) {
|
2012-02-29 12:12:52 +00:00
|
|
|
Expect(Token::IDENTIFIER, ok);
|
|
|
|
if (!*ok) return;
|
2013-06-06 14:38:26 +00:00
|
|
|
if (!scanner().is_literal_contextual_keyword(keyword)) {
|
2012-02-29 12:12:52 +00:00
|
|
|
*ok = false;
|
|
|
|
ReportUnexpectedToken(scanner().current_token());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
Literal* Parser::GetLiteralUndefined(int position) {
|
|
|
|
return factory()->NewLiteral(
|
|
|
|
isolate()->factory()->undefined_value(), position);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
Literal* Parser::GetLiteralTheHole(int position) {
|
|
|
|
return factory()->NewLiteral(
|
|
|
|
isolate()->factory()->the_hole_value(), RelocInfo::kNoPosition);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-16 14:24:12 +00:00
|
|
|
// Parses an identifier that is valid for the current scope, in particular it
|
2011-06-24 14:59:51 +00:00
|
|
|
// fails on strict mode future reserved keywords in a strict scope.
|
2008-07-03 15:10:15 +00:00
|
|
|
Handle<String> Parser::ParseIdentifier(bool* ok) {
|
2013-04-02 17:34:59 +00:00
|
|
|
Token::Value next = Next();
|
|
|
|
if (next == Token::IDENTIFIER ||
|
|
|
|
(top_scope_->is_classic_mode() &&
|
|
|
|
(next == Token::FUTURE_STRICT_RESERVED_WORD ||
|
|
|
|
(next == Token::YIELD && !is_generator())))) {
|
2013-05-21 10:45:58 +00:00
|
|
|
return GetSymbol();
|
2013-04-02 17:34:59 +00:00
|
|
|
} else {
|
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
return Handle<String>();
|
2011-06-24 14:59:51 +00:00
|
|
|
}
|
2011-02-04 18:36:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-24 14:59:51 +00:00
|
|
|
// Parses and identifier or a strict mode future reserved word, and indicate
|
|
|
|
// whether it is strict mode future reserved.
|
|
|
|
Handle<String> Parser::ParseIdentifierOrStrictReservedWord(
|
|
|
|
bool* is_strict_reserved, bool* ok) {
|
2013-04-02 17:34:59 +00:00
|
|
|
Token::Value next = Next();
|
|
|
|
if (next == Token::IDENTIFIER) {
|
|
|
|
*is_strict_reserved = false;
|
|
|
|
} else if (next == Token::FUTURE_STRICT_RESERVED_WORD ||
|
|
|
|
(next == Token::YIELD && !is_generator())) {
|
2011-06-24 14:59:51 +00:00
|
|
|
*is_strict_reserved = true;
|
2013-04-02 17:34:59 +00:00
|
|
|
} else {
|
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
return Handle<String>();
|
2011-02-04 18:36:37 +00:00
|
|
|
}
|
2013-05-21 10:45:58 +00:00
|
|
|
return GetSymbol();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2010-08-06 08:03:44 +00:00
|
|
|
|
|
|
|
Handle<String> Parser::ParseIdentifierName(bool* ok) {
|
|
|
|
Token::Value next = Next();
|
2011-02-04 18:36:37 +00:00
|
|
|
if (next != Token::IDENTIFIER &&
|
2011-06-24 14:59:51 +00:00
|
|
|
next != Token::FUTURE_RESERVED_WORD &&
|
|
|
|
next != Token::FUTURE_STRICT_RESERVED_WORD &&
|
|
|
|
!Token::IsKeyword(next)) {
|
2010-08-06 08:03:44 +00:00
|
|
|
ReportUnexpectedToken(next);
|
|
|
|
*ok = false;
|
|
|
|
return Handle<String>();
|
|
|
|
}
|
2013-05-21 10:45:58 +00:00
|
|
|
return GetSymbol();
|
2010-08-06 08:03:44 +00:00
|
|
|
}
|
|
|
|
|
2011-01-26 19:21:46 +00:00
|
|
|
|
2011-12-05 14:43:28 +00:00
|
|
|
void Parser::MarkAsLValue(Expression* expression) {
|
|
|
|
VariableProxy* proxy = expression != NULL
|
|
|
|
? expression->AsVariableProxy()
|
|
|
|
: NULL;
|
|
|
|
|
|
|
|
if (proxy != NULL) proxy->MarkAsLValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-26 19:21:46 +00:00
|
|
|
// Checks LHS expression for assignment and prefix/postfix increment/decrement
|
|
|
|
// in strict mode.
|
|
|
|
void Parser::CheckStrictModeLValue(Expression* expression,
|
|
|
|
const char* error,
|
|
|
|
bool* ok) {
|
2011-11-24 15:17:04 +00:00
|
|
|
ASSERT(!top_scope_->is_classic_mode());
|
2011-01-26 19:21:46 +00:00
|
|
|
VariableProxy* lhs = expression != NULL
|
|
|
|
? expression->AsVariableProxy()
|
|
|
|
: NULL;
|
|
|
|
|
|
|
|
if (lhs != NULL && !lhs->is_this() && IsEvalOrArguments(lhs->name())) {
|
|
|
|
ReportMessage(error, Vector<const char*>::empty());
|
|
|
|
*ok = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-06 11:41:15 +00:00
|
|
|
// Checks whether an octal literal was last seen between beg_pos and end_pos.
|
|
|
|
// If so, reports an error. Only called for strict mode.
|
2011-01-24 18:13:18 +00:00
|
|
|
void Parser::CheckOctalLiteral(int beg_pos, int end_pos, bool* ok) {
|
2011-05-06 11:41:15 +00:00
|
|
|
Scanner::Location octal = scanner().octal_position();
|
|
|
|
if (octal.IsValid() &&
|
|
|
|
beg_pos <= octal.beg_pos &&
|
|
|
|
octal.end_pos <= end_pos) {
|
|
|
|
ReportMessageAt(octal, "strict_octal_literal",
|
2011-01-24 18:13:18 +00:00
|
|
|
Vector<const char*>::empty());
|
|
|
|
scanner().clear_octal_position();
|
|
|
|
*ok = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-06 08:03:44 +00:00
|
|
|
|
2011-09-01 12:31:18 +00:00
|
|
|
void Parser::CheckConflictingVarDeclarations(Scope* scope, bool* ok) {
|
|
|
|
Declaration* decl = scope->CheckConflictingVarDeclarations();
|
|
|
|
if (decl != NULL) {
|
|
|
|
// In harmony mode we treat conflicting variable bindinds as early
|
|
|
|
// errors. See ES5 16 for a definition of early errors.
|
|
|
|
Handle<String> name = decl->proxy()->name();
|
2011-09-09 22:39:47 +00:00
|
|
|
SmartArrayPointer<char> c_string = name->ToCString(DISALLOW_NULLS);
|
2011-09-01 12:31:18 +00:00
|
|
|
const char* elms[2] = { "Variable", *c_string };
|
|
|
|
Vector<const char*> args(elms, 2);
|
|
|
|
int position = decl->proxy()->position();
|
|
|
|
Scanner::Location location = position == RelocInfo::kNoPosition
|
|
|
|
? Scanner::Location::invalid()
|
|
|
|
: Scanner::Location(position, position + 1);
|
|
|
|
ReportMessageAt(location, "redeclaration", args);
|
|
|
|
*ok = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-24 14:59:51 +00:00
|
|
|
// This function reads an identifier name and determines whether or not it
|
2011-02-04 18:36:37 +00:00
|
|
|
// is 'get' or 'set'.
|
2011-06-24 14:59:51 +00:00
|
|
|
Handle<String> Parser::ParseIdentifierNameOrGetOrSet(bool* is_get,
|
|
|
|
bool* is_set,
|
|
|
|
bool* ok) {
|
|
|
|
Handle<String> result = ParseIdentifierName(ok);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (!*ok) return Handle<String>();
|
2010-12-22 20:14:19 +00:00
|
|
|
if (scanner().is_literal_ascii() && scanner().literal_length() == 3) {
|
|
|
|
const char* token = scanner().literal_ascii_string().start();
|
|
|
|
*is_get = strncmp(token, "get", 3) == 0;
|
|
|
|
*is_set = !*is_get && strncmp(token, "set", 3) == 0;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2011-02-04 18:36:37 +00:00
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Parser support
|
|
|
|
|
|
|
|
|
|
|
|
bool Parser::TargetStackContainsLabel(Handle<String> label) {
|
2009-05-15 14:58:02 +00:00
|
|
|
for (Target* t = target_stack_; t != NULL; t = t->previous()) {
|
|
|
|
BreakableStatement* stat = t->node()->AsBreakableStatement();
|
2008-07-03 15:10:15 +00:00
|
|
|
if (stat != NULL && ContainsLabel(stat->labels(), label))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BreakableStatement* Parser::LookupBreakTarget(Handle<String> label, bool* ok) {
|
|
|
|
bool anonymous = label.is_null();
|
2009-05-15 14:58:02 +00:00
|
|
|
for (Target* t = target_stack_; t != NULL; t = t->previous()) {
|
|
|
|
BreakableStatement* stat = t->node()->AsBreakableStatement();
|
2008-07-03 15:10:15 +00:00
|
|
|
if (stat == NULL) continue;
|
|
|
|
if ((anonymous && stat->is_target_for_anonymous()) ||
|
|
|
|
(!anonymous && ContainsLabel(stat->labels(), label))) {
|
2009-05-15 14:58:02 +00:00
|
|
|
RegisterTargetUse(stat->break_target(), t->previous());
|
2008-07-03 15:10:15 +00:00
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IterationStatement* Parser::LookupContinueTarget(Handle<String> label,
|
|
|
|
bool* ok) {
|
|
|
|
bool anonymous = label.is_null();
|
2009-05-15 14:58:02 +00:00
|
|
|
for (Target* t = target_stack_; t != NULL; t = t->previous()) {
|
|
|
|
IterationStatement* stat = t->node()->AsIterationStatement();
|
2008-07-03 15:10:15 +00:00
|
|
|
if (stat == NULL) continue;
|
|
|
|
|
|
|
|
ASSERT(stat->is_target_for_anonymous());
|
|
|
|
if (anonymous || ContainsLabel(stat->labels(), label)) {
|
2009-05-15 14:58:02 +00:00
|
|
|
RegisterTargetUse(stat->continue_target(), t->previous());
|
2008-07-03 15:10:15 +00:00
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-07 14:42:37 +00:00
|
|
|
void Parser::RegisterTargetUse(Label* target, Target* stop) {
|
2009-05-15 14:58:02 +00:00
|
|
|
// Register that a break target found at the given stop in the
|
2009-03-10 12:11:56 +00:00
|
|
|
// target stack has been used from the top of the target stack. Add
|
|
|
|
// the break target to any TargetCollectors passed on the stack.
|
2009-05-15 14:58:02 +00:00
|
|
|
for (Target* t = target_stack_; t != stop; t = t->previous()) {
|
|
|
|
TargetCollector* collector = t->node()->AsTargetCollector();
|
2012-06-11 12:42:31 +00:00
|
|
|
if (collector != NULL) collector->AddTarget(target, zone());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
Expression* Parser::NewThrowReferenceError(Handle<String> message) {
|
2013-02-28 17:03:34 +00:00
|
|
|
return NewThrowError(isolate()->factory()->MakeReferenceError_string(),
|
2013-06-06 13:28:22 +00:00
|
|
|
message, HandleVector<Object>(NULL, 0));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
Expression* Parser::NewThrowSyntaxError(Handle<String> message,
|
2008-07-03 15:10:15 +00:00
|
|
|
Handle<Object> first) {
|
|
|
|
int argc = first.is_null() ? 0 : 1;
|
|
|
|
Vector< Handle<Object> > arguments = HandleVector<Object>(&first, argc);
|
2011-03-18 20:35:07 +00:00
|
|
|
return NewThrowError(
|
2013-06-06 13:28:22 +00:00
|
|
|
isolate()->factory()->MakeSyntaxError_string(), message, arguments);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 13:28:22 +00:00
|
|
|
Expression* Parser::NewThrowTypeError(Handle<String> message,
|
2008-07-03 15:10:15 +00:00
|
|
|
Handle<Object> first,
|
|
|
|
Handle<Object> second) {
|
|
|
|
ASSERT(!first.is_null() && !second.is_null());
|
|
|
|
Handle<Object> elements[] = { first, second };
|
|
|
|
Vector< Handle<Object> > arguments =
|
|
|
|
HandleVector<Object>(elements, ARRAY_SIZE(elements));
|
2011-03-18 20:35:07 +00:00
|
|
|
return NewThrowError(
|
2013-06-06 13:28:22 +00:00
|
|
|
isolate()->factory()->MakeTypeError_string(), message, arguments);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Expression* Parser::NewThrowError(Handle<String> constructor,
|
2013-06-06 13:28:22 +00:00
|
|
|
Handle<String> message,
|
2008-07-03 15:10:15 +00:00
|
|
|
Vector< Handle<Object> > arguments) {
|
|
|
|
int argc = arguments.length();
|
2011-03-18 20:35:07 +00:00
|
|
|
Handle<FixedArray> elements = isolate()->factory()->NewFixedArray(argc,
|
|
|
|
TENURED);
|
2008-07-03 15:10:15 +00:00
|
|
|
for (int i = 0; i < argc; i++) {
|
|
|
|
Handle<Object> element = arguments[i];
|
|
|
|
if (!element.is_null()) {
|
2011-02-10 15:02:13 +00:00
|
|
|
elements->set(i, *element);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
2012-01-26 21:47:57 +00:00
|
|
|
Handle<JSArray> array = isolate()->factory()->NewJSArrayWithElements(
|
|
|
|
elements, FAST_ELEMENTS, TENURED);
|
2011-02-10 15:02:13 +00:00
|
|
|
|
2013-10-14 09:24:58 +00:00
|
|
|
int pos = position();
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<Expression*>* args = new(zone()) ZoneList<Expression*>(2, zone());
|
2013-10-14 09:24:58 +00:00
|
|
|
args->Add(factory()->NewLiteral(message, pos), zone());
|
|
|
|
args->Add(factory()->NewLiteral(array, pos), zone());
|
2012-02-08 09:56:33 +00:00
|
|
|
CallRuntime* call_constructor =
|
2013-10-14 09:24:58 +00:00
|
|
|
factory()->NewCallRuntime(constructor, NULL, args, pos);
|
|
|
|
return factory()->NewThrow(call_constructor, pos);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2013-07-05 09:52:11 +00:00
|
|
|
|
2008-11-25 11:07:48 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Regular expressions
|
|
|
|
|
|
|
|
|
|
|
|
RegExpParser::RegExpParser(FlatStringReader* in,
|
|
|
|
Handle<String>* error,
|
2012-06-20 08:58:41 +00:00
|
|
|
bool multiline,
|
|
|
|
Zone* zone)
|
2013-09-04 07:05:11 +00:00
|
|
|
: isolate_(zone->isolate()),
|
2012-06-20 08:58:41 +00:00
|
|
|
zone_(zone),
|
2011-03-18 20:35:07 +00:00
|
|
|
error_(error),
|
|
|
|
captures_(NULL),
|
|
|
|
in_(in),
|
|
|
|
current_(kEndMarker),
|
|
|
|
next_pos_(0),
|
|
|
|
capture_count_(0),
|
|
|
|
has_more_(true),
|
|
|
|
multiline_(multiline),
|
|
|
|
simple_(false),
|
|
|
|
contains_anchor_(false),
|
|
|
|
is_scanned_for_captures_(false),
|
|
|
|
failed_(false) {
|
2010-12-20 10:44:41 +00:00
|
|
|
Advance();
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uc32 RegExpParser::Next() {
|
|
|
|
if (has_next()) {
|
|
|
|
return in()->Get(next_pos_);
|
|
|
|
} else {
|
|
|
|
return kEndMarker;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpParser::Advance() {
|
|
|
|
if (next_pos_ < in()->length()) {
|
2011-03-18 20:35:07 +00:00
|
|
|
StackLimitCheck check(isolate());
|
2008-12-01 15:32:20 +00:00
|
|
|
if (check.HasOverflowed()) {
|
2011-03-18 20:35:07 +00:00
|
|
|
ReportError(CStrVector(Isolate::kStackOverflowMessage));
|
2012-06-20 08:58:41 +00:00
|
|
|
} else if (zone()->excess_allocation()) {
|
2008-12-01 15:32:20 +00:00
|
|
|
ReportError(CStrVector("Regular expression too large"));
|
|
|
|
} else {
|
|
|
|
current_ = in()->Get(next_pos_);
|
|
|
|
next_pos_++;
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
} else {
|
|
|
|
current_ = kEndMarker;
|
|
|
|
has_more_ = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpParser::Reset(int pos) {
|
|
|
|
next_pos_ = pos;
|
2013-05-27 10:53:37 +00:00
|
|
|
has_more_ = (pos < in()->length());
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void RegExpParser::Advance(int dist) {
|
2010-12-20 10:44:41 +00:00
|
|
|
next_pos_ += dist - 1;
|
|
|
|
Advance();
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-12 10:22:56 +00:00
|
|
|
bool RegExpParser::simple() {
|
|
|
|
return simple_;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
2013-07-05 09:52:11 +00:00
|
|
|
|
2008-12-01 15:32:20 +00:00
|
|
|
RegExpTree* RegExpParser::ReportError(Vector<const char> message) {
|
|
|
|
failed_ = true;
|
2011-03-18 20:35:07 +00:00
|
|
|
*error_ = isolate()->factory()->NewStringFromAscii(message, NOT_TENURED);
|
2008-12-01 15:32:20 +00:00
|
|
|
// Zip to the end to make sure the no more input is read.
|
|
|
|
current_ = kEndMarker;
|
|
|
|
next_pos_ = in()->length();
|
2008-11-25 11:07:48 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Pattern ::
|
|
|
|
// Disjunction
|
2008-12-01 15:32:20 +00:00
|
|
|
RegExpTree* RegExpParser::ParsePattern() {
|
|
|
|
RegExpTree* result = ParseDisjunction(CHECK_FAILED);
|
2009-07-03 11:09:34 +00:00
|
|
|
ASSERT(!has_more());
|
2009-03-04 09:52:01 +00:00
|
|
|
// If the result of parsing is a literal string atom, and it has the
|
|
|
|
// same length as the input, then the atom is identical to the input.
|
|
|
|
if (result->IsAtom() && result->AsAtom()->length() == in()->length()) {
|
|
|
|
simple_ = true;
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Disjunction ::
|
|
|
|
// Alternative
|
|
|
|
// Alternative | Disjunction
|
|
|
|
// Alternative ::
|
|
|
|
// [empty]
|
|
|
|
// Term Alternative
|
|
|
|
// Term ::
|
|
|
|
// Assertion
|
|
|
|
// Atom
|
|
|
|
// Atom Quantifier
|
2008-12-01 15:32:20 +00:00
|
|
|
RegExpTree* RegExpParser::ParseDisjunction() {
|
2009-07-03 08:18:35 +00:00
|
|
|
// Used to store current state while parsing subexpressions.
|
2012-06-04 14:42:58 +00:00
|
|
|
RegExpParserState initial_state(NULL, INITIAL, 0, zone());
|
2009-07-03 08:18:35 +00:00
|
|
|
RegExpParserState* stored_state = &initial_state;
|
|
|
|
// Cache the builder in a local variable for quick access.
|
|
|
|
RegExpBuilder* builder = initial_state.builder();
|
2008-11-25 11:07:48 +00:00
|
|
|
while (true) {
|
|
|
|
switch (current()) {
|
|
|
|
case kEndMarker:
|
2009-07-03 08:18:35 +00:00
|
|
|
if (stored_state->IsSubexpression()) {
|
|
|
|
// Inside a parenthesized group when hitting end of input.
|
|
|
|
ReportError(CStrVector("Unterminated group") CHECK_FAILED);
|
|
|
|
}
|
|
|
|
ASSERT_EQ(INITIAL, stored_state->group_type());
|
|
|
|
// Parsing completed successfully.
|
|
|
|
return builder->ToRegExp();
|
|
|
|
case ')': {
|
|
|
|
if (!stored_state->IsSubexpression()) {
|
2009-07-03 11:09:34 +00:00
|
|
|
ReportError(CStrVector("Unmatched ')'") CHECK_FAILED);
|
2009-07-03 08:18:35 +00:00
|
|
|
}
|
|
|
|
ASSERT_NE(INITIAL, stored_state->group_type());
|
|
|
|
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
2009-07-03 08:18:35 +00:00
|
|
|
// End disjunction parsing and convert builder content to new single
|
|
|
|
// regexp atom.
|
|
|
|
RegExpTree* body = builder->ToRegExp();
|
|
|
|
|
|
|
|
int end_capture_index = captures_started();
|
|
|
|
|
|
|
|
int capture_index = stored_state->capture_index();
|
2013-06-06 13:28:22 +00:00
|
|
|
SubexpressionType group_type = stored_state->group_type();
|
2009-07-03 08:18:35 +00:00
|
|
|
|
|
|
|
// Restore previous state.
|
|
|
|
stored_state = stored_state->previous_state();
|
|
|
|
builder = stored_state->builder();
|
|
|
|
|
|
|
|
// Build result of subexpression.
|
2013-06-06 13:28:22 +00:00
|
|
|
if (group_type == CAPTURE) {
|
2011-04-04 06:29:02 +00:00
|
|
|
RegExpCapture* capture = new(zone()) RegExpCapture(body, capture_index);
|
2009-07-03 08:18:35 +00:00
|
|
|
captures_->at(capture_index - 1) = capture;
|
|
|
|
body = capture;
|
2013-06-06 13:28:22 +00:00
|
|
|
} else if (group_type != GROUPING) {
|
|
|
|
ASSERT(group_type == POSITIVE_LOOKAHEAD ||
|
|
|
|
group_type == NEGATIVE_LOOKAHEAD);
|
|
|
|
bool is_positive = (group_type == POSITIVE_LOOKAHEAD);
|
2011-04-04 06:29:02 +00:00
|
|
|
body = new(zone()) RegExpLookahead(body,
|
2009-07-03 08:18:35 +00:00
|
|
|
is_positive,
|
|
|
|
end_capture_index - capture_index,
|
|
|
|
capture_index);
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAtom(body);
|
2011-02-16 08:10:47 +00:00
|
|
|
// For compatability with JSC and ES3, we allow quantifiers after
|
|
|
|
// lookaheads, and break in all cases.
|
2009-07-03 08:18:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '|': {
|
|
|
|
Advance();
|
|
|
|
builder->NewAlternative();
|
2008-11-25 11:07:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case '*':
|
|
|
|
case '+':
|
|
|
|
case '?':
|
2009-07-22 12:33:16 +00:00
|
|
|
return ReportError(CStrVector("Nothing to repeat"));
|
2008-11-25 11:07:48 +00:00
|
|
|
case '^': {
|
|
|
|
Advance();
|
2009-02-03 11:43:55 +00:00
|
|
|
if (multiline_) {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAssertion(
|
2011-04-04 06:29:02 +00:00
|
|
|
new(zone()) RegExpAssertion(RegExpAssertion::START_OF_LINE));
|
2009-02-03 11:43:55 +00:00
|
|
|
} else {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAssertion(
|
2011-04-04 06:29:02 +00:00
|
|
|
new(zone()) RegExpAssertion(RegExpAssertion::START_OF_INPUT));
|
2009-02-03 11:43:55 +00:00
|
|
|
set_contains_anchor();
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case '$': {
|
|
|
|
Advance();
|
2013-06-06 13:28:22 +00:00
|
|
|
RegExpAssertion::AssertionType assertion_type =
|
2008-11-27 10:35:06 +00:00
|
|
|
multiline_ ? RegExpAssertion::END_OF_LINE :
|
|
|
|
RegExpAssertion::END_OF_INPUT;
|
2013-06-06 13:28:22 +00:00
|
|
|
builder->AddAssertion(new(zone()) RegExpAssertion(assertion_type));
|
2008-11-25 11:07:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
case '.': {
|
|
|
|
Advance();
|
|
|
|
// everything except \x0a, \x0d, \u2028 and \u2029
|
2011-05-23 13:00:11 +00:00
|
|
|
ZoneList<CharacterRange>* ranges =
|
2012-06-11 12:42:31 +00:00
|
|
|
new(zone()) ZoneList<CharacterRange>(2, zone());
|
|
|
|
CharacterRange::AddClassEscape('.', ranges, zone());
|
2011-04-04 06:29:02 +00:00
|
|
|
RegExpTree* atom = new(zone()) RegExpCharacterClass(ranges, false);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAtom(atom);
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '(': {
|
2013-06-06 13:28:22 +00:00
|
|
|
SubexpressionType subexpr_type = CAPTURE;
|
2009-07-03 08:18:35 +00:00
|
|
|
Advance();
|
|
|
|
if (current() == '?') {
|
|
|
|
switch (Next()) {
|
|
|
|
case ':':
|
2013-06-06 13:28:22 +00:00
|
|
|
subexpr_type = GROUPING;
|
2009-07-03 08:18:35 +00:00
|
|
|
break;
|
|
|
|
case '=':
|
2013-06-06 13:28:22 +00:00
|
|
|
subexpr_type = POSITIVE_LOOKAHEAD;
|
2009-07-03 08:18:35 +00:00
|
|
|
break;
|
|
|
|
case '!':
|
2013-06-06 13:28:22 +00:00
|
|
|
subexpr_type = NEGATIVE_LOOKAHEAD;
|
2009-07-03 08:18:35 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ReportError(CStrVector("Invalid group") CHECK_FAILED);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Advance(2);
|
|
|
|
} else {
|
|
|
|
if (captures_ == NULL) {
|
2012-06-11 12:42:31 +00:00
|
|
|
captures_ = new(zone()) ZoneList<RegExpCapture*>(2, zone());
|
2009-07-03 08:18:35 +00:00
|
|
|
}
|
|
|
|
if (captures_started() >= kMaxCaptures) {
|
|
|
|
ReportError(CStrVector("Too many captures") CHECK_FAILED);
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
captures_->Add(NULL, zone());
|
2009-07-03 08:18:35 +00:00
|
|
|
}
|
|
|
|
// Store current state and begin new disjunction parsing.
|
2013-06-06 13:28:22 +00:00
|
|
|
stored_state = new(zone()) RegExpParserState(stored_state, subexpr_type,
|
2012-06-04 14:42:58 +00:00
|
|
|
captures_started(), zone());
|
2009-07-03 08:18:35 +00:00
|
|
|
builder = stored_state->builder();
|
2011-02-16 08:10:47 +00:00
|
|
|
continue;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
case '[': {
|
2008-12-01 15:32:20 +00:00
|
|
|
RegExpTree* atom = ParseCharacterClass(CHECK_FAILED);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAtom(atom);
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Atom ::
|
|
|
|
// \ AtomEscape
|
|
|
|
case '\\':
|
|
|
|
switch (Next()) {
|
|
|
|
case kEndMarker:
|
2009-07-22 12:33:16 +00:00
|
|
|
return ReportError(CStrVector("\\ at end of pattern"));
|
2008-11-25 11:07:48 +00:00
|
|
|
case 'b':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAssertion(
|
2011-04-04 06:29:02 +00:00
|
|
|
new(zone()) RegExpAssertion(RegExpAssertion::BOUNDARY));
|
2008-11-25 11:07:48 +00:00
|
|
|
continue;
|
|
|
|
case 'B':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAssertion(
|
2011-04-04 06:29:02 +00:00
|
|
|
new(zone()) RegExpAssertion(RegExpAssertion::NON_BOUNDARY));
|
2008-11-25 11:07:48 +00:00
|
|
|
continue;
|
2011-02-16 08:10:47 +00:00
|
|
|
// AtomEscape ::
|
|
|
|
// CharacterClassEscape
|
|
|
|
//
|
|
|
|
// CharacterClassEscape :: one of
|
|
|
|
// d D s S w W
|
2008-11-25 11:07:48 +00:00
|
|
|
case 'd': case 'D': case 's': case 'S': case 'w': case 'W': {
|
|
|
|
uc32 c = Next();
|
|
|
|
Advance(2);
|
2011-05-23 13:00:11 +00:00
|
|
|
ZoneList<CharacterRange>* ranges =
|
2012-06-11 12:42:31 +00:00
|
|
|
new(zone()) ZoneList<CharacterRange>(2, zone());
|
|
|
|
CharacterRange::AddClassEscape(c, ranges, zone());
|
2011-04-04 06:29:02 +00:00
|
|
|
RegExpTree* atom = new(zone()) RegExpCharacterClass(ranges, false);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAtom(atom);
|
2009-07-03 11:09:34 +00:00
|
|
|
break;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
case '1': case '2': case '3': case '4': case '5': case '6':
|
|
|
|
case '7': case '8': case '9': {
|
|
|
|
int index = 0;
|
|
|
|
if (ParseBackReferenceIndex(&index)) {
|
2009-07-03 08:18:35 +00:00
|
|
|
RegExpCapture* capture = NULL;
|
|
|
|
if (captures_ != NULL && index <= captures_->length()) {
|
|
|
|
capture = captures_->at(index - 1);
|
|
|
|
}
|
|
|
|
if (capture == NULL) {
|
|
|
|
builder->AddEmpty();
|
2009-07-03 11:09:34 +00:00
|
|
|
break;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2011-04-04 06:29:02 +00:00
|
|
|
RegExpTree* atom = new(zone()) RegExpBackReference(capture);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddAtom(atom);
|
2009-07-03 11:09:34 +00:00
|
|
|
break;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
uc32 first_digit = Next();
|
|
|
|
if (first_digit == '8' || first_digit == '9') {
|
|
|
|
// Treat as identity escape
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(first_digit);
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance(2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// FALLTHROUGH
|
|
|
|
case '0': {
|
|
|
|
Advance();
|
|
|
|
uc32 octal = ParseOctalLiteral();
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(octal);
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// ControlEscape :: one of
|
|
|
|
// f n r t v
|
|
|
|
case 'f':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('\f');
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('\n');
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('\r');
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('\t');
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
Advance(2);
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('\v');
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
case 'c': {
|
2011-01-07 12:35:42 +00:00
|
|
|
Advance();
|
|
|
|
uc32 controlLetter = Next();
|
|
|
|
// Special case if it is an ASCII letter.
|
|
|
|
// Convert lower case letters to uppercase.
|
|
|
|
uc32 letter = controlLetter & ~('a' ^ 'A');
|
|
|
|
if (letter < 'A' || 'Z' < letter) {
|
|
|
|
// controlLetter is not in range 'A'-'Z' or 'a'-'z'.
|
|
|
|
// This is outside the specification. We match JSC in
|
|
|
|
// reading the backslash as a literal character instead
|
|
|
|
// of as starting an escape.
|
|
|
|
builder->AddCharacter('\\');
|
|
|
|
} else {
|
|
|
|
Advance(2);
|
|
|
|
builder->AddCharacter(controlLetter & 0x1f);
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'x': {
|
|
|
|
Advance(2);
|
|
|
|
uc32 value;
|
|
|
|
if (ParseHexEscape(2, &value)) {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(value);
|
2008-11-25 11:07:48 +00:00
|
|
|
} else {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('x');
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'u': {
|
|
|
|
Advance(2);
|
|
|
|
uc32 value;
|
|
|
|
if (ParseHexEscape(4, &value)) {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(value);
|
2008-11-25 11:07:48 +00:00
|
|
|
} else {
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter('u');
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// Identity escape.
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(Next());
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance(2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '{': {
|
|
|
|
int dummy;
|
|
|
|
if (ParseIntervalQuantifier(&dummy, &dummy)) {
|
2008-12-01 15:32:20 +00:00
|
|
|
ReportError(CStrVector("Nothing to repeat") CHECK_FAILED);
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
// fallthrough
|
|
|
|
}
|
|
|
|
default:
|
2009-07-03 08:18:35 +00:00
|
|
|
builder->AddCharacter(current());
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
break;
|
|
|
|
} // end switch(current())
|
|
|
|
|
|
|
|
int min;
|
|
|
|
int max;
|
|
|
|
switch (current()) {
|
|
|
|
// QuantifierPrefix ::
|
|
|
|
// *
|
|
|
|
// +
|
|
|
|
// ?
|
|
|
|
// {
|
|
|
|
case '*':
|
|
|
|
min = 0;
|
2008-12-17 10:59:14 +00:00
|
|
|
max = RegExpTree::kInfinity;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
min = 1;
|
2008-12-17 10:59:14 +00:00
|
|
|
max = RegExpTree::kInfinity;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
break;
|
|
|
|
case '?':
|
|
|
|
min = 0;
|
|
|
|
max = 1;
|
|
|
|
Advance();
|
|
|
|
break;
|
|
|
|
case '{':
|
|
|
|
if (ParseIntervalQuantifier(&min, &max)) {
|
2009-01-29 14:14:13 +00:00
|
|
|
if (max < min) {
|
|
|
|
ReportError(CStrVector("numbers out of order in {} quantifier.")
|
|
|
|
CHECK_FAILED);
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-06 13:28:22 +00:00
|
|
|
RegExpQuantifier::QuantifierType quantifier_type = RegExpQuantifier::GREEDY;
|
2008-11-25 11:07:48 +00:00
|
|
|
if (current() == '?') {
|
2013-06-06 13:28:22 +00:00
|
|
|
quantifier_type = RegExpQuantifier::NON_GREEDY;
|
2010-01-07 19:01:23 +00:00
|
|
|
Advance();
|
|
|
|
} else if (FLAG_regexp_possessive_quantifier && current() == '+') {
|
|
|
|
// FLAG_regexp_possessive_quantifier is a debug-only flag.
|
2013-06-06 13:28:22 +00:00
|
|
|
quantifier_type = RegExpQuantifier::POSSESSIVE;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
}
|
2013-06-06 13:28:22 +00:00
|
|
|
builder->AddQuantifierToAtom(min, max, quantifier_type);
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
// Currently only used in an ASSERT.
|
|
|
|
static bool IsSpecialClassEscape(uc32 c) {
|
|
|
|
switch (c) {
|
|
|
|
case 'd': case 'D':
|
|
|
|
case 's': case 'S':
|
|
|
|
case 'w': case 'W':
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
// In order to know whether an escape is a backreference or not we have to scan
|
|
|
|
// the entire regexp and find the number of capturing parentheses. However we
|
|
|
|
// don't want to scan the regexp twice unless it is necessary. This mini-parser
|
|
|
|
// is called when needed. It can see the difference between capturing and
|
|
|
|
// noncapturing parentheses and can skip character classes and backslash-escaped
|
|
|
|
// characters.
|
|
|
|
void RegExpParser::ScanForCaptures() {
|
2008-11-25 14:14:04 +00:00
|
|
|
// Start with captures started previous to current position
|
|
|
|
int capture_count = captures_started();
|
|
|
|
// Add count of captures after this position.
|
2008-11-25 11:07:48 +00:00
|
|
|
int n;
|
|
|
|
while ((n = current()) != kEndMarker) {
|
|
|
|
Advance();
|
|
|
|
switch (n) {
|
|
|
|
case '\\':
|
|
|
|
Advance();
|
|
|
|
break;
|
|
|
|
case '[': {
|
|
|
|
int c;
|
|
|
|
while ((c = current()) != kEndMarker) {
|
|
|
|
Advance();
|
|
|
|
if (c == '\\') {
|
|
|
|
Advance();
|
|
|
|
} else {
|
|
|
|
if (c == ']') break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '(':
|
2008-11-25 14:14:04 +00:00
|
|
|
if (current() != '?') capture_count++;
|
2008-11-25 11:07:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-11-25 14:14:04 +00:00
|
|
|
capture_count_ = capture_count;
|
2008-11-25 11:07:48 +00:00
|
|
|
is_scanned_for_captures_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool RegExpParser::ParseBackReferenceIndex(int* index_out) {
|
|
|
|
ASSERT_EQ('\\', current());
|
|
|
|
ASSERT('1' <= Next() && Next() <= '9');
|
2009-01-30 10:38:25 +00:00
|
|
|
// Try to parse a decimal literal that is no greater than the total number
|
|
|
|
// of left capturing parentheses in the input.
|
2008-11-25 11:07:48 +00:00
|
|
|
int start = position();
|
|
|
|
int value = Next() - '0';
|
|
|
|
Advance(2);
|
|
|
|
while (true) {
|
|
|
|
uc32 c = current();
|
|
|
|
if (IsDecimalDigit(c)) {
|
|
|
|
value = 10 * value + (c - '0');
|
2009-01-30 10:38:25 +00:00
|
|
|
if (value > kMaxCaptures) {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-11-25 14:14:04 +00:00
|
|
|
if (value > captures_started()) {
|
|
|
|
if (!is_scanned_for_captures_) {
|
|
|
|
int saved_position = position();
|
|
|
|
ScanForCaptures();
|
|
|
|
Reset(saved_position);
|
|
|
|
}
|
|
|
|
if (value > capture_count_) {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
*index_out = value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// QuantifierPrefix ::
|
|
|
|
// { DecimalDigits }
|
|
|
|
// { DecimalDigits , }
|
|
|
|
// { DecimalDigits , DecimalDigits }
|
2009-02-05 12:55:20 +00:00
|
|
|
//
|
|
|
|
// Returns true if parsing succeeds, and set the min_out and max_out
|
|
|
|
// values. Values are truncated to RegExpTree::kInfinity if they overflow.
|
2008-11-25 11:07:48 +00:00
|
|
|
bool RegExpParser::ParseIntervalQuantifier(int* min_out, int* max_out) {
|
|
|
|
ASSERT_EQ(current(), '{');
|
|
|
|
int start = position();
|
|
|
|
Advance();
|
|
|
|
int min = 0;
|
|
|
|
if (!IsDecimalDigit(current())) {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
while (IsDecimalDigit(current())) {
|
2009-02-05 12:55:20 +00:00
|
|
|
int next = current() - '0';
|
|
|
|
if (min > (RegExpTree::kInfinity - next) / 10) {
|
|
|
|
// Overflow. Skip past remaining decimal digits and return -1.
|
2009-02-05 13:24:13 +00:00
|
|
|
do {
|
|
|
|
Advance();
|
|
|
|
} while (IsDecimalDigit(current()));
|
2009-02-05 12:55:20 +00:00
|
|
|
min = RegExpTree::kInfinity;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
min = 10 * min + next;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
}
|
|
|
|
int max = 0;
|
|
|
|
if (current() == '}') {
|
|
|
|
max = min;
|
|
|
|
Advance();
|
|
|
|
} else if (current() == ',') {
|
|
|
|
Advance();
|
|
|
|
if (current() == '}') {
|
2008-12-17 10:59:14 +00:00
|
|
|
max = RegExpTree::kInfinity;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
} else {
|
|
|
|
while (IsDecimalDigit(current())) {
|
2009-02-05 12:55:20 +00:00
|
|
|
int next = current() - '0';
|
|
|
|
if (max > (RegExpTree::kInfinity - next) / 10) {
|
2009-02-05 13:24:13 +00:00
|
|
|
do {
|
|
|
|
Advance();
|
|
|
|
} while (IsDecimalDigit(current()));
|
|
|
|
max = RegExpTree::kInfinity;
|
2009-02-05 12:55:20 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
max = 10 * max + next;
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance();
|
|
|
|
}
|
|
|
|
if (current() != '}') {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Advance();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
*min_out = min;
|
|
|
|
*max_out = max;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uc32 RegExpParser::ParseOctalLiteral() {
|
|
|
|
ASSERT('0' <= current() && current() <= '7');
|
|
|
|
// For compatibility with some other browsers (not all), we parse
|
|
|
|
// up to three octal digits with a value below 256.
|
|
|
|
uc32 value = current() - '0';
|
|
|
|
Advance();
|
|
|
|
if ('0' <= current() && current() <= '7') {
|
|
|
|
value = value * 8 + current() - '0';
|
|
|
|
Advance();
|
|
|
|
if (value < 32 && '0' <= current() && current() <= '7') {
|
|
|
|
value = value * 8 + current() - '0';
|
|
|
|
Advance();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool RegExpParser::ParseHexEscape(int length, uc32 *value) {
|
|
|
|
int start = position();
|
|
|
|
uc32 val = 0;
|
|
|
|
bool done = false;
|
|
|
|
for (int i = 0; !done; i++) {
|
|
|
|
uc32 c = current();
|
|
|
|
int d = HexValue(c);
|
|
|
|
if (d < 0) {
|
|
|
|
Reset(start);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
val = val * 16 + d;
|
|
|
|
Advance();
|
|
|
|
if (i == length - 1) {
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*value = val;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-01 15:32:20 +00:00
|
|
|
uc32 RegExpParser::ParseClassCharacterEscape() {
|
2008-11-25 11:07:48 +00:00
|
|
|
ASSERT(current() == '\\');
|
|
|
|
ASSERT(has_next() && !IsSpecialClassEscape(Next()));
|
|
|
|
Advance();
|
|
|
|
switch (current()) {
|
|
|
|
case 'b':
|
|
|
|
Advance();
|
|
|
|
return '\b';
|
|
|
|
// ControlEscape :: one of
|
|
|
|
// f n r t v
|
|
|
|
case 'f':
|
|
|
|
Advance();
|
|
|
|
return '\f';
|
|
|
|
case 'n':
|
|
|
|
Advance();
|
|
|
|
return '\n';
|
|
|
|
case 'r':
|
|
|
|
Advance();
|
|
|
|
return '\r';
|
|
|
|
case 't':
|
|
|
|
Advance();
|
|
|
|
return '\t';
|
|
|
|
case 'v':
|
|
|
|
Advance();
|
|
|
|
return '\v';
|
2011-01-07 12:35:42 +00:00
|
|
|
case 'c': {
|
|
|
|
uc32 controlLetter = Next();
|
|
|
|
uc32 letter = controlLetter & ~('A' ^ 'a');
|
|
|
|
// For compatibility with JSC, inside a character class
|
|
|
|
// we also accept digits and underscore as control characters.
|
|
|
|
if ((controlLetter >= '0' && controlLetter <= '9') ||
|
|
|
|
controlLetter == '_' ||
|
|
|
|
(letter >= 'A' && letter <= 'Z')) {
|
|
|
|
Advance(2);
|
|
|
|
// Control letters mapped to ASCII control characters in the range
|
|
|
|
// 0x00-0x1f.
|
|
|
|
return controlLetter & 0x1f;
|
|
|
|
}
|
|
|
|
// We match JSC in reading the backslash as a literal
|
|
|
|
// character instead of as starting an escape.
|
|
|
|
return '\\';
|
|
|
|
}
|
2008-11-25 11:07:48 +00:00
|
|
|
case '0': case '1': case '2': case '3': case '4': case '5':
|
|
|
|
case '6': case '7':
|
|
|
|
// For compatibility, we interpret a decimal escape that isn't
|
|
|
|
// a back reference (and therefore either \0 or not valid according
|
|
|
|
// to the specification) as a 1..3 digit octal character code.
|
|
|
|
return ParseOctalLiteral();
|
|
|
|
case 'x': {
|
|
|
|
Advance();
|
|
|
|
uc32 value;
|
|
|
|
if (ParseHexEscape(2, &value)) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
// If \x is not followed by a two-digit hexadecimal, treat it
|
|
|
|
// as an identity escape.
|
|
|
|
return 'x';
|
|
|
|
}
|
|
|
|
case 'u': {
|
|
|
|
Advance();
|
|
|
|
uc32 value;
|
|
|
|
if (ParseHexEscape(4, &value)) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
// If \u is not followed by a four-digit hexadecimal, treat it
|
|
|
|
// as an identity escape.
|
|
|
|
return 'u';
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
// Extended identity escape. We accept any character that hasn't
|
|
|
|
// been matched by a more specific case, not just the subset required
|
|
|
|
// by the ECMAScript specification.
|
|
|
|
uc32 result = current();
|
|
|
|
Advance();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-01 15:32:20 +00:00
|
|
|
CharacterRange RegExpParser::ParseClassAtom(uc16* char_class) {
|
2008-12-01 14:29:28 +00:00
|
|
|
ASSERT_EQ(0, *char_class);
|
2008-11-25 11:07:48 +00:00
|
|
|
uc32 first = current();
|
|
|
|
if (first == '\\') {
|
|
|
|
switch (Next()) {
|
|
|
|
case 'w': case 'W': case 'd': case 'D': case 's': case 'S': {
|
2008-12-01 14:29:28 +00:00
|
|
|
*char_class = Next();
|
2008-11-25 11:07:48 +00:00
|
|
|
Advance(2);
|
2008-12-01 14:29:28 +00:00
|
|
|
return CharacterRange::Singleton(0); // Return dummy value.
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2008-12-18 14:30:53 +00:00
|
|
|
case kEndMarker:
|
2009-07-22 12:33:16 +00:00
|
|
|
return ReportError(CStrVector("\\ at end of pattern"));
|
2008-11-25 11:07:48 +00:00
|
|
|
default:
|
2008-12-01 15:32:20 +00:00
|
|
|
uc32 c = ParseClassCharacterEscape(CHECK_FAILED);
|
2008-11-25 11:07:48 +00:00
|
|
|
return CharacterRange::Singleton(c);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Advance();
|
|
|
|
return CharacterRange::Singleton(first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-13 08:33:32 +00:00
|
|
|
static const uc16 kNoCharClass = 0;
|
|
|
|
|
|
|
|
// Adds range or pre-defined character class to character ranges.
|
|
|
|
// If char_class is not kInvalidClass, it's interpreted as a class
|
|
|
|
// escape (i.e., 's' means whitespace, from '\s').
|
|
|
|
static inline void AddRangeOrEscape(ZoneList<CharacterRange>* ranges,
|
|
|
|
uc16 char_class,
|
2012-06-11 12:42:31 +00:00
|
|
|
CharacterRange range,
|
|
|
|
Zone* zone) {
|
2010-12-13 08:33:32 +00:00
|
|
|
if (char_class != kNoCharClass) {
|
2012-06-11 12:42:31 +00:00
|
|
|
CharacterRange::AddClassEscape(char_class, ranges, zone);
|
2010-12-13 08:33:32 +00:00
|
|
|
} else {
|
2012-06-11 12:42:31 +00:00
|
|
|
ranges->Add(range, zone);
|
2010-12-13 08:33:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-01 15:32:20 +00:00
|
|
|
RegExpTree* RegExpParser::ParseCharacterClass() {
|
2008-11-25 11:07:48 +00:00
|
|
|
static const char* kUnterminated = "Unterminated character class";
|
|
|
|
static const char* kRangeOutOfOrder = "Range out of order in character class";
|
|
|
|
|
|
|
|
ASSERT_EQ(current(), '[');
|
|
|
|
Advance();
|
|
|
|
bool is_negated = false;
|
|
|
|
if (current() == '^') {
|
|
|
|
is_negated = true;
|
|
|
|
Advance();
|
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
ZoneList<CharacterRange>* ranges =
|
|
|
|
new(zone()) ZoneList<CharacterRange>(2, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
while (has_more() && current() != ']') {
|
2010-12-13 08:33:32 +00:00
|
|
|
uc16 char_class = kNoCharClass;
|
2008-12-01 15:32:20 +00:00
|
|
|
CharacterRange first = ParseClassAtom(&char_class CHECK_FAILED);
|
2008-12-01 14:29:28 +00:00
|
|
|
if (current() == '-') {
|
|
|
|
Advance();
|
|
|
|
if (current() == kEndMarker) {
|
|
|
|
// If we reach the end we break out of the loop and let the
|
|
|
|
// following code report an error.
|
|
|
|
break;
|
|
|
|
} else if (current() == ']') {
|
2012-06-11 12:42:31 +00:00
|
|
|
AddRangeOrEscape(ranges, char_class, first, zone());
|
|
|
|
ranges->Add(CharacterRange::Singleton('-'), zone());
|
2008-12-01 14:29:28 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-12-13 08:33:32 +00:00
|
|
|
uc16 char_class_2 = kNoCharClass;
|
|
|
|
CharacterRange next = ParseClassAtom(&char_class_2 CHECK_FAILED);
|
|
|
|
if (char_class != kNoCharClass || char_class_2 != kNoCharClass) {
|
|
|
|
// Either end is an escaped character class. Treat the '-' verbatim.
|
2012-06-11 12:42:31 +00:00
|
|
|
AddRangeOrEscape(ranges, char_class, first, zone());
|
|
|
|
ranges->Add(CharacterRange::Singleton('-'), zone());
|
|
|
|
AddRangeOrEscape(ranges, char_class_2, next, zone());
|
2010-12-09 12:07:52 +00:00
|
|
|
continue;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2008-12-01 14:29:28 +00:00
|
|
|
if (first.from() > next.to()) {
|
2008-12-01 15:32:20 +00:00
|
|
|
return ReportError(CStrVector(kRangeOutOfOrder) CHECK_FAILED);
|
2008-12-01 14:29:28 +00:00
|
|
|
}
|
2012-06-11 12:42:31 +00:00
|
|
|
ranges->Add(CharacterRange::Range(first.from(), next.to()), zone());
|
2008-12-01 14:29:28 +00:00
|
|
|
} else {
|
2012-06-11 12:42:31 +00:00
|
|
|
AddRangeOrEscape(ranges, char_class, first, zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!has_more()) {
|
2008-12-01 15:32:20 +00:00
|
|
|
return ReportError(CStrVector(kUnterminated) CHECK_FAILED);
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
Advance();
|
|
|
|
if (ranges->length() == 0) {
|
2012-06-11 12:42:31 +00:00
|
|
|
ranges->Add(CharacterRange::Everything(), zone());
|
2008-11-25 11:07:48 +00:00
|
|
|
is_negated = !is_negated;
|
|
|
|
}
|
2011-04-04 06:29:02 +00:00
|
|
|
return new(zone()) RegExpCharacterClass(ranges, is_negated);
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// The Parser interface.
|
|
|
|
|
|
|
|
ParserMessage::~ParserMessage() {
|
|
|
|
for (int i = 0; i < args().length(); i++)
|
|
|
|
DeleteArray(args()[i]);
|
|
|
|
DeleteArray(args().start());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ScriptDataImpl::~ScriptDataImpl() {
|
2010-09-07 12:52:16 +00:00
|
|
|
if (owns_store_) store_.Dispose();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ScriptDataImpl::Length() {
|
2010-05-25 06:43:13 +00:00
|
|
|
return store_.length() * sizeof(unsigned);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-25 06:38:19 +00:00
|
|
|
const char* ScriptDataImpl::Data() {
|
|
|
|
return reinterpret_cast<const char*>(store_.start());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-11 12:13:24 +00:00
|
|
|
bool ScriptDataImpl::HasError() {
|
|
|
|
return has_error();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-15 10:54:35 +00:00
|
|
|
void ScriptDataImpl::Initialize() {
|
2010-09-17 12:55:27 +00:00
|
|
|
// Prepares state for use.
|
2010-11-23 11:46:36 +00:00
|
|
|
if (store_.length() >= PreparseDataConstants::kHeaderSize) {
|
|
|
|
function_index_ = PreparseDataConstants::kHeaderSize;
|
|
|
|
int symbol_data_offset = PreparseDataConstants::kHeaderSize
|
|
|
|
+ store_[PreparseDataConstants::kFunctionsSizeOffset];
|
2010-09-15 10:54:35 +00:00
|
|
|
if (store_.length() > symbol_data_offset) {
|
|
|
|
symbol_data_ = reinterpret_cast<byte*>(&store_[symbol_data_offset]);
|
|
|
|
} else {
|
|
|
|
// Partial preparse causes no symbol information.
|
|
|
|
symbol_data_ = reinterpret_cast<byte*>(&store_[0] + store_.length());
|
|
|
|
}
|
|
|
|
symbol_data_end_ = reinterpret_cast<byte*>(&store_[0] + store_.length());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ScriptDataImpl::ReadNumber(byte** source) {
|
|
|
|
// Reads a number from symbol_data_ in base 128. The most significant
|
|
|
|
// bit marks that there are more digits.
|
|
|
|
// If the first byte is 0x80 (kNumberTerminator), it would normally
|
|
|
|
// represent a leading zero. Since that is useless, and therefore won't
|
|
|
|
// appear as the first digit of any actual value, it is used to
|
|
|
|
// mark the end of the input stream.
|
|
|
|
byte* data = *source;
|
|
|
|
if (data >= symbol_data_end_) return -1;
|
|
|
|
byte input = *data;
|
2010-11-23 11:46:36 +00:00
|
|
|
if (input == PreparseDataConstants::kNumberTerminator) {
|
2010-09-15 10:54:35 +00:00
|
|
|
// End of stream marker.
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
int result = input & 0x7f;
|
|
|
|
data++;
|
|
|
|
while ((input & 0x80u) != 0) {
|
|
|
|
if (data >= symbol_data_end_) return -1;
|
|
|
|
input = *data;
|
|
|
|
result = (result << 7) | (input & 0x7f);
|
|
|
|
data++;
|
|
|
|
}
|
|
|
|
*source = data;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-19 09:02:59 +00:00
|
|
|
// Create a Scanner for the preparser to use as input, and preparse the source.
|
2013-09-04 07:05:11 +00:00
|
|
|
ScriptDataImpl* PreParserApi::PreParse(Isolate* isolate,
|
|
|
|
Utf16CharacterStream* source) {
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
CompleteParserRecorder recorder;
|
2011-11-24 13:24:30 +00:00
|
|
|
HistogramTimerScope timer(isolate->counters()->pre_parse());
|
2011-11-01 07:47:15 +00:00
|
|
|
Scanner scanner(isolate->unicode_cache());
|
2011-03-18 20:35:07 +00:00
|
|
|
intptr_t stack_limit = isolate->stack_guard()->real_climit();
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
preparser::PreParser preparser(&scanner, &recorder, stack_limit);
|
|
|
|
preparser.set_allow_lazy(true);
|
|
|
|
preparser.set_allow_generators(FLAG_harmony_generators);
|
2013-06-06 14:38:26 +00:00
|
|
|
preparser.set_allow_for_of(FLAG_harmony_iteration);
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
preparser.set_allow_harmony_scoping(FLAG_harmony_scoping);
|
2013-07-19 09:57:35 +00:00
|
|
|
preparser.set_allow_harmony_numeric_literals(FLAG_harmony_numeric_literals);
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
scanner.Initialize(source);
|
|
|
|
preparser::PreParser::PreParseResult result = preparser.PreParseProgram();
|
2011-11-25 09:36:31 +00:00
|
|
|
if (result == preparser::PreParser::kPreParseStackOverflow) {
|
2011-03-18 20:35:07 +00:00
|
|
|
isolate->StackOverflow();
|
2010-11-17 12:00:22 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extract the accumulated data from the recorder as a single
|
|
|
|
// contiguous vector that we are responsible for disposing.
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
Vector<unsigned> store = recorder.ExtractData();
|
2010-11-17 12:00:22 +00:00
|
|
|
return new ScriptDataImpl(store);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-27 12:33:48 +00:00
|
|
|
bool RegExpParser::ParseRegExp(FlatStringReader* input,
|
|
|
|
bool multiline,
|
2012-06-20 08:58:41 +00:00
|
|
|
RegExpCompileData* result,
|
|
|
|
Zone* zone) {
|
2008-11-25 11:07:48 +00:00
|
|
|
ASSERT(result != NULL);
|
2012-06-20 08:58:41 +00:00
|
|
|
RegExpParser parser(input, &result->error, multiline, zone);
|
2008-12-12 10:22:56 +00:00
|
|
|
RegExpTree* tree = parser.ParsePattern();
|
2008-12-01 15:32:20 +00:00
|
|
|
if (parser.failed()) {
|
2008-12-12 10:22:56 +00:00
|
|
|
ASSERT(tree == NULL);
|
2008-11-25 11:07:48 +00:00
|
|
|
ASSERT(!result->error.is_null());
|
|
|
|
} else {
|
2008-12-12 10:22:56 +00:00
|
|
|
ASSERT(tree != NULL);
|
2008-11-25 11:07:48 +00:00
|
|
|
ASSERT(result->error.is_null());
|
2008-12-12 10:22:56 +00:00
|
|
|
result->tree = tree;
|
|
|
|
int capture_count = parser.captures_started();
|
|
|
|
result->simple = tree->IsAtom() && parser.simple() && capture_count == 0;
|
2009-02-03 11:43:55 +00:00
|
|
|
result->contains_anchor = parser.contains_anchor();
|
2008-12-12 10:22:56 +00:00
|
|
|
result->capture_count = capture_count;
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
2008-12-01 15:32:20 +00:00
|
|
|
return !parser.failed();
|
2008-11-25 11:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
bool Parser::Parse() {
|
|
|
|
ASSERT(info()->function() == NULL);
|
2010-10-04 11:35:46 +00:00
|
|
|
FunctionLiteral* result = NULL;
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
if (info()->is_lazy()) {
|
|
|
|
ASSERT(!info()->is_eval());
|
|
|
|
if (info()->shared_info()->is_function()) {
|
|
|
|
result = ParseLazy();
|
2012-02-14 14:14:51 +00:00
|
|
|
} else {
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
result = ParseProgram();
|
2012-02-14 14:14:51 +00:00
|
|
|
}
|
2010-02-01 10:31:55 +00:00
|
|
|
} else {
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
ScriptDataImpl* pre_parse_data = info()->pre_parse_data();
|
|
|
|
set_pre_parse_data(pre_parse_data);
|
|
|
|
if (pre_parse_data != NULL && pre_parse_data->has_error()) {
|
|
|
|
Scanner::Location loc = pre_parse_data->MessageLocation();
|
|
|
|
const char* message = pre_parse_data->BuildMessage();
|
|
|
|
Vector<const char*> args = pre_parse_data->BuildArgs();
|
|
|
|
ReportMessageAt(loc, message, args);
|
2010-10-04 11:35:46 +00:00
|
|
|
DeleteArray(message);
|
|
|
|
for (int i = 0; i < args.length(); i++) {
|
|
|
|
DeleteArray(args[i]);
|
|
|
|
}
|
|
|
|
DeleteArray(args.start());
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
ASSERT(info()->isolate()->has_pending_exception());
|
2010-10-04 11:35:46 +00:00
|
|
|
} else {
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
result = ParseProgram();
|
2010-10-04 11:35:46 +00:00
|
|
|
}
|
2010-02-01 10:31:55 +00:00
|
|
|
}
|
Refactor parser mode configuration for correctness
This patch refactors the parser and preparser interface to be more
readable and type-safe. It has no behavior changes.
Previously, parsers and preparsers were configured via bitfield called
parser_flags in the Parser constructor, and flags in
PreParser::PreParseProgram, ParserApi::Parse, and ParserApi::PreParse.
This was error-prone in practice: six call sites passed incorrectly
typed values to this interface (a boolean FLAG value, a boolean false
and a boolean true value). None of these errors were caught by the
compiler because it's just an "int".
The parser flags interface was also awkward because it encoded a
language mode, but the language mode was only used to turn on harmony
scoping or not -- it wasn't used to actually set the parser's language
mode.
Fundamentally these errors came in because of the desire for a
procedural parser interface, in ParserApi. Because we need to be able
to configure the parser in various ways, the flags argument got added;
but no one understood how to use the flags properly. Also they were
only used by constructors: callers packed bits, and the constructors
unpacked them into booleans on the parser or preparser.
The solution is to allow parser construction, configuration, and
invocation to be separated. This patch does that.
It passes the existing tests.
BUG=
Review URL: https://codereview.chromium.org/13450007
Patch from Andy Wingo <wingo@igalia.com>.
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14151 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-04-05 13:01:06 +00:00
|
|
|
info()->SetFunction(result);
|
2010-10-04 11:35:46 +00:00
|
|
|
return (result != NULL);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} } // namespace v8::internal
|