Mozilla/mozilla/js/tamarin/core/ClassClosure.cpp
gerv%gerv.net 0a3a70857a Bug 236613: fix formatting of MPL/LGPL/GPL tri-license.
git-svn-id: svn://10.0.0.236/trunk@220064 18797224-902f-48f8-a5cc-f745e15eee43
2007-02-13 18:05:02 +00:00

200 lines
5.7 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is [Open Source Virtual Machine.].
*
* The Initial Developer of the Original Code is
* Adobe System Incorporated.
* Portions created by the Initial Developer are Copyright (C) 2004-2006
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Adobe AS3 Team
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include "avmplus.h"
namespace avmplus
{
ClassClosure::ClassClosure(VTable *cvtable)
: ScriptObject(cvtable, NULL)
{
AvmAssert(traits()->sizeofInstance >= sizeof(ClassClosure));
// prototype will be set by caller
AvmAssert(cvtable->traits->itraits != NULL);
AvmAssert(ivtable() != NULL);
}
void ClassClosure::createVanillaPrototype()
{
prototype = toplevel()->objectClass->construct();
}
Atom ClassClosure::get_prototype()
{
// Illegal to apply this getter to anything but instances of Class
// (verifier should ensure this)
return prototype ? prototype->atom() : undefinedAtom;
}
void ClassClosure::set_prototype(Atom value)
{
// ISSUE can value be null/undefined?
if (AvmCore::isNullOrUndefined(value))
{
this->prototype = NULL;
}
else
{
if (!AvmCore::isObject(value))
{
toplevel()->throwTypeError(kPrototypeTypeError);
}
// allow any prototype object. if the object has methods or slots, so be it
this->prototype = AvmCore::atomToScriptObject(value);
}
}
VTable* ClassClosure::ivtable() const
{
return vtable->ivtable;
}
// this = argv[0] (ignored)
// arg1 = argv[1]
// argN = argv[argc]
Atom ClassClosure::construct(int argc, Atom* argv)
{
VTable* ivtable = this->ivtable();
AvmAssert(ivtable != NULL);
AvmAssert(argv != NULL); // need at least one arg spot passed in
if (prototype == NULL) // ES3 spec, 13.2.2 (we've already ensured prototype is either an Object or null)
prototype = AvmCore::atomToScriptObject(toplevel()->objectClass->get_prototype());
ScriptObject *obj = createInstance(ivtable, prototype);
if (vtable->call != NULL)
{
// this is a function
argv[0] = obj->atom(); // new object is receiver
MethodEnv* call = vtable->call;
Atom result = call->coerceEnter(argc, argv);
// for E3 13.2.2 compliance, check result and return it if (Type(result) is Object)
/* ISSUE does this apply to class constructors too?
answer: no. from E4: A constructor may invoke a return statement as long as that
statement does not supply a value; a constructor cannot return a value. The newly
created object is returned automatically. A constructor’s return type must be omitted.
A constructor always returns a new instance. */
return AvmCore::isNull(result) || AvmCore::isObject(result) ? result : obj->atom();
}
else
{
// this is a class
Atom a = obj->atom();
argv[0] = a; // new object is receiver
ivtable->init->coerceEnter(argc, argv);
// this is a class. always return new instance.
return a;
}
}
// this = argv[0]
// arg1 = argv[1]
// argN = argv[argc]
Atom ClassClosure::call(int argc, Atom* argv)
{
MethodEnv* call = vtable->call;
Toplevel* toplevel = this->toplevel();
if (call)
{
// invoke function body
// TODO if we kept track of a type's call signature we could do this at verify time
if (AvmCore::isNullOrUndefined(argv[0]))
{
// use callee's global object as this.
// see E3 15.3.4.4
AvmAssert(call->vtable->scope->getSize() > 0);
argv[0] = call->vtable->scope->getScope(0);
}
// make sure receiver is legal for callee
argv[0] = toplevel->coerce(argv[0], call->method->paramTraits(0));
return call->coerceEnter(argc, argv);
}
else
{
// explicit coercion of a class object.
if (argc != 1)
{
toplevel->throwArgumentError(kCoerceArgumentCountError, toplevel->core()->toErrorString(argc));
}
return toplevel->coerce(argv[1], (Traits*)ivtable()->traits);
}
}
int ClassClosure::get_length()
{
MethodEnv* call = vtable->call;
if (call)
return call->method->param_count;
else
return vtable->init->method->param_count;
}
#ifdef DEBUGGER
uint32 ClassClosure::size() const
{
uint32 size = ScriptObject::size();
size += vtable->size();
return size;
}
#endif
#ifdef AVMPLUS_VERBOSE
Stringp ClassClosure::format(AvmCore* core) const
{
if (traits()->name)
{
return traits()->format(core);
}
else
{
Stringp prefix = core->newString("CC{}@");
return core->concatStrings(prefix, core->formatAtomPtr(atom()));
}
}
#endif
}