mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-04-02 20:42:49 +00:00
Fixes for missing global object properties, various bug fixes.
This commit is contained in:
parent
a11195cc80
commit
d0c0b2fd5e
@ -224,7 +224,7 @@ void NameReference::emitTypeOf(ByteCodeGen *bcg)
|
||||
|
||||
void NameReference::emitDelete(ByteCodeGen *bcg)
|
||||
{
|
||||
bcg->addOp(DeleteOp);
|
||||
bcg->addOp(DeleteNameOp);
|
||||
bcg->addStringRef(mName);
|
||||
}
|
||||
|
||||
@ -266,7 +266,7 @@ void ElementReference::emitCodeSequence(ByteCodeGen *bcg)
|
||||
|
||||
void ElementReference::emitDelete(ByteCodeGen *bcg)
|
||||
{
|
||||
bcg->addOpAdjustDepth(DeleteElementOp, -mDepth);
|
||||
bcg->addOpAdjustDepth(DeleteElementOp, -(mDepth - 1));
|
||||
bcg->addShort(mDepth);
|
||||
}
|
||||
|
||||
@ -299,6 +299,7 @@ ByteCodeData gByteCodeData[OpCodeCount] = {
|
||||
{ -1, "NewThis", },
|
||||
{ -128, "NewInstance", },
|
||||
{ 0, "Delete", },
|
||||
{ 1, "DeleteName", },
|
||||
{ 0, "TypeOf", },
|
||||
{ -1, "InstanceOf", },
|
||||
{ -1, "As", },
|
||||
@ -362,23 +363,16 @@ ByteCodeModule::ByteCodeModule(ByteCodeGen *bcg, JSFunction *f)
|
||||
mCodeMap = new PC_Position[mCodeMapLength];
|
||||
memcpy(mCodeMap, bcg->mPC_Map->begin(), mCodeMapLength * sizeof(PC_Position));
|
||||
|
||||
mStringPoolContents = new String[bcg->mStringPoolContents.size()];
|
||||
|
||||
int index = 0;
|
||||
for (std::vector<String>::iterator s_i = bcg->mStringPoolContents.begin(),
|
||||
s_end = bcg->mStringPoolContents.end(); (s_i != s_end); s_i++, index++)
|
||||
mStringPoolContents[index] = *s_i;
|
||||
|
||||
mNumberPoolContents = new float64[bcg->mNumberPoolContents.size()];
|
||||
index = 0;
|
||||
for (std::vector<float64>::iterator f_i = bcg->mNumberPoolContents.begin(),
|
||||
f_end = bcg->mNumberPoolContents.end(); (f_i != f_end); f_i++, index++)
|
||||
mNumberPoolContents[index] = *f_i;
|
||||
|
||||
mLocalsCount = bcg->mScopeChain->countVars();
|
||||
mStackDepth = toUInt32(bcg->mStackMax);
|
||||
}
|
||||
|
||||
ByteCodeModule::~ByteCodeModule()
|
||||
{
|
||||
delete mCodeBase;
|
||||
delete mCodeMap;
|
||||
}
|
||||
|
||||
size_t ByteCodeModule::getPositionForPC(uint32 pc)
|
||||
{
|
||||
if (mCodeMapLength == 0)
|
||||
@ -461,6 +455,8 @@ void ByteCodeGen::addOp(uint8 op)
|
||||
|
||||
void ByteCodeGen::addNumberRef(float64 f)
|
||||
{
|
||||
addFloat64(f);
|
||||
/*
|
||||
NumberPool::iterator i = mNumberPool.find(f);
|
||||
if (i != mNumberPool.end())
|
||||
addLong(i->second);
|
||||
@ -469,10 +465,15 @@ void ByteCodeGen::addNumberRef(float64 f)
|
||||
mNumberPool[f] = mNumberPoolContents.size();
|
||||
mNumberPoolContents.push_back(f);
|
||||
}
|
||||
*/
|
||||
}
|
||||
|
||||
void ByteCodeGen::addStringRef(const String &str)
|
||||
{
|
||||
const StringAtom &s = m_cx->mWorld.identifiers[str];
|
||||
addPointer((void *)(&s));
|
||||
|
||||
/*
|
||||
StringPool::iterator i = mStringPool.find(str);
|
||||
if (i != mStringPool.end())
|
||||
addLong(i->second);
|
||||
@ -481,6 +482,7 @@ void ByteCodeGen::addStringRef(const String &str)
|
||||
mStringPool[str] = mStringPoolContents.size();
|
||||
mStringPoolContents.push_back(str);
|
||||
}
|
||||
*/
|
||||
}
|
||||
|
||||
|
||||
@ -889,7 +891,7 @@ bool ByteCodeGen::genCodeForStatement(StmtNode *p, ByteCodeGen *static_cg, uint3
|
||||
objectWriteRef->emitCodeSequence(this);
|
||||
addOp(GetInvokePropertyOp);
|
||||
// addIdentifierRef(widenCString("Iterator"), widenCString("forin"));
|
||||
addStringRef(widenCString("forin"));
|
||||
addStringRef(m_cx->Forin_StringAtom);
|
||||
addOpAdjustDepth(InvokeOp, -1);
|
||||
addLong(0);
|
||||
addByte(Explicit);
|
||||
@ -902,7 +904,7 @@ bool ByteCodeGen::genCodeForStatement(StmtNode *p, ByteCodeGen *static_cg, uint3
|
||||
setLabel(labelAtTopOfBlock);
|
||||
iteratorReadRef->emitCodeSequence(this);
|
||||
addOp(GetPropertyOp);
|
||||
addStringRef(widenCString("value"));
|
||||
addStringRef(m_cx->Value_StringAtom);
|
||||
value->emitCodeSequence(this);
|
||||
addOp(PopOp);
|
||||
|
||||
@ -915,7 +917,7 @@ bool ByteCodeGen::genCodeForStatement(StmtNode *p, ByteCodeGen *static_cg, uint3
|
||||
setLabel(labelAtIncrement);
|
||||
objectReadRef->emitCodeSequence(this);
|
||||
addOp(GetInvokePropertyOp);
|
||||
addStringRef(widenCString("next"));
|
||||
addStringRef(m_cx->Next_StringAtom);
|
||||
iteratorReadRef->emitCodeSequence(this);
|
||||
addOpAdjustDepth(InvokeOp, -2);
|
||||
addLong(1);
|
||||
@ -936,7 +938,7 @@ bool ByteCodeGen::genCodeForStatement(StmtNode *p, ByteCodeGen *static_cg, uint3
|
||||
setLabel(breakLabel);
|
||||
objectReadRef->emitCodeSequence(this);
|
||||
addOp(GetInvokePropertyOp);
|
||||
addStringRef(widenCString("done"));
|
||||
addStringRef(m_cx->Done_StringAtom);
|
||||
iteratorReadRef->emitCodeSequence(this);
|
||||
addOpAdjustDepth(InvokeOp, -2);
|
||||
addLong(1);
|
||||
@ -1348,6 +1350,11 @@ bool ByteCodeGen::genCodeForStatement(StmtNode *p, ByteCodeGen *static_cg, uint3
|
||||
Reference *ByteCodeGen::genReference(ExprNode *p, Access acc)
|
||||
{
|
||||
switch (p->getKind()) {
|
||||
case ExprNode::parentheses:
|
||||
{
|
||||
UnaryExprNode *u = checked_cast<UnaryExprNode *>(p);
|
||||
return genReference(u->op, acc);
|
||||
}
|
||||
case ExprNode::index:
|
||||
{
|
||||
InvokeExprNode *i = checked_cast<InvokeExprNode *>(p);
|
||||
@ -2085,6 +2092,8 @@ BinaryOpEquals:
|
||||
{
|
||||
InvokeExprNode *i = checked_cast<InvokeExprNode *>(p);
|
||||
Reference *ref = genReference(i->op, Read);
|
||||
if (ref == NULL)
|
||||
m_cx->reportError(Exception::referenceError, "Unrecogizable call target", p->pos);
|
||||
|
||||
ref->emitInvokeSequence(this);
|
||||
|
||||
@ -2250,8 +2259,8 @@ BinaryOpEquals:
|
||||
}
|
||||
fnc->setArgCounts(reqArgCount, optArgCount, (f->function.restParameter != NULL));
|
||||
|
||||
if (mScopeChain->isPossibleUncheckedFunction(&f->function))
|
||||
fnc->setIsPrototype(true);
|
||||
if (mScopeChain->isPossibleUncheckedFunction(&f->function))
|
||||
fnc->setIsPrototype(true);
|
||||
|
||||
m_cx->buildRuntimeForFunction(f->function, fnc);
|
||||
ByteCodeGen bcg(m_cx, mScopeChain);
|
||||
@ -2311,6 +2320,22 @@ BinaryOpEquals:
|
||||
addOp(JuxtaposeOp);
|
||||
}
|
||||
break;
|
||||
case ExprNode::comma:
|
||||
{
|
||||
BinaryExprNode *c = checked_cast<BinaryExprNode *>(p);
|
||||
genExpr(c->op1);
|
||||
return genExpr(c->op2);
|
||||
}
|
||||
break;
|
||||
case ExprNode::Void:
|
||||
{
|
||||
UnaryExprNode *v = checked_cast<UnaryExprNode *>(p);
|
||||
genExpr(v->op);
|
||||
addOp(PopOp);
|
||||
addOp(LoadConstantUndefinedOp);
|
||||
return Object_Type;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
NOT_REACHED("Not Implemented Yet");
|
||||
}
|
||||
@ -2428,13 +2453,14 @@ uint32 printInstruction(Formatter &f, uint32 i, const ByteCodeModule& bcm)
|
||||
case SetPropertyOp:
|
||||
case LoadConstantStringOp:
|
||||
case DeleteOp:
|
||||
case DeleteNameOp:
|
||||
f << *bcm.getString(bcm.getLong(i));
|
||||
i += 4;
|
||||
break;
|
||||
|
||||
case LoadConstantNumberOp:
|
||||
f << bcm.getNumber(bcm.getLong(i));
|
||||
i += 4;
|
||||
f << bcm.getNumber(&bcm.mCodeBase[i]);
|
||||
i += 8;
|
||||
break;
|
||||
|
||||
case LoadTypeOp:
|
||||
|
@ -95,7 +95,8 @@ typedef enum {
|
||||
NewObjectOp, // --> <object>
|
||||
NewThisOp, // <type> -->
|
||||
NewInstanceOp, // <argc> <type> <args> --> <object>
|
||||
DeleteOp, // <index> <object> --> <boolean>
|
||||
DeleteOp, // <poolindex> <object> --> <boolean>
|
||||
DeleteNameOp, // <poolindex> --> <boolean>
|
||||
TypeOfOp, // <object> --> <string>
|
||||
InstanceOfOp, // <object> <object> --> <boolean>
|
||||
AsOp, // <object> <type> --> <object>
|
||||
@ -171,6 +172,7 @@ extern ByteCodeData gByteCodeData[OpCodeCount];
|
||||
public:
|
||||
|
||||
ByteCodeModule(ByteCodeGen *bcg, JSFunction *f);
|
||||
~ByteCodeModule();
|
||||
|
||||
#ifdef DEBUG
|
||||
void* operator new(size_t s) { void *t = STD::malloc(s); trace_alloc("ByteCodeModule", s, t); return t; }
|
||||
@ -180,8 +182,8 @@ extern ByteCodeData gByteCodeData[OpCodeCount];
|
||||
uint32 getLong(uint32 index) const { return *((uint32 *)&mCodeBase[index]); }
|
||||
uint16 getShort(uint32 index) const { return *((uint16 *)&mCodeBase[index]); }
|
||||
int32 getOffset(uint32 index) const { return *((int32 *)&mCodeBase[index]); }
|
||||
const String *getString(uint32 index) const { return &mStringPoolContents[index]; }
|
||||
float64 getNumber(uint32 index) const { return mNumberPoolContents[index]; }
|
||||
const String *getString(uint32 index) const { return (const String *)(index); }
|
||||
float64 getNumber(uint8 *p) const { return *((float64 *)p); }
|
||||
|
||||
void setSource(const String &source, const String &sourceLocation)
|
||||
{
|
||||
@ -200,9 +202,6 @@ extern ByteCodeData gByteCodeData[OpCodeCount];
|
||||
uint8 *mCodeBase;
|
||||
uint32 mLength;
|
||||
|
||||
String *mStringPoolContents;
|
||||
float64 *mNumberPoolContents;
|
||||
|
||||
PC_Position *mCodeMap;
|
||||
uint32 mCodeMapLength;
|
||||
|
||||
@ -259,6 +258,12 @@ extern ByteCodeData gByteCodeData[OpCodeCount];
|
||||
mStackMax(0)
|
||||
{ }
|
||||
|
||||
ByteCodeGen::~ByteCodeGen()
|
||||
{
|
||||
delete mBuffer;
|
||||
delete mPC_Map;
|
||||
}
|
||||
|
||||
#ifdef DEBUG
|
||||
void* operator new(size_t s) { void *t = STD::malloc(s); trace_alloc("ByteCodeGen", s, t); return t; }
|
||||
void operator delete(void* t) { trace_release("ByteCodeGen", t); STD::free(t); }
|
||||
@ -344,6 +349,7 @@ extern ByteCodeData gByteCodeData[OpCodeCount];
|
||||
|
||||
void addPointer(void *v) { ASSERT(sizeof(void *) == sizeof(uint32)); addLong((uint32)(v)); } // XXX Pointer size dependant !!!
|
||||
|
||||
void addFloat64(float64 v) { mBuffer->insert(mBuffer->end(), (uint8 *)&v, (uint8 *)(&v) + sizeof(float64)); }
|
||||
void addLong(uint32 v) { mBuffer->insert(mBuffer->end(), (uint8 *)&v, (uint8 *)(&v) + sizeof(uint32)); }
|
||||
void addOffset(int32 v) { mBuffer->insert(mBuffer->end(), (uint8 *)&v, (uint8 *)(&v) + sizeof(int32)); }
|
||||
void setOffset(uint32 index, int32 v) { *((int32 *)(mBuffer->begin() + index)) = v; } // XXX dubious pointer usage
|
||||
|
@ -67,6 +67,39 @@ namespace JS2Runtime {
|
||||
|
||||
inline char narrow(char16 ch) { return char(ch); }
|
||||
|
||||
JSValue Context::readEvalString(const String &str, const String& fileName, ScopeChain *scopeChain, const JSValue& thisValue)
|
||||
{
|
||||
JSValue result = kUndefinedValue;
|
||||
|
||||
Arena a;
|
||||
Parser p(mWorld, a, mFlags, str, fileName);
|
||||
mReader = &p.lexer.reader;
|
||||
StmtNode *parsedStatements = p.parseProgram();
|
||||
ASSERT(p.lexer.peek(true).hasKind(Token::end));
|
||||
if (mDebugFlag)
|
||||
{
|
||||
PrettyPrinter f(stdOut, 30);
|
||||
{
|
||||
PrettyPrinter::Block b(f, 2);
|
||||
f << "Program =";
|
||||
f.linearBreak(1);
|
||||
StmtNode::printStatements(f, parsedStatements);
|
||||
}
|
||||
f.end();
|
||||
stdOut << '\n';
|
||||
}
|
||||
|
||||
buildRuntime(parsedStatements);
|
||||
JS2Runtime::ByteCodeModule* bcm = genCode(parsedStatements, fileName);
|
||||
if (bcm) {
|
||||
setReader(NULL);
|
||||
bcm->setSource(str, fileName);
|
||||
result = interpret(bcm, 0, scopeChain, thisValue, NULL, 0);
|
||||
delete bcm;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
JSValue Context::readEvalFile(const String& fileName)
|
||||
{
|
||||
String buffer;
|
||||
@ -81,33 +114,7 @@ JSValue Context::readEvalFile(const String& fileName)
|
||||
while ((ch = getc(f)) != EOF)
|
||||
buffer += static_cast<char>(ch);
|
||||
fclose(f);
|
||||
|
||||
Arena a;
|
||||
Parser p(mWorld, a, mFlags, buffer, fileName);
|
||||
mReader = &p.lexer.reader;
|
||||
StmtNode *parsedStatements = p.parseProgram();
|
||||
ASSERT(p.lexer.peek(true).hasKind(Token::end));
|
||||
if (mDebugFlag)
|
||||
{
|
||||
PrettyPrinter f(stdOut, 30);
|
||||
{
|
||||
PrettyPrinter::Block b(f, 2);
|
||||
f << "Program =";
|
||||
f.linearBreak(1);
|
||||
StmtNode::printStatements(f, parsedStatements);
|
||||
}
|
||||
f.end();
|
||||
stdOut << '\n';
|
||||
}
|
||||
|
||||
buildRuntime(parsedStatements);
|
||||
JS2Runtime::ByteCodeModule* bcm = genCode(parsedStatements, fileName);
|
||||
if (bcm) {
|
||||
setReader(NULL);
|
||||
bcm->setSource(buffer, fileName);
|
||||
result = interpret(bcm, 0, NULL, JSValue(getGlobalObject()), NULL, 0);
|
||||
delete bcm;
|
||||
}
|
||||
result = readEvalString(buffer, fileName, NULL, JSValue(getGlobalObject()));
|
||||
}
|
||||
return result;
|
||||
}
|
||||
@ -188,6 +195,7 @@ JSValue Context::interpret(JS2Runtime::ByteCodeModule *bcm, int offset, ScopeCha
|
||||
if (mThis.isObject())
|
||||
mScopeChain->addScope(mThis.object);
|
||||
// mScopeChain->addScope(mActivationStack.top());
|
||||
|
||||
mCurModule = bcm;
|
||||
uint8 *pc = mCurModule->mCodeBase + offset;
|
||||
uint8 *endPC = mCurModule->mCodeBase + mCurModule->mLength;
|
||||
@ -202,6 +210,15 @@ JSValue Context::interpret(JS2Runtime::ByteCodeModule *bcm, int offset, ScopeCha
|
||||
Activation *prev = mActivationStack.top();
|
||||
mActivationStack.pop();
|
||||
|
||||
// the following (delete's) are a bit iffy - depends on whether
|
||||
// a closure capturing the contents has come along...
|
||||
if (mThis.isObject())
|
||||
mScopeChain->popScope();
|
||||
delete[] mStack;
|
||||
delete[] mLocals;
|
||||
if (scopeChain == NULL)
|
||||
delete mScopeChain;
|
||||
|
||||
mCurModule = prev->mModule;
|
||||
mStack = prev->mStack;
|
||||
mStackTop = prev->mStackTop;
|
||||
@ -211,7 +228,7 @@ JSValue Context::interpret(JS2Runtime::ByteCodeModule *bcm, int offset, ScopeCha
|
||||
mArgumentBase = prev->mArgumentBase;
|
||||
mThis = prev->mThis;
|
||||
mScopeChain = prev->mScopeChain;
|
||||
|
||||
delete prev;
|
||||
return result;
|
||||
}
|
||||
|
||||
@ -229,7 +246,7 @@ JSValue *Context::buildArgumentBlock(JSFunction *target, uint32 &argCount)
|
||||
}
|
||||
|
||||
uint32 i;
|
||||
uint32 argBlockSize = maxExpectedArgCount + (target->hasRestParameter() ? 1 : 0);
|
||||
uint32 argBlockSize = max(argCount, maxExpectedArgCount) + (target->hasRestParameter() ? 1 : 0);
|
||||
// room for all required & optional arguments
|
||||
// plus the rest parameter if it exists.
|
||||
argBase = new JSValue[argBlockSize];
|
||||
@ -319,8 +336,19 @@ JSValue *Context::buildArgumentBlock(JSFunction *target, uint32 &argCount)
|
||||
restArgument.object->setProperty(this, *id, (NamespaceList *)(NULL), v);
|
||||
}
|
||||
}
|
||||
else
|
||||
reportError(Exception::referenceError, "Extra argument, no rest argument");
|
||||
else {
|
||||
if (target->isChecked())
|
||||
reportError(Exception::referenceError, "Extra argument, no rest argument");
|
||||
else {
|
||||
JSValue v = getValue(i + argStart);
|
||||
if (v.isObject() && (v.object->mType == NamedArgument_Type)) {
|
||||
NamedArgument *arg = static_cast<NamedArgument *>(v.object);
|
||||
argBase[i] = arg->mValue;
|
||||
}
|
||||
else
|
||||
argBase[i] = v;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -505,32 +533,34 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
|
||||
if (!targetValue->isFunction()) {
|
||||
if (targetValue->isType()) {
|
||||
/*
|
||||
// " Type() "
|
||||
// - it's a cast expression, we call the
|
||||
// default constructor, overriding the supplied 'this'.
|
||||
//
|
||||
|
||||
// XXX Note that this is different behaviour from JS1.5, where
|
||||
// (e.g.) Array(2) is an invocation of the constructor.
|
||||
|
||||
if ((argCount != 0) || ((callFlags & ThisFlags) != NoThis))
|
||||
reportError(Exception::referenceError, "Type cast can only take one argument");
|
||||
JSValue v = popValue();
|
||||
popValue(); // don't need the target anymore
|
||||
pushValue(mapValueToType(v, targetValue->type));
|
||||
break; // all done
|
||||
*/
|
||||
|
||||
// how to distinguish between a cast and an invocation of
|
||||
// the superclass constructor from a constructor????
|
||||
|
||||
// XXX help
|
||||
target = targetValue->type->getDefaultConstructor();
|
||||
if ((callFlags & SuperInvoke) == SuperInvoke) {
|
||||
// in this case, calling the constructor requires passing the 'this' value
|
||||
// through.
|
||||
target = targetValue->type->getDefaultConstructor();
|
||||
mThis = oldThis;
|
||||
}
|
||||
else {
|
||||
// " Type() "
|
||||
// - it's a cast expression, we call the
|
||||
// default constructor, overriding the supplied 'this'.
|
||||
//
|
||||
|
||||
// XXX Note that this is different behaviour from JS1.5, where
|
||||
// (e.g.) Array(2) is an invocation of the constructor.
|
||||
|
||||
// in this case, calling the constructor requires passing the 'this' value
|
||||
// through.
|
||||
mThis = oldThis;
|
||||
if ((argCount > 1) || ((callFlags & ThisFlags) != NoThis))
|
||||
reportError(Exception::referenceError, "Type cast can only take one argument");
|
||||
JSValue v;
|
||||
if (argCount > 0)
|
||||
v = popValue();
|
||||
popValue(); // don't need the target anymore
|
||||
pushValue(mapValueToType(v, targetValue->type));
|
||||
mThis = oldThis;
|
||||
break; // all done
|
||||
}
|
||||
}
|
||||
else
|
||||
reportError(Exception::referenceError, "Not a function");
|
||||
@ -564,6 +594,14 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
mScopeChain = target->getScopeChain();
|
||||
if (mThis.isObject())
|
||||
mScopeChain->addScope(mThis.object);
|
||||
|
||||
if (!target->isChecked()) {
|
||||
JSArrayInstance *args = (JSArrayInstance *)Array_Type->newInstance(this);
|
||||
for (uint32 i = 0; i < argCount; i++)
|
||||
args->setProperty(this, *numberToString(i), NULL, argBase[i]);
|
||||
mScopeChain->defineVariable(this, Arguments_StringAtom, NULL, Array_Type, JSValue(args));
|
||||
}
|
||||
|
||||
// mScopeChain->addScope(mActivationStack.top());
|
||||
mCurModule = target->getByteCode();
|
||||
pc = mCurModule->mCodeBase;
|
||||
@ -575,7 +613,17 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
mStackTop = 0;
|
||||
}
|
||||
else {
|
||||
// native functions may still need access to information
|
||||
// about the currently executing function.
|
||||
mActivationStack.push(new Activation(mLocals, mStack, mStackTop - (cleanUp + 1),
|
||||
mScopeChain,
|
||||
mArgumentBase, oldThis,
|
||||
pc, mCurModule));
|
||||
JSValue result = (target->getNativeCode())(this, mThis, argBase, argCount);
|
||||
Activation *prev = mActivationStack.top();
|
||||
delete prev;
|
||||
mActivationStack.pop();
|
||||
|
||||
mThis = oldThis;
|
||||
resizeStack(stackSize() - (cleanUp + 1));
|
||||
pushValue(result);
|
||||
@ -584,14 +632,18 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
}
|
||||
break;
|
||||
case ReturnVoidOp:
|
||||
{
|
||||
{
|
||||
if (mActivationStack.empty())
|
||||
return result;
|
||||
Activation *prev = mActivationStack.top();
|
||||
if (prev->mPC == NULL) // NULL is used to indicate that we want the loop to exit
|
||||
return result; // (even though there is more activation stack to go
|
||||
// - used to implement Xetters from XProperty ops. e.g.)
|
||||
if (prev->mPC == NULL) { // NULL is used to indicate that we want the loop to exit
|
||||
// (even though there is more activation stack to go
|
||||
return result; // - used to implement Xetters from XProperty ops. e.g.)
|
||||
|
||||
}
|
||||
mActivationStack.pop();
|
||||
delete[] mLocals;
|
||||
delete[] mStack;
|
||||
|
||||
mCurModule = prev->mModule;
|
||||
pc = prev->mPC;
|
||||
@ -602,20 +654,22 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
mLocals = prev->mLocals;
|
||||
mArgumentBase = prev->mArgumentBase;
|
||||
mThis = prev->mThis;
|
||||
mScopeChain = prev->mScopeChain;
|
||||
mScopeChain = prev->mScopeChain;
|
||||
delete prev;
|
||||
}
|
||||
break;
|
||||
case ReturnOp:
|
||||
{
|
||||
JSValue result = popValue();
|
||||
|
||||
JSValue result = popValue();
|
||||
if (mActivationStack.empty())
|
||||
return result;
|
||||
return result;
|
||||
Activation *prev = mActivationStack.top();
|
||||
if (prev->mPC == NULL)
|
||||
if (prev->mPC == NULL) {
|
||||
return result;
|
||||
|
||||
}
|
||||
mActivationStack.pop();
|
||||
delete[] mLocals;
|
||||
delete[] mStack;
|
||||
|
||||
mCurModule = prev->mModule;
|
||||
pc = prev->mPC;
|
||||
@ -628,6 +682,7 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
mThis = prev->mThis;
|
||||
mScopeChain = prev->mScopeChain;
|
||||
pushValue(result);
|
||||
delete prev;
|
||||
}
|
||||
break;
|
||||
case LoadTypeOp:
|
||||
@ -653,9 +708,8 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
break;
|
||||
case LoadConstantNumberOp:
|
||||
{
|
||||
uint32 index = *((uint32 *)pc);
|
||||
pc += sizeof(uint32);
|
||||
pushValue(JSValue(mCurModule->getNumber(index)));
|
||||
pushValue(JSValue(mCurModule->getNumber(pc)));
|
||||
pc += sizeof(float64);
|
||||
}
|
||||
break;
|
||||
case LoadConstantUndefinedOp:
|
||||
@ -673,6 +727,17 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
case LoadConstantZeroOp:
|
||||
pushValue(kPositiveZero);
|
||||
break;
|
||||
case DeleteNameOp:
|
||||
{
|
||||
uint32 index = *((uint32 *)pc);
|
||||
pc += sizeof(uint32);
|
||||
const String &name = *mCurModule->getString(index);
|
||||
PropertyIterator it;
|
||||
if (mScopeChain->hasOwnProperty(name, CURRENT_ATTR, Read, &it))
|
||||
mScopeChain->deleteProperty(name, CURRENT_ATTR);
|
||||
pushValue(kTrueValue);
|
||||
}
|
||||
break;
|
||||
case DeleteOp:
|
||||
{
|
||||
JSValue base = popValue();
|
||||
@ -694,24 +759,24 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
{
|
||||
JSValue v = popValue();
|
||||
if (v.isUndefined())
|
||||
pushValue(JSValue(new String(widenCString("undefined"))));
|
||||
pushValue(JSValue(&Undefined_StringAtom));
|
||||
else
|
||||
if (v.isNull())
|
||||
pushValue(JSValue(new String(widenCString("object"))));
|
||||
pushValue(JSValue(&Object_StringAtom));
|
||||
else
|
||||
if (v.isBool())
|
||||
pushValue(JSValue(new String(widenCString("boolean"))));
|
||||
pushValue(JSValue(&Boolean_StringAtom));
|
||||
else
|
||||
if (v.isNumber())
|
||||
pushValue(JSValue(new String(widenCString("number"))));
|
||||
pushValue(JSValue(&Number_StringAtom));
|
||||
else
|
||||
if (v.isString())
|
||||
pushValue(JSValue(new String(widenCString("string"))));
|
||||
pushValue(JSValue(&String_StringAtom));
|
||||
else
|
||||
if (v.isFunction())
|
||||
pushValue(JSValue(new String(widenCString("function"))));
|
||||
pushValue(JSValue(&Function_StringAtom));
|
||||
else
|
||||
pushValue(JSValue(new String(widenCString("object"))));
|
||||
pushValue(JSValue(&Object_StringAtom));
|
||||
}
|
||||
break;
|
||||
case AsOp:
|
||||
@ -770,7 +835,7 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
JSFunction *obj = t.function;
|
||||
PropertyIterator i;
|
||||
JSFunction *target = NULL;
|
||||
if (obj->hasProperty(widenCString("hasInstance"), CURRENT_ATTR, Read, &i)) {
|
||||
if (obj->hasProperty(HasInstance_StringAtom, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue hi = obj->getPropertyValue(i);
|
||||
if (hi.isFunction())
|
||||
target = hi.function;
|
||||
@ -822,22 +887,36 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
|
||||
// Use the type of the base to dispatch on...
|
||||
JSObject *obj = NULL;
|
||||
if (!baseValue->isObject() && !baseValue->isType())
|
||||
obj = baseValue->toObject(this).object;
|
||||
else
|
||||
if (baseValue->isType())
|
||||
obj = baseValue->type;
|
||||
else
|
||||
if (baseValue->isFunction())
|
||||
obj = baseValue->function;
|
||||
else
|
||||
if (baseValue->isObject())
|
||||
obj = baseValue->object;
|
||||
else
|
||||
obj = baseValue->toObject(this).object;
|
||||
|
||||
JSFunction *target = obj->getType()->getUnaryOperator(Index);
|
||||
|
||||
if (target) {
|
||||
JSValue result;
|
||||
if (target->isNative())
|
||||
result = target->getNativeCode()(this, kNullValue, baseValue, dimCount + 1);
|
||||
if (target->isNative()) {
|
||||
JSValue *argBase = new JSValue[dimCount + 1];
|
||||
for (int i = 0; i < (dimCount + 1); i++)
|
||||
argBase[i] = baseValue[i];
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
result = target->getNativeCode()(this, argBase[0], argBase, dimCount + 1);
|
||||
delete[] argBase;
|
||||
}
|
||||
else {
|
||||
uint32 argCount = dimCount + 1;
|
||||
JSValue *argBase = buildArgumentBlock(target, argCount);
|
||||
result = interpret(target->getByteCode(), 0, target->getScopeChain(), kNullValue, argBase, argCount);
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
result = interpret(target->getByteCode(), 0, target->getScopeChain(), argBase[0], argBase, argCount);
|
||||
delete[] argBase;
|
||||
}
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
pushValue(result);
|
||||
}
|
||||
else { // XXX or should this be implemented in Object_Type as operator "[]" ?
|
||||
@ -871,14 +950,21 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
insertValue(v, mStackTop - dimCount);
|
||||
|
||||
JSValue result;
|
||||
if (target->isNative())
|
||||
if (target->isNative()) {
|
||||
JSValue *argBase = new JSValue[dimCount + 2];
|
||||
for (int i = 0; i < (dimCount + 2); i++)
|
||||
argBase[i] = baseValue[i];
|
||||
resizeStack(stackSize() - (dimCount + 2));
|
||||
result = target->getNativeCode()(this, *baseValue, baseValue, (dimCount + 2));
|
||||
delete[] argBase;
|
||||
}
|
||||
else {
|
||||
uint32 argCount = dimCount + 2;
|
||||
JSValue *argBase = buildArgumentBlock(target, argCount);
|
||||
resizeStack(stackSize() - (dimCount + 2));
|
||||
result = interpret(target->getByteCode(), 0, target->getScopeChain(), *baseValue, argBase, argCount);
|
||||
delete[] argBase;
|
||||
}
|
||||
resizeStack(stackSize() - (dimCount + 2));
|
||||
pushValue(result);
|
||||
}
|
||||
else { // XXX or should this be implemented in Object_Type as operator "[]=" ?
|
||||
@ -911,14 +997,21 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
|
||||
if (target) {
|
||||
JSValue result;
|
||||
if (target->isNative())
|
||||
result = target->getNativeCode()(this, kNullValue, baseValue, dimCount + 1);
|
||||
if (target->isNative()) {
|
||||
JSValue *argBase = new JSValue[dimCount + 1];
|
||||
for (int i = 0; i < (dimCount + 1); i++)
|
||||
argBase[i] = baseValue[i];
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
result = target->getNativeCode()(this, argBase[0], argBase, dimCount + 1);
|
||||
delete[] argBase;
|
||||
}
|
||||
else {
|
||||
uint32 argCount = dimCount + 1;
|
||||
JSValue *argBase = buildArgumentBlock(target, argCount);
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
result = interpret(target->getByteCode(), 0, target->getScopeChain(), kNullValue, argBase, argCount);
|
||||
delete[] argBase;
|
||||
}
|
||||
resizeStack(stackSize() - (dimCount + 1));
|
||||
pushValue(result);
|
||||
}
|
||||
else { // XXX or should this be implemented in Object_Type as operator "delete[]" ?
|
||||
@ -1110,7 +1203,7 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
target = typeValue->function;
|
||||
newThis = Object_Type->newInstance(this);
|
||||
PropertyIterator i;
|
||||
if (target->hasProperty(widenCString("prototype"), CURRENT_ATTR, Read, &i)) {
|
||||
if (target->hasProperty(Prototype_StringAtom, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = target->getPropertyValue(i);
|
||||
newThis.object->mPrototype = v.toObject(this).object;
|
||||
}
|
||||
@ -1133,17 +1226,33 @@ JSValue Context::interpret(uint8 *pc, uint8 *endPC)
|
||||
ASSERT(target);
|
||||
|
||||
JSValue result;
|
||||
if (target->isNative())
|
||||
result = target->getNativeCode()(this, newThis, argBase, argCount);
|
||||
if (target->isNative()) {
|
||||
JSValue *tArgBase = new JSValue[argCount];
|
||||
for (int i = 0; i < argCount; i++)
|
||||
tArgBase[i] = argBase[i];
|
||||
resizeStack(stackSize() - cleanUp);
|
||||
result = target->getNativeCode()(this, newThis, tArgBase, argCount);
|
||||
}
|
||||
else {
|
||||
argBase = buildArgumentBlock(target, argCount);
|
||||
resizeStack(stackSize() - cleanUp);
|
||||
if (!target->isChecked()) {
|
||||
JSArrayInstance *args = (JSArrayInstance *)Array_Type->newInstance(this);
|
||||
for (uint32 i = 0; i < argCount; i++)
|
||||
args->setProperty(this, *numberToString(i), NULL, argBase[i]);
|
||||
mScopeChain->defineVariable(this, Arguments_StringAtom, NULL, Array_Type, JSValue(args));
|
||||
}
|
||||
result = interpret(target->getByteCode(), 0, target->getScopeChain(), newThis, argBase, argCount);
|
||||
}
|
||||
resizeStack(stackSize() - cleanUp);
|
||||
|
||||
if (!isPrototypeFunctionCall)
|
||||
// If it's a prototype function, we don't care what it returns,
|
||||
// we have the 'this' already
|
||||
if (isPrototypeFunctionCall) {
|
||||
// If it's a prototype function, the return value is only
|
||||
// interesting if it's not a primitive, in which case it
|
||||
// overrides the newly constructed object. Weird, huh.
|
||||
if (!result.isPrimitive())
|
||||
newThis = result;
|
||||
}
|
||||
else
|
||||
// otherwise, constructor has potentially made the 'this', so retain it
|
||||
newThis = result;
|
||||
|
||||
@ -1438,10 +1547,6 @@ static JSValue objectPlus(Context *cx, const JSValue& /*thisValue*/, JSValue *ar
|
||||
JSValue &r1 = argv[0];
|
||||
JSValue &r2 = argv[1];
|
||||
if (r1.isNumber() && r2.isNumber()) {
|
||||
// this is the ECMA3 implementation. Suppose somebody has
|
||||
// added (can they?) an operator "+" for Numbers - we should
|
||||
// oughta dispatch to that function here. (And all other
|
||||
// cases below, arrgh - how deep does this go? my brain hurts)
|
||||
return JSValue(r1.toNumber(cx).f64 + r2.toNumber(cx).f64);
|
||||
}
|
||||
|
||||
@ -1507,11 +1612,11 @@ static JSValue integerMultiply(Context * /*cx*/, const JSValue& /*thisValue*/, J
|
||||
return JSValue(r1.f64 * r2.f64);
|
||||
}
|
||||
|
||||
static JSValue objectMultiply(Context * /*cx*/, const JSValue& /*thisValue*/, JSValue *argv, uint32 /*argc*/)
|
||||
static JSValue objectMultiply(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 /*argc*/)
|
||||
{
|
||||
JSValue &r1 = argv[0];
|
||||
JSValue &r2 = argv[1];
|
||||
return JSValue(r1.f64 * r2.f64);
|
||||
return JSValue(r1.toNumber(cx).f64 * r2.toNumber(cx).f64);
|
||||
}
|
||||
|
||||
|
||||
@ -1551,7 +1656,17 @@ static JSValue objectRemainder(Context *cx, const JSValue& /*thisValue*/, JSValu
|
||||
{
|
||||
JSValue &r1 = argv[0];
|
||||
JSValue &r2 = argv[1];
|
||||
return JSValue(fd::fmod(r1.toNumber(cx).f64, r2.toNumber(cx).f64));
|
||||
|
||||
float64 f1 = r1.toNumber(cx).f64;
|
||||
float64 f2 = r2.toNumber(cx).f64;
|
||||
|
||||
#ifdef XP_PC
|
||||
/* Workaround MS fmod bug where 42 % (1/0) => NaN, not 42. */
|
||||
if (JSDOUBLE_IS_FINITE(f1) && JSDOUBLE_IS_INFINITE(f2))
|
||||
return JSValue(f1);
|
||||
#endif
|
||||
|
||||
return JSValue(fd::fmod(f1, f2));
|
||||
}
|
||||
|
||||
|
||||
@ -1689,7 +1804,7 @@ static JSValue objectLessEqual(Context *cx, const JSValue& /*thisValue*/, JSValu
|
||||
JSValue &r1 = argv[0];
|
||||
JSValue &r2 = argv[1];
|
||||
JSValue result = objectCompare(cx, r2, r1);
|
||||
if (result.isTrue() || result.isUndefined())
|
||||
if (result.isUndefined() || result.isTrue())
|
||||
return kFalseValue;
|
||||
else
|
||||
return kTrueValue;
|
||||
@ -1876,7 +1991,14 @@ JSValue JSValue::valueToObject(Context *cx, const JSValue& value)
|
||||
float64 stringToNumber(const String *string)
|
||||
{
|
||||
const char16 *numEnd;
|
||||
return stringToDouble(string->begin(), string->end(), numEnd);
|
||||
const char16 *sBegin = string->begin();
|
||||
if (sBegin)
|
||||
if ((sBegin[0] == '0') && ((sBegin[1] & ~0x20) == 'X'))
|
||||
return stringToInteger(sBegin, string->end(), numEnd, 16);
|
||||
else
|
||||
return stringToDouble(sBegin, string->end(), numEnd);
|
||||
else
|
||||
return 0.0;
|
||||
}
|
||||
|
||||
JSValue JSValue::valueToNumber(Context *cx, const JSValue& value)
|
||||
@ -1893,6 +2015,8 @@ JSValue JSValue::valueToNumber(Context *cx, const JSValue& value)
|
||||
return JSValue((value.boolean) ? 1.0 : 0.0);
|
||||
case undefined_tag:
|
||||
return kNaNValue;
|
||||
case null_tag:
|
||||
return kPositiveZero;
|
||||
default:
|
||||
NOT_REACHED("Bad tag");
|
||||
return kUndefinedValue;
|
||||
@ -1922,18 +2046,16 @@ JSValue JSValue::valueToString(Context *cx, const JSValue& value)
|
||||
case string_tag:
|
||||
return value;
|
||||
case boolean_tag:
|
||||
strp = (value.boolean)
|
||||
? new JavaScript::String(widenCString("true"))
|
||||
: new JavaScript::String(widenCString("false"));
|
||||
strp = (value.boolean) ? &cx->True_StringAtom : &cx->False_StringAtom;
|
||||
break;
|
||||
case type_tag:
|
||||
strp = value.type->mClassName;
|
||||
break;
|
||||
case undefined_tag:
|
||||
strp = new JavaScript::String(widenCString("undefined"));
|
||||
strp = &cx->Undefined_StringAtom;
|
||||
break;
|
||||
case null_tag:
|
||||
strp = new JavaScript::String(widenCString("null"));
|
||||
strp = &cx->Null_StringAtom;
|
||||
break;
|
||||
default:
|
||||
NOT_REACHED("Bad tag");
|
||||
@ -1941,26 +2063,20 @@ JSValue JSValue::valueToString(Context *cx, const JSValue& value)
|
||||
if (obj) {
|
||||
JSFunction *target = NULL;
|
||||
PropertyIterator i;
|
||||
if (obj->hasProperty(widenCString("toString"), CURRENT_ATTR, Read, &i)) {
|
||||
if (obj->hasProperty(cx->ToString_StringAtom, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = obj->getPropertyValue(i);
|
||||
if (v.isFunction())
|
||||
target = v.function;
|
||||
}
|
||||
if (target == NULL) {
|
||||
if (obj->hasProperty(widenCString("valueOf"), CURRENT_ATTR, Read, &i)) {
|
||||
if (obj->hasProperty(cx->ValueOf_StringAtom, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = obj->getPropertyValue(i);
|
||||
if (v.isFunction())
|
||||
target = v.function;
|
||||
}
|
||||
}
|
||||
if (target) {
|
||||
if (!target->isNative()) {
|
||||
// here we need to get the interpreter to do the job
|
||||
ASSERT(false);
|
||||
}
|
||||
else
|
||||
return (target->getNativeCode())(cx, value, NULL, 0);
|
||||
}
|
||||
if (target)
|
||||
return cx->invokeFunction(target, value, NULL, 0);
|
||||
throw new Exception(Exception::runtimeError, "toString"); // XXX
|
||||
return kUndefinedValue; // keep compilers happy
|
||||
}
|
||||
@ -1969,7 +2085,7 @@ JSValue JSValue::valueToString(Context *cx, const JSValue& value)
|
||||
|
||||
}
|
||||
|
||||
JSValue JSValue::toPrimitive(Context *, Hint) const
|
||||
JSValue JSValue::toPrimitive(Context *cx, Hint hint) const
|
||||
{
|
||||
JSObject *obj;
|
||||
switch (tag) {
|
||||
@ -1990,74 +2106,47 @@ JSValue JSValue::toPrimitive(Context *, Hint) const
|
||||
NOT_REACHED("Bad tag");
|
||||
return kUndefinedValue;
|
||||
}
|
||||
/*
|
||||
JSFunction *target = NULL;
|
||||
JSValue result;
|
||||
JSValues argv(1);
|
||||
argv[0] = *this;
|
||||
|
||||
// The following is [[DefaultValue]]
|
||||
//
|
||||
if ((hint == NumberHint) || (hint == NoHint)) {
|
||||
const JSValue &valueOf = obj->getProperty(widenCString("valueOf"));
|
||||
if (valueOf.isFunction()) {
|
||||
target = valueOf.function;
|
||||
if (target->isNative()) {
|
||||
result = static_cast<JSNativeFunction*>(target)->mCode(cx, argv);
|
||||
}
|
||||
else {
|
||||
Context new_cx(cx);
|
||||
result = new_cx.interpret(target->getICode(), argv);
|
||||
}
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
const JSValue &toString = obj->getProperty(widenCString("toString"));
|
||||
if (toString.isFunction()) {
|
||||
target = toString.function;
|
||||
if (target->isNative()) {
|
||||
result = static_cast<JSNativeFunction*>(target)->mCode(cx, argv);
|
||||
}
|
||||
else {
|
||||
Context new_cx(cx);
|
||||
result = new_cx.interpret(target->getICode(), argv);
|
||||
}
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
}
|
||||
else {
|
||||
const JSValue &toString = obj->getProperty(widenCString("toString"));
|
||||
if (toString.isFunction()) {
|
||||
target = toString.function;
|
||||
if (target->isNative()) {
|
||||
result = static_cast<JSNativeFunction*>(target)->mCode(cx, argv);
|
||||
}
|
||||
else {
|
||||
Context new_cx(cx);
|
||||
result = new_cx.interpret(target->getICode(), argv);
|
||||
}
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
const JSValue &valueOf = obj->getProperty(widenCString("valueOf"));
|
||||
if (valueOf.isFunction()) {
|
||||
target = valueOf.function;
|
||||
if (target->isNative()) {
|
||||
result = static_cast<JSNativeFunction*>(target)->mCode(cx, argv);
|
||||
}
|
||||
else {
|
||||
Context new_cx(cx);
|
||||
result = new_cx.interpret(target->getICode(), argv);
|
||||
}
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
}
|
||||
throw Exception(Exception::runtimeError, "toPrimitive"); // XXX
|
||||
*/
|
||||
ASSERT(obj);
|
||||
JSFunction *target = NULL;
|
||||
JSValue result;
|
||||
PropertyIterator i;
|
||||
|
||||
StringAtom *first = &cx->ValueOf_StringAtom;
|
||||
StringAtom *second = &cx->ToString_StringAtom;
|
||||
|
||||
if (hint == StringHint) {
|
||||
first = &cx->ToString_StringAtom;
|
||||
second = &cx->ValueOf_StringAtom;
|
||||
}
|
||||
|
||||
|
||||
if (obj->hasProperty(*first, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = obj->getPropertyValue(i);
|
||||
if (v.isFunction()) {
|
||||
target = v.function;
|
||||
if (target) {
|
||||
result = cx->invokeFunction(target, *this, NULL, 0);
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (obj->hasProperty(*second, CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = obj->getPropertyValue(i);
|
||||
if (v.isFunction()) {
|
||||
target = v.function;
|
||||
if (target) {
|
||||
result = cx->invokeFunction(target, *this, NULL, 0);
|
||||
if (result.isPrimitive())
|
||||
return result;
|
||||
}
|
||||
}
|
||||
}
|
||||
throw new Exception(Exception::runtimeError, "toPrimitive"); // XXX
|
||||
return kUndefinedValue;
|
||||
|
||||
}
|
||||
|
||||
int JSValue::operator==(const JSValue& value) const
|
||||
@ -2197,16 +2286,18 @@ JSValue JSValue::valueToBoolean(Context *cx, const JSValue& value)
|
||||
JSObject *obj = NULL;
|
||||
switch (value.tag) {
|
||||
case f64_tag:
|
||||
return JSValue(!(value.f64 == 0.0) || JSDOUBLE_IS_NaN(value.f64));
|
||||
if (JSDOUBLE_IS_NaN(value.f64))
|
||||
return kFalseValue;
|
||||
if (value.f64 == 0.0)
|
||||
return kFalseValue;
|
||||
return kTrueValue;
|
||||
case string_tag:
|
||||
return JSValue(value.string->length() != 0);
|
||||
case boolean_tag:
|
||||
return value;
|
||||
case object_tag:
|
||||
obj = value.object;
|
||||
break;
|
||||
case function_tag:
|
||||
obj = value.function;
|
||||
return kTrueValue;
|
||||
break;
|
||||
case undefined_tag:
|
||||
return kFalseValue;
|
||||
@ -2214,25 +2305,6 @@ JSValue JSValue::valueToBoolean(Context *cx, const JSValue& value)
|
||||
NOT_REACHED("Bad tag");
|
||||
return kUndefinedValue;
|
||||
}
|
||||
ASSERT(obj);
|
||||
JSFunction *target = NULL;
|
||||
PropertyIterator i;
|
||||
if (obj->hasProperty(widenCString("toBoolean"), CURRENT_ATTR, Read, &i)) {
|
||||
JSValue v = obj->getPropertyValue(i);
|
||||
if (v.isFunction())
|
||||
target = v.function;
|
||||
}
|
||||
if (target) {
|
||||
if (!target->isNative()) {
|
||||
// here we need to get the interpreter to do the job
|
||||
ASSERT(false);
|
||||
}
|
||||
else {
|
||||
JSValue args = value;
|
||||
return (target->getNativeCode())(cx, value, &args, 1);
|
||||
}
|
||||
}
|
||||
throw new Exception(Exception::runtimeError, "toBoolean"); // XXX
|
||||
}
|
||||
|
||||
|
||||
|
@ -180,6 +180,13 @@ JSValue JSObject::getPropertyValue(PropertyIterator &i)
|
||||
return *prop->mData.vp;
|
||||
}
|
||||
|
||||
Property *JSObject::insertNewProperty(const String &name, NamespaceList *names, PropertyAttribute attrFlags, JSType *type, const JSValue &v)
|
||||
{
|
||||
Property *prop = new Property(new JSValue(v), type, attrFlags);
|
||||
const PropertyMap::value_type e(name, new NamespacedProperty(prop, names));
|
||||
mProperties.insert(e);
|
||||
return prop;
|
||||
}
|
||||
|
||||
void JSObject::defineGetterMethod(Context * /*cx*/, const String &name, AttributeStmtNode *attr, JSFunction *f)
|
||||
{
|
||||
@ -264,7 +271,7 @@ Property *JSObject::defineVariable(Context *cx, const String &name, NamespaceLis
|
||||
}
|
||||
|
||||
// add a property (with a value)
|
||||
Property *JSObject::defineVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type, JSValue v)
|
||||
Property *JSObject::defineVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type, const JSValue v)
|
||||
{
|
||||
NamespaceList *names = (attr) ? attr->attributeValue->mNamespaceList : NULL;
|
||||
PropertyAttribute attrFlags = (attr) ? attr->attributeValue->mTrueFlags : 0;
|
||||
@ -278,6 +285,11 @@ Property *JSObject::defineVariable(Context *cx, const String &name, AttributeStm
|
||||
else
|
||||
cx->reportError(Exception::typeError, "Duplicate definition '{0}'", name);
|
||||
}
|
||||
else {
|
||||
// override the existing value
|
||||
PROPERTY_VALUEPOINTER(it) = new JSValue(v);
|
||||
return PROPERTY(it);
|
||||
}
|
||||
}
|
||||
|
||||
Property *prop = new Property(new JSValue(v), type, attrFlags);
|
||||
@ -285,7 +297,7 @@ Property *JSObject::defineVariable(Context *cx, const String &name, AttributeStm
|
||||
mProperties.insert(e);
|
||||
return prop;
|
||||
}
|
||||
Property *JSObject::defineVariable(Context *cx, const String &name, NamespaceList *names, JSType *type, JSValue v)
|
||||
Property *JSObject::defineVariable(Context *cx, const String &name, NamespaceList *names, JSType *type, const JSValue v)
|
||||
{
|
||||
PropertyIterator it;
|
||||
if (hasOwnProperty(name, names, Read, &it)) {
|
||||
@ -483,7 +495,7 @@ void JSInstance::setProperty(Context *cx, const String &name, NamespaceList *nam
|
||||
|
||||
void JSArrayInstance::getProperty(Context *cx, const String &name, NamespaceList *names)
|
||||
{
|
||||
if (name.compare(widenCString("length")) == 0)
|
||||
if (name.compare(cx->Length_StringAtom) == 0)
|
||||
cx->pushValue(JSValue((float64)mLength));
|
||||
else
|
||||
JSInstance::getProperty(cx, name, names);
|
||||
@ -491,7 +503,7 @@ void JSArrayInstance::getProperty(Context *cx, const String &name, NamespaceList
|
||||
|
||||
void JSArrayInstance::setProperty(Context *cx, const String &name, NamespaceList *names, const JSValue &v)
|
||||
{
|
||||
if (name.compare(widenCString("length")) == 0) {
|
||||
if (name.compare(cx->Length_StringAtom) == 0) {
|
||||
uint32 newLength = (uint32)(v.toUInt32(cx).f64);
|
||||
if (newLength != v.toNumber(cx).f64)
|
||||
cx->reportError(Exception::rangeError, "out of range value for length");
|
||||
@ -506,10 +518,14 @@ void JSArrayInstance::setProperty(Context *cx, const String &name, NamespaceList
|
||||
mLength = newLength;
|
||||
}
|
||||
else {
|
||||
if (findNamespacedProperty(name, names) == mProperties.end())
|
||||
defineVariable(cx, name, names, Object_Type, v);
|
||||
else
|
||||
JSInstance::setProperty(cx, name, names, v);
|
||||
PropertyIterator it = findNamespacedProperty(name, names);
|
||||
if (it == mProperties.end())
|
||||
insertNewProperty(name, names, 0, Object_Type, v);
|
||||
else {
|
||||
Property *prop = PROPERTY(it);
|
||||
ASSERT(prop->mFlag == ValuePointer);
|
||||
*prop->mData.vp = v;
|
||||
}
|
||||
JSValue v = JSValue(&name);
|
||||
JSValue v_int = v.toUInt32(cx);
|
||||
if ((v_int.f64 != two32minus1) && (v_int.toString(cx).string->compare(name) == 0)) {
|
||||
@ -521,7 +537,7 @@ void JSArrayInstance::setProperty(Context *cx, const String &name, NamespaceList
|
||||
|
||||
void JSStringInstance::getProperty(Context *cx, const String &name, NamespaceList *names)
|
||||
{
|
||||
if (name.compare(widenCString("length")) == 0) {
|
||||
if (name.compare(cx->Length_StringAtom) == 0) {
|
||||
cx->pushValue(JSValue((float64)mLength));
|
||||
}
|
||||
else
|
||||
@ -622,6 +638,7 @@ void ScopeChain::setNameValue(Context *cx, const String& name, AttributeStmtNode
|
||||
if ((*s)->hasProperty(name, names, Write, &i)) {
|
||||
if (PROPERTY_KIND(i) == ValuePointer) {
|
||||
*PROPERTY_VALUEPOINTER(i) = v;
|
||||
break;
|
||||
}
|
||||
else
|
||||
ASSERT(false); // what else needs to be implemented ?
|
||||
@ -832,7 +849,7 @@ void Context::buildRuntime(StmtNode *p)
|
||||
}
|
||||
|
||||
// Generate bytecode for the linked list of statements in p
|
||||
JS2Runtime::ByteCodeModule *Context::genCode(StmtNode *p, String /*sourceName*/)
|
||||
JS2Runtime::ByteCodeModule *Context::genCode(StmtNode *p, const String &/*sourceName*/)
|
||||
{
|
||||
mScopeChain->addScope(getGlobalObject());
|
||||
JS2Runtime::ByteCodeGen bcg(this, mScopeChain);
|
||||
@ -850,23 +867,23 @@ JS2Runtime::ByteCodeModule *Context::genCode(StmtNode *p, String /*sourceName*/)
|
||||
*/
|
||||
bool ScopeChain::isPossibleUncheckedFunction(FunctionDefinition *f)
|
||||
{
|
||||
bool result = false;
|
||||
bool result = false;
|
||||
if ((f->resultType == NULL)
|
||||
&& (f->restParameter == NULL)
|
||||
&& (f->optParameters == NULL)
|
||||
&& (f->prefix == FunctionName::normal)
|
||||
&& (topClass() == NULL)) {
|
||||
result = true;
|
||||
VariableBinding *b = f->parameters;
|
||||
while (b) {
|
||||
if (b->type != NULL) {
|
||||
result = false;
|
||||
break;
|
||||
}
|
||||
b = b->next;
|
||||
}
|
||||
&& (f->restParameter == NULL)
|
||||
&& (f->optParameters == NULL)
|
||||
&& (f->prefix == FunctionName::normal)
|
||||
&& (topClass() == NULL)) {
|
||||
result = true;
|
||||
VariableBinding *b = f->parameters;
|
||||
while (b) {
|
||||
if (b->type != NULL) {
|
||||
result = false;
|
||||
break;
|
||||
}
|
||||
b = b->next;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
// The first pass over the tree - it just installs the names of each declaration
|
||||
@ -935,6 +952,7 @@ void ScopeChain::collectNames(StmtNode *p)
|
||||
{
|
||||
ForStmtNode *f = checked_cast<ForStmtNode *>(p);
|
||||
if (f->initializer) collectNames(f->initializer);
|
||||
collectNames(f->stmt);
|
||||
}
|
||||
break;
|
||||
case StmtNode::Const:
|
||||
@ -972,6 +990,8 @@ void ScopeChain::collectNames(StmtNode *p)
|
||||
bool isOperator = (f->attributeValue->mTrueFlags & Property::Operator) == Property::Operator;
|
||||
bool isPrototype = (f->attributeValue->mTrueFlags & Property::Prototype) == Property::Prototype;
|
||||
|
||||
JSFunction *fnc = new JSFunction(NULL, this);
|
||||
|
||||
/* Determine whether a function is unchecked, which is the case if -
|
||||
XXX strict mode is disabled at the point of the function definition;
|
||||
the function is not a class or interface member;
|
||||
@ -982,11 +1002,11 @@ void ScopeChain::collectNames(StmtNode *p)
|
||||
*/
|
||||
if (!isPrototype
|
||||
&& (!isOperator)
|
||||
&& isPossibleUncheckedFunction(&f->function)) {
|
||||
&& isPossibleUncheckedFunction(&f->function)) {
|
||||
isPrototype = true;
|
||||
fnc->setIsUnchecked();
|
||||
}
|
||||
|
||||
JSFunction *fnc = new JSFunction(NULL, this);
|
||||
fnc->setIsPrototype(isPrototype);
|
||||
fnc->setIsConstructor(isConstructor);
|
||||
fnc->setFunctionName(&f->function);
|
||||
@ -1081,7 +1101,11 @@ void ScopeChain::collectNames(StmtNode *p)
|
||||
defineSetterMethod(m_cx, name, f, fnc);
|
||||
break;
|
||||
case FunctionName::normal:
|
||||
f->attributeValue->mTrueFlags |= Property::Const;
|
||||
// make a function into a const declaration, but only if any types
|
||||
// have been specified - otherwise it's a 1.5 atyle definition and
|
||||
// duplicates are allowed
|
||||
if (!isPossibleUncheckedFunction(&f->function))
|
||||
f->attributeValue->mTrueFlags |= Property::Const;
|
||||
if (isStatic)
|
||||
defineStaticMethod(m_cx, name, f, fnc);
|
||||
else
|
||||
@ -1313,7 +1337,7 @@ Reference *JSType::genReference(bool hasBase, const String& name, NamespaceList
|
||||
return NULL;
|
||||
}
|
||||
|
||||
JSType::JSType(Context *cx, const StringAtom *name, JSType *super)
|
||||
JSType::JSType(Context *cx, const StringAtom *name, JSType *super, JSObject *protoObj)
|
||||
: JSObject(Type_Type),
|
||||
mSuperType(super),
|
||||
mVariableCount(0),
|
||||
@ -1333,10 +1357,15 @@ JSType::JSType(Context *cx, const StringAtom *name, JSType *super)
|
||||
mUnaryOperators[i] = NULL;
|
||||
|
||||
// every class gets a prototype object
|
||||
mPrototypeObject = new JSObject();
|
||||
if (protoObj)
|
||||
mPrototypeObject = protoObj;
|
||||
else
|
||||
mPrototypeObject = new JSObject();
|
||||
// and that object is prototype-linked to the super-type's prototype object
|
||||
if (mSuperType)
|
||||
mPrototypeObject->mPrototype = mSuperType->mPrototypeObject;
|
||||
|
||||
// defineVariable(cx, cx->Prototype_StringAtom, NULL, Object_Type, JSValue(mPrototypeObject));
|
||||
}
|
||||
|
||||
JSType::JSType(JSType *xClass) // used for constructing the static component type
|
||||
@ -1355,6 +1384,8 @@ JSType::JSType(JSType *xClass) // used for constructing the static component
|
||||
mUnaryOperators[i] = NULL;
|
||||
}
|
||||
|
||||
// Establish the super class - connects the prototype's prototype
|
||||
// and accounts for the super class's instance fields & methods
|
||||
void JSType::setSuperType(JSType *super)
|
||||
{
|
||||
mSuperType = super;
|
||||
@ -1424,9 +1455,10 @@ Reference *Activation::genReference(bool /* hasBase */, const String& name, Name
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
/*
|
||||
Process the statements in the function body, handling parameters and local
|
||||
variables to collect names & types.
|
||||
*/
|
||||
void Context::buildRuntimeForFunction(FunctionDefinition &f, JSFunction *fnc)
|
||||
{
|
||||
fnc->mParameterBarrel = new ParameterBarrel();
|
||||
@ -1448,6 +1480,10 @@ void Context::buildRuntimeForFunction(FunctionDefinition &f, JSFunction *fnc)
|
||||
mScopeChain->popScope();
|
||||
}
|
||||
|
||||
/*
|
||||
The incoming AttributeStmtNode has a list of attributes - evaluate those and return
|
||||
the resultant Attribute value. If there are no attributes, return the default value.
|
||||
*/
|
||||
void Context::setAttributeValue(AttributeStmtNode *s, PropertyAttribute defaultValue)
|
||||
{
|
||||
Attribute *attributeValue = NULL;
|
||||
@ -1508,6 +1544,7 @@ void Context::buildRuntimeForStmt(StmtNode *p)
|
||||
{
|
||||
ForStmtNode *f = checked_cast<ForStmtNode *>(p);
|
||||
if (f->initializer) buildRuntimeForStmt(f->initializer);
|
||||
buildRuntimeForStmt(f->stmt);
|
||||
}
|
||||
break;
|
||||
case StmtNode::Var:
|
||||
@ -1696,13 +1733,67 @@ static JSValue Object_done(Context *, const JSValue& /*thisValue*/, JSValue * /*
|
||||
|
||||
|
||||
|
||||
static JSValue Function_Constructor(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
static JSValue Function_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
JSValue v = thisValue;
|
||||
if (v.isNull())
|
||||
v = Function_Type->newInstance(cx);
|
||||
// XXX use the arguments to compile a string into a function
|
||||
ASSERT(v.isObject());
|
||||
|
||||
String s;
|
||||
if (argc == 0)
|
||||
s = widenCString("() { }");
|
||||
else {
|
||||
if (argc == 1)
|
||||
s = widenCString("() {") + *argv[0].toString(cx).string + "}";
|
||||
else {
|
||||
s = widenCString("("); // ')'
|
||||
for (int i = 0; i < (argc - 1); i++) {
|
||||
s += *argv[i].toString(cx).string;
|
||||
if (i < (argc - 2))
|
||||
s += widenCString(", ");
|
||||
}
|
||||
/* ( */ s += ") {" + *argv[argc - 1].toString(cx).string + "}";
|
||||
}
|
||||
}
|
||||
|
||||
JSFunction *fnc = NULL;
|
||||
/***************************************************************/
|
||||
{
|
||||
Arena a;
|
||||
Parser p(cx->mWorld, a, cx->mFlags, s, widenCString("function constructor"));
|
||||
cx->mReader = &p.lexer.reader;
|
||||
|
||||
FunctionExprNode *f = p.parseFunctionExpression(0);
|
||||
if (!p.lexer.peek(true).hasKind(Token::end))
|
||||
cx->reportError(Exception::syntaxError, "Unexpected stuff after the function body");
|
||||
|
||||
fnc = new JSFunction(NULL, cx->mScopeChain);
|
||||
|
||||
uint32 reqArgCount = 0;
|
||||
uint32 optArgCount = 0;
|
||||
|
||||
VariableBinding *b = f->function.parameters;
|
||||
while ((b != f->function.optParameters) && (b != f->function.restParameter)) {
|
||||
reqArgCount++;
|
||||
b = b->next;
|
||||
}
|
||||
while (b != f->function.restParameter) {
|
||||
optArgCount++;
|
||||
b = b->next;
|
||||
}
|
||||
fnc->setArgCounts(reqArgCount, optArgCount, (f->function.restParameter != NULL));
|
||||
|
||||
if (cx->mScopeChain->isPossibleUncheckedFunction(&f->function))
|
||||
fnc->setIsPrototype(true);
|
||||
|
||||
cx->buildRuntimeForFunction(f->function, fnc);
|
||||
ByteCodeGen bcg(cx, cx->mScopeChain);
|
||||
bcg.genCodeForFunction(f->function, f->pos, fnc, false, NULL);
|
||||
}
|
||||
/***************************************************************/
|
||||
|
||||
v = JSValue(fnc);
|
||||
return v;
|
||||
}
|
||||
|
||||
@ -1720,7 +1811,7 @@ static JSValue Function_hasInstance(Context *cx, const JSValue& thisValue, JSVal
|
||||
return kFalseValue;
|
||||
|
||||
ASSERT(thisValue.isFunction());
|
||||
thisValue.function->getProperty(cx, widenCString("prototype"), CURRENT_ATTR);
|
||||
thisValue.function->getProperty(cx, cx->Prototype_StringAtom, CURRENT_ATTR);
|
||||
JSValue p = cx->popValue();
|
||||
|
||||
if (!p.isObject())
|
||||
@ -1749,14 +1840,23 @@ static JSValue Number_Constructor(Context *cx, const JSValue& thisValue, JSValue
|
||||
return v;
|
||||
}
|
||||
|
||||
static JSValue Number_toString(Context *, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
static JSValue Number_toString(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
if (thisValue.getType() != Number_Type)
|
||||
cx->reportError(Exception::typeError, "Number.toString called on something other than a Number object");
|
||||
JSObject *thisObj = thisValue.object;
|
||||
return JSValue(numberToString(*((double *)(thisObj->mPrivate))));
|
||||
}
|
||||
|
||||
|
||||
static JSValue Number_valueOf(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
if (thisValue.getType() != Number_Type)
|
||||
cx->reportError(Exception::typeError, "Number.valueOf called on something other than a Number object");
|
||||
JSObject *thisObj = thisValue.object;
|
||||
return JSValue(*((double *)(thisObj->mPrivate)));
|
||||
}
|
||||
|
||||
static JSValue Integer_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
@ -1800,15 +1900,26 @@ static JSValue Boolean_Constructor(Context *cx, const JSValue& thisValue, JSValu
|
||||
return v;
|
||||
}
|
||||
|
||||
static JSValue Boolean_toString(Context *, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
static JSValue Boolean_toString(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
|
||||
if (thisObj->mPrivate != 0)
|
||||
return JSValue(new String(widenCString("true")));
|
||||
return JSValue(&cx->True_StringAtom);
|
||||
else
|
||||
return JSValue(new String(widenCString("false")));
|
||||
return JSValue(&cx->False_StringAtom);
|
||||
}
|
||||
|
||||
static JSValue Boolean_valueOf(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
|
||||
if (thisObj->mPrivate != 0)
|
||||
return kTrueValue; //JSValue(1.0);
|
||||
else
|
||||
return kFalseValue; //kPositiveZero;
|
||||
}
|
||||
|
||||
static JSValue ExtendAttribute_Invoke(Context * /*cx*/, const JSValue& /*thisValue*/, JSValue *argv, uint32 argc)
|
||||
@ -1821,6 +1932,50 @@ static JSValue ExtendAttribute_Invoke(Context * /*cx*/, const JSValue& /*thisVal
|
||||
|
||||
return JSValue(x);
|
||||
}
|
||||
|
||||
static JSValue GlobalObject_Eval(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 /*argc*/)
|
||||
{
|
||||
if (!argv[0].isString())
|
||||
return argv[0];
|
||||
const String *sourceStr = argv[0].toString(cx).string;
|
||||
|
||||
Activation *prev = cx->mActivationStack.top();
|
||||
|
||||
return cx->readEvalString(*sourceStr, widenCString("eval source"), cx->mScopeChain, prev->mThis);
|
||||
}
|
||||
|
||||
static JSValue GlobalObject_ParseInt(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 argc)
|
||||
{
|
||||
ASSERT(argc >= 1);
|
||||
int radix = 0; // default for stringToInteger
|
||||
if (argc == 2)
|
||||
radix = argv[1].toInt32(cx).f64;
|
||||
if ((radix < 0) || (radix > 36))
|
||||
return kNaNValue;
|
||||
|
||||
const String *string = argv[0].toString(cx).string;
|
||||
const char16 *numEnd;
|
||||
const char16 *sBegin = string->begin();
|
||||
float64 f = 0.0;
|
||||
if (sBegin)
|
||||
f = stringToInteger(sBegin, string->end(), numEnd, radix);
|
||||
return JSValue(f);
|
||||
}
|
||||
|
||||
static JSValue GlobalObject_ParseFloat(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 argc)
|
||||
{
|
||||
ASSERT(argc == 1);
|
||||
|
||||
const String *string = argv[0].toString(cx).string;
|
||||
const char16 *numEnd;
|
||||
const char16 *sBegin = string->begin();
|
||||
float64 f = 0.0;
|
||||
if (sBegin)
|
||||
f = stringToDouble(sBegin, string->end(), numEnd);
|
||||
return JSValue(f);
|
||||
}
|
||||
|
||||
|
||||
|
||||
JSValue JSFunction::runArgInitializer(Context *cx, uint32 a, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
@ -1884,7 +2039,7 @@ void Context::initClass(JSType *type, ClassDef *cdef, PrototypeFunctions *pdef)
|
||||
FunctionName *fnName = new FunctionName();
|
||||
fun->setFunctionName(fnName);
|
||||
fun->setArgCounts(pdef->mDef[i].length, 0, false);
|
||||
type->mPrototypeObject->defineVariable(this, *name, //widenCString(pdef->mDef[i].name),
|
||||
type->mPrototypeObject->defineVariable(this, *name,
|
||||
(NamespaceList *)(NULL),
|
||||
pdef->mDef[i].result,
|
||||
JSValue(fun));
|
||||
@ -1901,36 +2056,40 @@ void Context::initClass(JSType *type, ClassDef *cdef, PrototypeFunctions *pdef)
|
||||
static JSValue arrayMaker(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 argc)
|
||||
{
|
||||
ASSERT(argc == 2);
|
||||
ASSERT(argv[0].isType() && argv[1].isType());
|
||||
|
||||
ASSERT(argv[0].isType());
|
||||
JSType *baseType = argv[0].type;
|
||||
ASSERT(baseType == Array_Type);
|
||||
|
||||
JSType *elementType = argv[1].type;
|
||||
if ((baseType == Array_Type) && argv[1].isType()) {
|
||||
JSType *elementType = argv[1].type;
|
||||
JSType *result = new JSArrayType(cx, elementType, NULL, Object_Type);
|
||||
result->setDefaultConstructor(cx, Array_Type->getDefaultConstructor());
|
||||
return JSValue(result);
|
||||
}
|
||||
else {
|
||||
// then it's just <type>[] - an element reference
|
||||
argv[0].type->getProperty(cx, *argv[1].toString(cx).string, NULL);
|
||||
return cx->popValue();
|
||||
}
|
||||
|
||||
JSType *result = new JSArrayType(cx, elementType, NULL, Object_Type);
|
||||
result->setDefaultConstructor(cx, Array_Type->getDefaultConstructor());
|
||||
|
||||
return JSValue(result);
|
||||
}
|
||||
|
||||
void Context::initBuiltins()
|
||||
{
|
||||
ClassDef builtInClasses[] =
|
||||
{
|
||||
{ "Object", Object_Constructor, &kNullValue },
|
||||
{ "Type", NULL, &kNullValue },
|
||||
{ "Function", Function_Constructor, &kNullValue },
|
||||
{ "Number", Number_Constructor, &kPositiveZero },
|
||||
{ "Integer", Integer_Constructor, &kPositiveZero },
|
||||
{ "String", String_Constructor, &kNullValue },
|
||||
{ "Array", Array_Constructor, &kNullValue },
|
||||
{ "Boolean", Boolean_Constructor, &kFalseValue },
|
||||
{ "Void", NULL, &kNullValue },
|
||||
{ "Unit", NULL, &kNullValue },
|
||||
{ "Attribute", NULL, &kNullValue },
|
||||
{ "NamedArgument", NULL, &kNullValue },
|
||||
{ "Date", Date_Constructor, &kPositiveZero },
|
||||
{ "Object", Object_Constructor, &kUndefinedValue },
|
||||
{ "Type", NULL, &kNullValue },
|
||||
{ "Function", Function_Constructor, &kNullValue },
|
||||
{ "Number", Number_Constructor, &kPositiveZero },
|
||||
{ "Integer", Integer_Constructor, &kPositiveZero },
|
||||
{ "String", String_Constructor, &kNullValue },
|
||||
{ "Array", Array_Constructor, &kNullValue },
|
||||
{ "Boolean", Boolean_Constructor, &kFalseValue },
|
||||
{ "Void", NULL, &kNullValue },
|
||||
{ "Unit", NULL, &kNullValue },
|
||||
{ "Attribute", NULL, &kNullValue },
|
||||
{ "NamedArgument", NULL, &kNullValue },
|
||||
{ "Date", Date_Constructor, &kPositiveZero },
|
||||
};
|
||||
|
||||
Object_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[0].name)], NULL);
|
||||
@ -1938,20 +2097,43 @@ void Context::initBuiltins()
|
||||
// XXX aren't all the built-ins thus?
|
||||
|
||||
Type_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[1].name)], Object_Type);
|
||||
Function_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[2].name)], Object_Type);
|
||||
Object_Type->mType = Type_Type;
|
||||
|
||||
//
|
||||
// ECMA 1.5 says Function.prototype is a Function object, it's [[class]] property is "Function"
|
||||
// ECMA 2.0 says Function.prototype is an Object NOT an instance of the class
|
||||
// here we sort of manage that by having a JSFunction object as the prototype object, not a JSFunctionInstance
|
||||
// (which we don't actually have, hmm).
|
||||
// For String, etc. this same issue needs to be finessed
|
||||
|
||||
JSFunction *funProto = new JSFunction(Object_Type, NULL);
|
||||
Function_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[2].name)], Object_Type, funProto);
|
||||
Function_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(funProto));
|
||||
|
||||
Number_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[3].name)], Object_Type);
|
||||
Number_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(Number_Type->mPrototypeObject));
|
||||
|
||||
Integer_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[4].name)], Object_Type);
|
||||
|
||||
String_Type = new JSStringType(this, &mWorld.identifiers[widenCString(builtInClasses[5].name)], Object_Type);
|
||||
String_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(String_Type->mPrototypeObject));
|
||||
|
||||
Array_Type = new JSArrayType(this, Object_Type, &mWorld.identifiers[widenCString(builtInClasses[6].name)], Object_Type);
|
||||
Array_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(Array_Type->mPrototypeObject));
|
||||
|
||||
Boolean_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[7].name)], Object_Type);
|
||||
Boolean_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(Boolean_Type->mPrototypeObject));
|
||||
|
||||
Void_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[8].name)], Object_Type);
|
||||
Unit_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[9].name)], Object_Type);
|
||||
Attribute_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[10].name)], Object_Type);
|
||||
NamedArgument_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[11].name)], Object_Type);
|
||||
|
||||
Date_Type = new JSType(this, &mWorld.identifiers[widenCString(builtInClasses[12].name)], Object_Type);
|
||||
Date_Type->defineVariable(this, Prototype_StringAtom, NULL, Object_Type, JSValue(Date_Type->mPrototypeObject));
|
||||
|
||||
|
||||
String_Type->defineVariable(this, widenCString("fromCharCode"), NULL, String_Type, JSValue(new JSFunction(String_fromCharCode, String_Type)));
|
||||
String_Type->defineVariable(this, FromCharCode_StringAtom, NULL, String_Type, JSValue(new JSFunction(String_fromCharCode, String_Type)));
|
||||
|
||||
|
||||
ProtoFunDef objectProtos[] =
|
||||
@ -1974,6 +2156,7 @@ void Context::initBuiltins()
|
||||
{
|
||||
{ "toString", String_Type, 0, Number_toString },
|
||||
{ "toSource", String_Type, 0, Number_toString },
|
||||
{ "valueOf", Number_Type, 0, Number_valueOf },
|
||||
{ NULL }
|
||||
};
|
||||
ProtoFunDef integerProtos[] =
|
||||
@ -1986,12 +2169,13 @@ void Context::initBuiltins()
|
||||
{
|
||||
{ "toString", String_Type, 0, Boolean_toString },
|
||||
{ "toSource", String_Type, 0, Boolean_toString },
|
||||
{ "valueOf", Number_Type, 0, Boolean_valueOf },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
ASSERT(mGlobal);
|
||||
*mGlobal = Object_Type->newInstance(this);
|
||||
initClass(Object_Type, &builtInClasses[0], new PrototypeFunctions(&objectProtos[0]) );
|
||||
initClass(Object_Type, &builtInClasses[0], new PrototypeFunctions(&objectProtos[0]) );
|
||||
|
||||
initClass(Type_Type, &builtInClasses[1], NULL );
|
||||
initClass(Function_Type, &builtInClasses[2], new PrototypeFunctions(&functionProtos[0]) );
|
||||
@ -2008,6 +2192,9 @@ void Context::initBuiltins()
|
||||
|
||||
Type_Type->defineUnaryOperator(Index, new JSFunction(arrayMaker, Type_Type));
|
||||
|
||||
Array_Type->defineUnaryOperator(Index, new JSFunction(Array_GetElement, Object_Type));
|
||||
Array_Type->defineUnaryOperator(IndexEqual, new JSFunction(Array_SetElement, Object_Type));
|
||||
|
||||
}
|
||||
|
||||
|
||||
@ -2076,13 +2263,38 @@ JS2Runtime::Operator Context::getOperator(uint32 parameterCount, const String &n
|
||||
|
||||
|
||||
Context::Context(JSObject **global, World &world, Arena &a, Pragma::Flags flags)
|
||||
: VirtualKeyWord(world.identifiers["virtual"]),
|
||||
ConstructorKeyWord(world.identifiers["constructor"]),
|
||||
OperatorKeyWord(world.identifiers["operator"]),
|
||||
FixedKeyWord(world.identifiers["fixed"]),
|
||||
DynamicKeyWord(world.identifiers["dynamic"]),
|
||||
ExtendKeyWord(world.identifiers["extend"]),
|
||||
PrototypeKeyWord(world.identifiers["prototype"]),
|
||||
|
||||
: Virtual_StringAtom (world.identifiers["virtual"]),
|
||||
Constructor_StringAtom (world.identifiers["constructor"]),
|
||||
Operator_StringAtom (world.identifiers["operator"]),
|
||||
Fixed_StringAtom (world.identifiers["fixed"]),
|
||||
Dynamic_StringAtom (world.identifiers["dynamic"]),
|
||||
Extend_StringAtom (world.identifiers["extend"]),
|
||||
Prototype_StringAtom (world.identifiers["prototype"]),
|
||||
Forin_StringAtom (world.identifiers["forin"]),
|
||||
Value_StringAtom (world.identifiers["value"]),
|
||||
Next_StringAtom (world.identifiers["next"]),
|
||||
Done_StringAtom (world.identifiers["done"]),
|
||||
Undefined_StringAtom (world.identifiers["undefined"]),
|
||||
Object_StringAtom (world.identifiers["object"]),
|
||||
Boolean_StringAtom (world.identifiers["boolean"]),
|
||||
Number_StringAtom (world.identifiers["number"]),
|
||||
String_StringAtom (world.identifiers["string"]),
|
||||
Function_StringAtom (world.identifiers["function"]),
|
||||
HasInstance_StringAtom (world.identifiers["hasInstance"]),
|
||||
True_StringAtom (world.identifiers["true"]),
|
||||
False_StringAtom (world.identifiers["false"]),
|
||||
Null_StringAtom (world.identifiers["null"]),
|
||||
ToString_StringAtom (world.identifiers["toString"]),
|
||||
ValueOf_StringAtom (world.identifiers["valueOf"]),
|
||||
Length_StringAtom (world.identifiers["length"]),
|
||||
FromCharCode_StringAtom (world.identifiers["fromCharCode"]),
|
||||
Math_StringAtom (world.identifiers["Math"]),
|
||||
NaN_StringAtom (world.identifiers["NaN"]),
|
||||
Eval_StringAtom (world.identifiers["eval"]),
|
||||
Infinity_StringAtom (world.identifiers["Infinity"]),
|
||||
Empty_StringAtom (world.identifiers[""]),
|
||||
Arguments_StringAtom (world.identifiers["arguments"]),
|
||||
|
||||
mWorld(world),
|
||||
mScopeChain(NULL),
|
||||
@ -2101,19 +2313,27 @@ Context::Context(JSObject **global, World &world, Arena &a, Pragma::Flags flags)
|
||||
mGlobal(global)
|
||||
|
||||
{
|
||||
uint32 i;
|
||||
|
||||
initOperatorTable();
|
||||
|
||||
mScopeChain = new ScopeChain(this, mWorld);
|
||||
if (Object_Type == NULL) {
|
||||
initBuiltins();
|
||||
JSObject *mathObj = Object_Type->newInstance(this);
|
||||
getGlobalObject()->defineVariable(this, widenCString("Math"), (NamespaceList *)(NULL), Object_Type, JSValue(mathObj));
|
||||
initMathObject(this, mathObj);
|
||||
getGlobalObject()->defineVariable(this, widenCString("undefined"), (NamespaceList *)(NULL), Void_Type, kUndefinedValue);
|
||||
getGlobalObject()->defineVariable(this, widenCString("NaN"), (NamespaceList *)(NULL), Void_Type, kNaNValue);
|
||||
getGlobalObject()->defineVariable(this, widenCString("Infinity"), (NamespaceList *)(NULL), Void_Type, kPositiveInfinity);
|
||||
initDateObject(this);
|
||||
}
|
||||
|
||||
|
||||
JSObject *mathObj = Object_Type->newInstance(this);
|
||||
getGlobalObject()->defineVariable(this, Math_StringAtom, (NamespaceList *)(NULL), Object_Type, JSValue(mathObj));
|
||||
initMathObject(this, mathObj);
|
||||
initDateObject(this);
|
||||
|
||||
Number_Type->defineVariable(this, widenCString("MAX_VALUE"), NULL, Number_Type, JSValue(maxValue));
|
||||
Number_Type->defineVariable(this, widenCString("MIN_VALUE"), NULL, Number_Type, JSValue(minValue));
|
||||
Number_Type->defineVariable(this, widenCString("NaN"), NULL, Number_Type, JSValue(nan));
|
||||
Number_Type->defineVariable(this, widenCString("POSITIVE_INFINITY"), NULL, Number_Type, JSValue(positiveInfinity));
|
||||
Number_Type->defineVariable(this, widenCString("NEGATIVE_INFINITY"), NULL, Number_Type, JSValue(negativeInfinity));
|
||||
|
||||
initOperators();
|
||||
|
||||
struct Attribute_Init {
|
||||
@ -2144,13 +2364,45 @@ Context::Context(JSObject **global, World &world, Arena &a, Pragma::Flags flags)
|
||||
{ "const", Property::Const, 0 },
|
||||
};
|
||||
|
||||
for (uint32 i = 0; i < (sizeof(attribute_init) / sizeof(Attribute_Init)); i++) {
|
||||
for (i = 0; i < (sizeof(attribute_init) / sizeof(Attribute_Init)); i++) {
|
||||
Attribute *attr = new Attribute(attribute_init[i].trueFlags, attribute_init[i].falseFlags);
|
||||
getGlobalObject()->defineVariable(this, widenCString(attribute_init[i].name), (NamespaceList *)(NULL), Attribute_Type, JSValue(attr));
|
||||
}
|
||||
|
||||
JSFunction *x = new JSFunction(ExtendAttribute_Invoke, Attribute_Type);
|
||||
getGlobalObject()->defineVariable(this, widenCString("extend"), (NamespaceList *)(NULL), Attribute_Type, JSValue(x));
|
||||
getGlobalObject()->defineVariable(this, Extend_StringAtom, (NamespaceList *)(NULL), Attribute_Type, JSValue(x));
|
||||
|
||||
|
||||
|
||||
|
||||
ProtoFunDef globalObjectFunctions[] = {
|
||||
{ "eval", Object_Type, 1, GlobalObject_Eval },
|
||||
{ "parseInt", Number_Type, 2, GlobalObject_ParseInt },
|
||||
{ "parseFloat", Number_Type, 2, GlobalObject_ParseFloat },
|
||||
};
|
||||
|
||||
for (i = 0; i < (sizeof(globalObjectFunctions) / sizeof(ProtoFunDef)); i++) {
|
||||
x = new JSFunction(globalObjectFunctions[i].imp, globalObjectFunctions[i].result);
|
||||
x->setArgCounts(globalObjectFunctions[i].length, 0, false);
|
||||
x->setIsPrototype(true);
|
||||
getGlobalObject()->defineVariable(this, widenCString(globalObjectFunctions[i].name), (NamespaceList *)(NULL), globalObjectFunctions[i].result, JSValue(x));
|
||||
}
|
||||
|
||||
/*
|
||||
x = new JSFunction(GlobalObject_Eval, Object_Type);
|
||||
x->setArgCounts(1, 0, false);
|
||||
x->setIsPrototype(true);
|
||||
getGlobalObject()->defineVariable(this, Eval_StringAtom, (NamespaceList *)(NULL), Object_Type, JSValue(x));
|
||||
|
||||
x = new JSFunction(GlobalObject_ParseInt, Number_Type);
|
||||
x->setArgCounts(2, 0, false);
|
||||
x->setIsPrototype(true);
|
||||
getGlobalObject()->defineVariable(this, widenCString("parseInt"), (NamespaceList *)(NULL), Object_Type, JSValue(x));
|
||||
*/
|
||||
|
||||
getGlobalObject()->defineVariable(this, Undefined_StringAtom, (NamespaceList *)(NULL), Void_Type, kUndefinedValue);
|
||||
getGlobalObject()->defineVariable(this, NaN_StringAtom, (NamespaceList *)(NULL), Void_Type, kNaNValue);
|
||||
getGlobalObject()->defineVariable(this, Infinity_StringAtom, (NamespaceList *)(NULL), Void_Type, kPositiveInfinity);
|
||||
|
||||
}
|
||||
|
||||
|
@ -174,6 +174,7 @@ static const double two31 = 2147483648.0;
|
||||
bool isType() const { return (tag == type_tag); }
|
||||
bool isFunction() const { return (tag == function_tag); }
|
||||
bool isString() const { return (tag == string_tag); }
|
||||
bool isPrimitive() const { return (tag != object_tag) && (tag != type_tag) && (tag != function_tag); }
|
||||
|
||||
bool isUndefined() const { return (tag == undefined_tag); }
|
||||
bool isNull() const { return (tag == null_tag); }
|
||||
@ -221,15 +222,15 @@ static const double two31 = 2147483648.0;
|
||||
};
|
||||
Formatter& operator<<(Formatter& f, const JSValue& value);
|
||||
|
||||
extern const JSValue kUndefinedValue;
|
||||
extern const JSValue kNaNValue;
|
||||
extern const JSValue kTrueValue;
|
||||
extern const JSValue kFalseValue;
|
||||
extern const JSValue kNullValue;
|
||||
extern const JSValue kNegativeZero;
|
||||
extern const JSValue kPositiveZero;
|
||||
extern const JSValue kNegativeInfinity;
|
||||
extern const JSValue kPositiveInfinity;
|
||||
extern JSValue kUndefinedValue;
|
||||
extern JSValue kNaNValue;
|
||||
extern JSValue kTrueValue;
|
||||
extern JSValue kFalseValue;
|
||||
extern JSValue kNullValue;
|
||||
extern JSValue kNegativeZero;
|
||||
extern JSValue kPositiveZero;
|
||||
extern JSValue kNegativeInfinity;
|
||||
extern JSValue kPositiveInfinity;
|
||||
|
||||
|
||||
|
||||
@ -546,6 +547,10 @@ XXX ...couldn't get this to work...
|
||||
virtual Property *defineVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type);
|
||||
virtual Property *defineVariable(Context *cx, const String &name, NamespaceList *names, JSType *type);
|
||||
|
||||
// add a property/value into the map
|
||||
// - assumes the map doesn't already have this property
|
||||
Property *insertNewProperty(const String &name, NamespaceList *names, PropertyAttribute attrFlags, JSType *type, const JSValue &v);
|
||||
|
||||
virtual Property *defineStaticVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type)
|
||||
{
|
||||
return JSObject::defineVariable(cx, name, attr, type);
|
||||
@ -576,8 +581,8 @@ XXX ...couldn't get this to work...
|
||||
virtual void defineGetterMethod(Context *cx, const String &name, AttributeStmtNode *attr, JSFunction *f);
|
||||
virtual void defineSetterMethod(Context *cx, const String &name, AttributeStmtNode *attr, JSFunction *f);
|
||||
|
||||
virtual Property *defineVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type, JSValue v);
|
||||
virtual Property *defineVariable(Context *cx, const String &name, NamespaceList *names, JSType *type, JSValue v);
|
||||
virtual Property *defineVariable(Context *cx, const String &name, AttributeStmtNode *attr, JSType *type, const JSValue v);
|
||||
virtual Property *defineVariable(Context *cx, const String &name, NamespaceList *names, JSType *type, const JSValue v);
|
||||
|
||||
virtual Reference *genReference(bool hasBase, const String& name, NamespaceList *names, Access acc, uint32 depth);
|
||||
|
||||
@ -687,7 +692,7 @@ XXX ...couldn't get this to work...
|
||||
|
||||
class JSType : public JSObject {
|
||||
public:
|
||||
JSType(Context *cx, const StringAtom *name, JSType *super);
|
||||
JSType(Context *cx, const StringAtom *name, JSType *super, JSObject *protoObj = NULL);
|
||||
JSType(JSType *xClass); // used for constructing the static component type
|
||||
|
||||
virtual ~JSType() { } // keeping gcc happy
|
||||
@ -1075,6 +1080,19 @@ XXX ...couldn't get this to work...
|
||||
return top->hasProperty(name, names, acc, p);
|
||||
}
|
||||
|
||||
bool hasOwnProperty(const String& name, NamespaceList *names, Access acc, PropertyIterator *p)
|
||||
{
|
||||
JSObject *top = mScopeStack.back();
|
||||
return top->hasOwnProperty(name, names, acc, p);
|
||||
}
|
||||
|
||||
// delete a property from the top object (already know it's there)
|
||||
void deleteProperty(const String &name, NamespaceList *names)
|
||||
{
|
||||
JSObject *top = mScopeStack.back();
|
||||
top->deleteProperty(name, names);
|
||||
}
|
||||
|
||||
// generate a reference to the given name
|
||||
Reference *getName(const String& name, NamespaceList *names, Access acc);
|
||||
|
||||
@ -1197,7 +1215,8 @@ XXX ...couldn't get this to work...
|
||||
if (scopeChain) {
|
||||
mScopeChain = new ScopeChain(*scopeChain);
|
||||
}
|
||||
mPrototype = Function_Type->mPrototypeObject;
|
||||
if (Function_Type) // protect against bootstrap
|
||||
mPrototype = Function_Type->mPrototypeObject;
|
||||
mActivation.mContainer = this;
|
||||
}
|
||||
|
||||
@ -1219,7 +1238,8 @@ XXX ...couldn't get this to work...
|
||||
mRestParameterName(NULL),
|
||||
mClass(NULL)
|
||||
{
|
||||
mPrototype = Function_Type->mPrototypeObject;
|
||||
if (Function_Type) // protect against bootstrap
|
||||
mPrototype = Function_Type->mPrototypeObject;
|
||||
mActivation.mContainer = this;
|
||||
}
|
||||
|
||||
@ -1420,13 +1440,37 @@ XXX ...couldn't get this to work...
|
||||
void operator delete(void* t) { trace_release("Context", t); STD::free(t); }
|
||||
#endif
|
||||
|
||||
StringAtom& VirtualKeyWord;
|
||||
StringAtom& ConstructorKeyWord;
|
||||
StringAtom& OperatorKeyWord;
|
||||
StringAtom& FixedKeyWord;
|
||||
StringAtom& DynamicKeyWord;
|
||||
StringAtom& ExtendKeyWord;
|
||||
StringAtom& PrototypeKeyWord;
|
||||
StringAtom& Virtual_StringAtom;
|
||||
StringAtom& Constructor_StringAtom;
|
||||
StringAtom& Operator_StringAtom;
|
||||
StringAtom& Fixed_StringAtom;
|
||||
StringAtom& Dynamic_StringAtom;
|
||||
StringAtom& Extend_StringAtom;
|
||||
StringAtom& Prototype_StringAtom;
|
||||
StringAtom& Forin_StringAtom;
|
||||
StringAtom& Value_StringAtom;
|
||||
StringAtom& Next_StringAtom;
|
||||
StringAtom& Done_StringAtom;
|
||||
StringAtom& Undefined_StringAtom;
|
||||
StringAtom& Object_StringAtom;
|
||||
StringAtom& Boolean_StringAtom;
|
||||
StringAtom& Number_StringAtom;
|
||||
StringAtom& String_StringAtom;
|
||||
StringAtom& Function_StringAtom;
|
||||
StringAtom& HasInstance_StringAtom;
|
||||
StringAtom& True_StringAtom;
|
||||
StringAtom& False_StringAtom;
|
||||
StringAtom& Null_StringAtom;
|
||||
StringAtom& ToString_StringAtom;
|
||||
StringAtom& ValueOf_StringAtom;
|
||||
StringAtom& Length_StringAtom;
|
||||
StringAtom& FromCharCode_StringAtom;
|
||||
StringAtom& Math_StringAtom;
|
||||
StringAtom& NaN_StringAtom;
|
||||
StringAtom& Eval_StringAtom;
|
||||
StringAtom& Infinity_StringAtom;
|
||||
StringAtom& Empty_StringAtom;
|
||||
StringAtom& Arguments_StringAtom;
|
||||
|
||||
void initBuiltins();
|
||||
void initClass(JSType *type, ClassDef *cdef, PrototypeFunctions *pdef);
|
||||
@ -1479,7 +1523,7 @@ XXX ...couldn't get this to work...
|
||||
uint32 mStackTop;
|
||||
uint32 mStackMax;
|
||||
|
||||
void pushValue(JSValue v)
|
||||
void pushValue(JSValue &v)
|
||||
{
|
||||
ASSERT(mStackTop < mStackMax);
|
||||
mStack[mStackTop++] = v;
|
||||
@ -1516,7 +1560,7 @@ XXX ...couldn't get this to work...
|
||||
// put the value in at 'index', lifting everything above that up by one
|
||||
void insertValue(JSValue v, uint32 index)
|
||||
{
|
||||
ASSERT((mStackTop + 1) < mStackMax);
|
||||
ASSERT(mStackTop < mStackMax); // we're effectively pushing one entry
|
||||
for (uint32 i = mStackTop - 1; i >= index; i--)
|
||||
mStack[i + 1] = mStack[i];
|
||||
mStack[index] = v;
|
||||
@ -1565,11 +1609,12 @@ XXX ...couldn't get this to work...
|
||||
|
||||
|
||||
JSValue readEvalFile(const String& fileName);
|
||||
JSValue readEvalString(const String &str, const String& fileName, ScopeChain *scopeChain, const JSValue& thisValue);
|
||||
|
||||
void buildRuntime(StmtNode *p);
|
||||
void buildRuntimeForFunction(FunctionDefinition &f, JSFunction *fnc);
|
||||
void buildRuntimeForStmt(StmtNode *p);
|
||||
ByteCodeModule *genCode(StmtNode *p, String sourceName);
|
||||
ByteCodeModule *genCode(StmtNode *p, const String &sourceName);
|
||||
|
||||
JSValue interpret(ByteCodeModule *bcm, int offset, ScopeChain *scopeChain, const JSValue& thisValue, JSValue *argv, uint32 argc);
|
||||
JSValue interpret(uint8 *pc, uint8 *endPC);
|
||||
|
@ -59,8 +59,7 @@ JSValue Array_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
thatValue = Array_Type->newInstance(cx);
|
||||
ASSERT(thatValue.isObject());
|
||||
JSObject *thisObj = thatValue.object;
|
||||
ASSERT(dynamic_cast<JSArrayInstance *>(thisObj));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)thisObj;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
if (argc > 0) {
|
||||
if (argc == 1) {
|
||||
arrInst->mLength = (uint32)(argv[0].toNumber(cx).f64);
|
||||
@ -82,13 +81,12 @@ static JSValue Array_toString(Context *cx, const JSValue& thisValue, JSValue * /
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
ASSERT(dynamic_cast<JSArrayInstance *>(thisObj));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)thisObj;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
|
||||
ContextStackReplacement csr(cx);
|
||||
|
||||
if (arrInst->mLength == 0)
|
||||
return JSValue(new String(widenCString("")));
|
||||
return JSValue(&cx->Empty_StringAtom);
|
||||
else {
|
||||
String *s = new String();
|
||||
for (uint32 i = 0; i < arrInst->mLength; i++) {
|
||||
@ -108,8 +106,7 @@ static JSValue Array_toSource(Context *cx, const JSValue& thisValue, JSValue * /
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
ASSERT(dynamic_cast<JSArrayInstance *>(thisObj));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)thisObj;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
|
||||
ContextStackReplacement csr(cx);
|
||||
|
||||
@ -136,8 +133,7 @@ static JSValue Array_push(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
ASSERT(dynamic_cast<JSArrayInstance *>(thisObj));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)thisObj;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
|
||||
for (uint32 i = 0; i < argc; i++) {
|
||||
const String *id = numberToString(i + arrInst->mLength);
|
||||
@ -152,8 +148,7 @@ static JSValue Array_pop(Context *cx, const JSValue& thisValue, JSValue * /*argv
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
ASSERT(dynamic_cast<JSArrayInstance *>(thisObj));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)thisObj;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
|
||||
ContextStackReplacement csr(cx);
|
||||
|
||||
@ -186,8 +181,8 @@ JSValue Array_concat(Context *cx, const JSValue& thisValue, JSValue *argv, uint3
|
||||
A->setProperty(cx, *id, CURRENT_ATTR, E);
|
||||
}
|
||||
else {
|
||||
ASSERT(E.isObject() && dynamic_cast<JSArrayInstance *>(E.object));
|
||||
JSArrayInstance *arrInst = (JSArrayInstance *)(E.object);
|
||||
ASSERT(E.isObject());
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(E.object);
|
||||
for (uint32 k = 0; k < arrInst->mLength; k++) {
|
||||
const String *id = numberToString(k);
|
||||
arrInst->getProperty(cx, *id, NULL);
|
||||
@ -209,7 +204,7 @@ static JSValue Array_join(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
|
||||
@ -244,7 +239,7 @@ static JSValue Array_reverse(Context *cx, const JSValue& thisValue, JSValue * /*
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
|
||||
@ -291,12 +286,12 @@ static JSValue Array_shift(Context *cx, const JSValue& thisValue, JSValue * /*ar
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
|
||||
if (length == 0) {
|
||||
thisObj->setProperty(cx, widenCString("length"), CURRENT_ATTR, result);
|
||||
thisObj->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, result);
|
||||
return kUndefinedValue;
|
||||
}
|
||||
|
||||
@ -317,7 +312,7 @@ static JSValue Array_shift(Context *cx, const JSValue& thisValue, JSValue * /*ar
|
||||
}
|
||||
|
||||
thisObj->deleteProperty(*numberToString(length - 1), CURRENT_ATTR);
|
||||
thisObj->setProperty(cx, widenCString("length"), CURRENT_ATTR, JSValue((float64)(length - 1)) );
|
||||
thisObj->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, JSValue((float64)(length - 1)) );
|
||||
|
||||
return result;
|
||||
}
|
||||
@ -331,7 +326,7 @@ static JSValue Array_slice(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
|
||||
JSArrayInstance *A = (JSArrayInstance *)Array_Type->newInstance(cx);
|
||||
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
|
||||
@ -386,7 +381,7 @@ static JSValue Array_slice(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
n++;
|
||||
start++;
|
||||
}
|
||||
A->setProperty(cx, widenCString("length"), CURRENT_ATTR, JSValue((float64)n) );
|
||||
A->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, JSValue((float64)n) );
|
||||
return JSValue(A);
|
||||
}
|
||||
|
||||
@ -403,7 +398,7 @@ static JSValue Array_splice(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
|
||||
@ -444,7 +439,7 @@ static JSValue Array_splice(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
A->setProperty(cx, *id2, CURRENT_ATTR, cx->popValue());
|
||||
}
|
||||
}
|
||||
A->setProperty(cx, widenCString("length"), CURRENT_ATTR, JSValue((float64)deleteCount) );
|
||||
A->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, JSValue((float64)deleteCount) );
|
||||
|
||||
uint32 newItemCount = argc - 2;
|
||||
if (newItemCount < deleteCount) {
|
||||
@ -484,7 +479,7 @@ static JSValue Array_splice(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
const String *id1 = numberToString(k++);
|
||||
thisObj->setProperty(cx, *id1, CURRENT_ATTR, argv[i]);
|
||||
}
|
||||
thisObj->setProperty(cx, widenCString("length"), CURRENT_ATTR, JSValue((float64)(length - deleteCount + newItemCount)) );
|
||||
thisObj->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, JSValue((float64)(length - deleteCount + newItemCount)) );
|
||||
|
||||
return JSValue(A);
|
||||
}
|
||||
@ -497,7 +492,7 @@ static JSValue Array_unshift(Context *cx, const JSValue& thisValue, JSValue *arg
|
||||
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
thisObj->getProperty(cx, widenCString("length"), CURRENT_ATTR);
|
||||
thisObj->getProperty(cx, cx->Length_StringAtom, CURRENT_ATTR);
|
||||
JSValue result = cx->popValue();
|
||||
uint32 length = (uint32)(result.toUInt32(cx).f64);
|
||||
uint32 k;
|
||||
@ -518,11 +513,83 @@ static JSValue Array_unshift(Context *cx, const JSValue& thisValue, JSValue *arg
|
||||
const String *id1 = numberToString(k);
|
||||
thisObj->setProperty(cx, *id1, CURRENT_ATTR, argv[k]);
|
||||
}
|
||||
thisObj->setProperty(cx, widenCString("length"), CURRENT_ATTR, JSValue((float64)(length + argc)) );
|
||||
thisObj->setProperty(cx, cx->Length_StringAtom, CURRENT_ATTR, JSValue((float64)(length + argc)) );
|
||||
|
||||
return thisValue;
|
||||
}
|
||||
|
||||
// An index has to pass the test that :
|
||||
// ToString(ToUint32(ToString(index))) == ToString(index)
|
||||
//
|
||||
// the 'ToString(index)' operation is the default behaviour of '[]'
|
||||
//
|
||||
// isArrayIndex() is called when we know that index is a Number
|
||||
//
|
||||
static bool isArrayIndex(Context *cx, JSValue &index, uint32 &intIndex)
|
||||
{
|
||||
ASSERT(index.isNumber());
|
||||
JSValue v = index.toUInt32(cx);
|
||||
if ((v.f64 == index.f64) && (v.f64 < two32minus1)) {
|
||||
intIndex = (uint32)v.f64;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
JSValue Array_GetElement(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
// the 'this' value is the first argument, then the index
|
||||
if (argc != 2)
|
||||
cx->reportError(Exception::referenceError, "[] only supports single dimension");
|
||||
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
ASSERT(thisObj->mType == Array_Type);
|
||||
|
||||
JSValue index = argv[1];
|
||||
const String *name = index.toString(cx).string;
|
||||
|
||||
arrInst->getProperty(cx, *name, NULL);
|
||||
return cx->popValue();
|
||||
}
|
||||
|
||||
JSValue Array_SetElement(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
// the 'this' value is the first argument, then the rValue, and finally the index
|
||||
if (argc != 3)
|
||||
cx->reportError(Exception::referenceError, "[]= only supports single dimension");
|
||||
|
||||
ASSERT(thisValue.isObject());
|
||||
JSObject *thisObj = thisValue.object;
|
||||
JSArrayInstance *arrInst = checked_cast<JSArrayInstance *>(thisObj);
|
||||
ASSERT(thisObj->mType == Array_Type);
|
||||
|
||||
JSValue index = argv[2];
|
||||
const String *name = index.toString(cx).string;
|
||||
|
||||
if (index.isNumber()) {
|
||||
uint32 intIndex;
|
||||
if (isArrayIndex(cx, index, intIndex)) {
|
||||
PropertyIterator it = thisObj->findNamespacedProperty(*name, NULL);
|
||||
if (it == thisObj->mProperties.end())
|
||||
thisObj->insertNewProperty(*name, NULL, 0, Object_Type, argv[1]);
|
||||
else {
|
||||
Property *prop = PROPERTY(it);
|
||||
ASSERT(prop->mFlag == ValuePointer);
|
||||
*prop->mData.vp = argv[1];
|
||||
}
|
||||
if (intIndex >= arrInst->mLength)
|
||||
arrInst->mLength = intIndex + 1;
|
||||
}
|
||||
}
|
||||
else {
|
||||
arrInst->setProperty(cx, *name, NULL, argv[1]);
|
||||
}
|
||||
return argv[0];
|
||||
}
|
||||
|
||||
|
||||
|
||||
Context::PrototypeFunctions *getArrayProtos()
|
||||
{
|
||||
|
@ -35,13 +35,13 @@ namespace JavaScript {
|
||||
|
||||
namespace JS2Runtime {
|
||||
|
||||
|
||||
extern JSValue Array_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc);
|
||||
extern JSValue Array_concat(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc);
|
||||
|
||||
Context::PrototypeFunctions *getArrayProtos();
|
||||
|
||||
|
||||
extern JSValue Array_GetElement(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc);
|
||||
extern JSValue Array_SetElement(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc);
|
||||
|
||||
}
|
||||
|
||||
|
@ -97,7 +97,7 @@ static float64 firstDayOfMonth[2][12] = {
|
||||
#define Day(t) floor((t) / msPerDay)
|
||||
#define TIMECLIP(d) ((JSDOUBLE_IS_FINITE(d) \
|
||||
&& !((d < 0 ? -d : d) > HalfTimeDomain)) \
|
||||
? JSValue::float64ToInteger(d + (+0.)) : kNaNValue.f64)
|
||||
? JSValue::float64ToInteger(d + (+0.)) : nan)
|
||||
#define LocalTime(t) ((t) + LocalTZA + DaylightSavingTA(t))
|
||||
#define DayFromMonth(m, leap) firstDayOfMonth[leap][(int32)m];
|
||||
#define DayFromYear(y) (365 * ((y)-1970) + fd::floor(((y)-1969)/4.0) \
|
||||
@ -368,7 +368,7 @@ static JSValue Date_makeTime(Context *cx, const JSValue& thisValue, JSValue *arg
|
||||
for (i = 0; i < argc; i++) {
|
||||
argv[i] = argv[i].toNumber(cx);
|
||||
if (JSDOUBLE_IS_NaN(argv[i])) {
|
||||
*date = kNaNValue.f64;
|
||||
*date = nan;
|
||||
return kNaNValue;
|
||||
}
|
||||
args[i] = argv[i].toInteger(cx).f64;
|
||||
@ -604,7 +604,7 @@ static float64 date_parseString(const String &s)
|
||||
|
||||
syntax:
|
||||
/* syntax error */
|
||||
return kNaNValue.f64;
|
||||
return nan;
|
||||
}
|
||||
|
||||
#define MAXARGS 7
|
||||
@ -673,7 +673,7 @@ JSValue Date_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv, u
|
||||
/* if any arg is NaN, make a NaN date object
|
||||
and return */
|
||||
if (!JSDOUBLE_IS_FINITE(double_arg)) {
|
||||
*((float64 *)(thisObj->mPrivate)) = kNaNValue.f64;
|
||||
*((float64 *)(thisObj->mPrivate)) = nan;
|
||||
return thatValue;
|
||||
}
|
||||
array[loop] = JSValue::float64ToInteger(double_arg);
|
||||
@ -1189,7 +1189,7 @@ static JSValue Date_setYear(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
result = *date;
|
||||
year = argv[0].toNumber(cx).f64;
|
||||
if (!JSDOUBLE_IS_FINITE(year)) {
|
||||
*date = kNaNValue.f64;
|
||||
*date = nan;
|
||||
return JSValue(*date);
|
||||
}
|
||||
|
||||
@ -1293,7 +1293,7 @@ static JSValue Date_valueOf(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
/* Convert to number only if the hint was given, otherwise favor string. */
|
||||
if (argc == 1) {
|
||||
const String *str = argv[0].toString(cx).string;
|
||||
if (str->compare(widenCString("number")) == 0)
|
||||
if (str->compare(&cx->Number_StringAtom) == 0)
|
||||
return Date_getTime(cx, thisValue, argv, argc);
|
||||
}
|
||||
return Date_toString(cx, thisValue, argv, argc);
|
||||
|
@ -57,20 +57,19 @@ JSValue String_Constructor(Context *cx, const JSValue& thisValue, JSValue *argv,
|
||||
thatValue = String_Type->newInstance(cx);
|
||||
ASSERT(thatValue.isObject());
|
||||
JSObject *thisObj = thatValue.object;
|
||||
ASSERT(dynamic_cast<JSStringInstance *>(thisObj));
|
||||
JSStringInstance *strInst = (JSStringInstance *)thisObj;
|
||||
JSStringInstance *strInst = checked_cast<JSStringInstance *>(thisObj);
|
||||
|
||||
if (argc > 0)
|
||||
thisObj->mPrivate = (void *)(new String(*argv[0].toString(cx).string));
|
||||
else
|
||||
thisObj->mPrivate = (void *)(new String(widenCString("")));
|
||||
thisObj->mPrivate = (void *)(&cx->Empty_StringAtom);
|
||||
strInst->mLength = ((String *)(thisObj->mPrivate))->size();
|
||||
return thatValue;
|
||||
}
|
||||
|
||||
JSValue String_fromCharCode(Context *cx, const JSValue& /*thisValue*/, JSValue *argv, uint32 argc)
|
||||
{
|
||||
String *resultStr = new String();
|
||||
String *resultStr = new String(); // can't use cx->mEmptyString because we're modifying this below
|
||||
resultStr->reserve(argc);
|
||||
for (uint32 i = 0; i < argc; i++)
|
||||
*resultStr += (char16)(argv[i].toUInt16(cx).f64);
|
||||
@ -78,13 +77,26 @@ JSValue String_fromCharCode(Context *cx, const JSValue& /*thisValue*/, JSValue *
|
||||
return JSValue(resultStr);
|
||||
}
|
||||
|
||||
static JSValue String_toString(Context * /*cx*/, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
static JSValue String_toString(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
if (thisValue.getType() != String_Type)
|
||||
cx->reportError(Exception::typeError, "String.toString called on something other than a string thing");
|
||||
JSObject *thisObj = thisValue.object;
|
||||
return JSValue((String *)thisObj->mPrivate);
|
||||
}
|
||||
|
||||
static JSValue String_valueOf(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
if (thisValue.getType() != String_Type)
|
||||
cx->reportError(Exception::typeError, "String.valueOf called on something other than a string thing");
|
||||
JSObject *thisObj = thisValue.object;
|
||||
return JSValue((String *)thisObj->mPrivate);
|
||||
}
|
||||
|
||||
|
||||
|
||||
struct MatchResult {
|
||||
bool failure;
|
||||
uint32 endIndex;
|
||||
@ -180,16 +192,6 @@ step11:
|
||||
|
||||
}
|
||||
|
||||
static JSValue String_valueOf(Context *cx, const JSValue& thisValue, JSValue * /*argv*/, uint32 /*argc*/)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
if (thisValue.isString())
|
||||
return thisValue;
|
||||
else
|
||||
cx->reportError(Exception::typeError, "String.valueOf called on something other than a string thing");
|
||||
return kUndefinedValue;
|
||||
}
|
||||
|
||||
static JSValue String_charAt(Context *cx, const JSValue& thisValue, JSValue *argv, uint32 argc)
|
||||
{
|
||||
ASSERT(thisValue.isObject());
|
||||
@ -200,7 +202,7 @@ static JSValue String_charAt(Context *cx, const JSValue& thisValue, JSValue *arg
|
||||
pos = (uint32)(argv[0].toInt32(cx).f64);
|
||||
|
||||
if ((pos < 0) || (pos >= str->size()))
|
||||
return JSValue(new String()); // have an empty string kValue somewhere?
|
||||
return JSValue(&cx->Empty_StringAtom);
|
||||
else
|
||||
return JSValue(new String(1, (*str)[pos]));
|
||||
|
||||
@ -367,7 +369,7 @@ static JSValue String_slice(Context *cx, const JSValue& thisValue, JSValue *argv
|
||||
end = sourceLength;
|
||||
|
||||
if (start > end)
|
||||
return JSValue(new String());
|
||||
return JSValue(&cx->Empty_StringAtom);
|
||||
return JSValue(new String(sourceString->substr(start, end - start)));
|
||||
}
|
||||
|
||||
@ -420,7 +422,7 @@ Context::PrototypeFunctions *getStringProtos()
|
||||
Context::ProtoFunDef stringProtos[] =
|
||||
{
|
||||
{ "toString", String_Type, 0, String_toString },
|
||||
{ "valueof", String_Type, 0, String_valueOf },
|
||||
{ "valueOf", String_Type, 0, String_valueOf },
|
||||
{ "charAt", String_Type, 1, String_charAt },
|
||||
{ "charCodeAt", Number_Type, 1, String_charCodeAt },
|
||||
{ "concat", String_Type, 1, String_concat },
|
||||
|
@ -225,9 +225,11 @@ static PRLock *freelist_lock;
|
||||
//
|
||||
|
||||
|
||||
double JS::positiveInfinity;
|
||||
double JS::negativeInfinity;
|
||||
double JS::nan;
|
||||
double JS::positiveInfinity;
|
||||
double JS::negativeInfinity;
|
||||
double JS::nan;
|
||||
double JS::minValue;
|
||||
double JS::maxValue;
|
||||
|
||||
|
||||
struct InitNumerics {InitNumerics();};
|
||||
@ -241,21 +243,24 @@ InitNumerics::InitNumerics()
|
||||
word1(JS::negativeInfinity) = 0;
|
||||
word0(JS::nan) = 0x7FFFFFFF;
|
||||
word1(JS::nan) = 0xFFFFFFFF;
|
||||
word0(JS::minValue) = 0;
|
||||
word1(JS::minValue) = 1;
|
||||
JS::maxValue = 1.7976931348623157E+308;
|
||||
}
|
||||
|
||||
|
||||
// had to move these here since they depend upon the values
|
||||
// initialized above, and we can't guarantee order other than
|
||||
// lexically in a single file.
|
||||
const JSValue JS::JS2Runtime::kUndefinedValue;
|
||||
const JSValue JS::JS2Runtime::kNaNValue = JSValue(nan);
|
||||
const JSValue JS::JS2Runtime::kTrueValue = JSValue(true);
|
||||
const JSValue JS::JS2Runtime::kFalseValue = JSValue(false);
|
||||
const JSValue JS::JS2Runtime::kNullValue = JSValue(JSValue::null_tag);
|
||||
const JSValue JS::JS2Runtime::kNegativeZero = JSValue(-0.0);
|
||||
const JSValue JS::JS2Runtime::kPositiveZero = JSValue(0.0);
|
||||
const JSValue JS::JS2Runtime::kNegativeInfinity = JSValue(negativeInfinity);
|
||||
const JSValue JS::JS2Runtime::kPositiveInfinity = JSValue(positiveInfinity);
|
||||
JSValue JS::JS2Runtime::kUndefinedValue;
|
||||
JSValue JS::JS2Runtime::kNaNValue = JSValue(nan);
|
||||
JSValue JS::JS2Runtime::kTrueValue = JSValue(true);
|
||||
JSValue JS::JS2Runtime::kFalseValue = JSValue(false);
|
||||
JSValue JS::JS2Runtime::kNullValue = JSValue(JSValue::null_tag);
|
||||
JSValue JS::JS2Runtime::kNegativeZero = JSValue(-0.0);
|
||||
JSValue JS::JS2Runtime::kPositiveZero = JSValue(0.0);
|
||||
JSValue JS::JS2Runtime::kNegativeInfinity = JSValue(negativeInfinity);
|
||||
JSValue JS::JS2Runtime::kPositiveInfinity = JSValue(positiveInfinity);
|
||||
|
||||
//
|
||||
// Portable double-precision floating point to string and back conversions
|
||||
|
@ -66,6 +66,8 @@ namespace JavaScript {
|
||||
extern double positiveInfinity;
|
||||
extern double negativeInfinity;
|
||||
extern double nan;
|
||||
extern double minValue;
|
||||
extern double maxValue;
|
||||
|
||||
//
|
||||
// Portable double-precision floating point to string and back conversions
|
||||
|
@ -137,7 +137,7 @@ namespace JS2Runtime {
|
||||
NamespaceList *mNext;
|
||||
};
|
||||
|
||||
typedef std::pair<Property *, NamespaceList *> NamespacedProperty;
|
||||
typedef std::pair<Property *, NamespaceList *> NamespacedProperty;
|
||||
|
||||
typedef std::multimap<String, NamespacedProperty *, std::less<const String> > PropertyMap;
|
||||
typedef PropertyMap::iterator PropertyIterator;
|
||||
|
Loading…
x
Reference in New Issue
Block a user