mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
19391253d9
--HG-- rename : content/xbl/src/XBLChildrenElement.cpp => dom/xbl/XBLChildrenElement.cpp rename : content/xbl/src/XBLChildrenElement.h => dom/xbl/XBLChildrenElement.h rename : content/xbl/builtin/android/jar.mn => dom/xbl/builtin/android/jar.mn rename : content/xbl/builtin/android/moz.build => dom/xbl/builtin/android/moz.build rename : content/xbl/builtin/android/platformHTMLBindings.xml => dom/xbl/builtin/android/platformHTMLBindings.xml rename : content/xbl/builtin/browser-base.inc => dom/xbl/builtin/browser-base.inc rename : content/xbl/builtin/editor-base.inc => dom/xbl/builtin/editor-base.inc rename : content/xbl/builtin/emacs/jar.mn => dom/xbl/builtin/emacs/jar.mn rename : content/xbl/builtin/emacs/moz.build => dom/xbl/builtin/emacs/moz.build rename : content/xbl/builtin/emacs/platformHTMLBindings.xml => dom/xbl/builtin/emacs/platformHTMLBindings.xml rename : content/xbl/builtin/input-fields-base.inc => dom/xbl/builtin/input-fields-base.inc rename : content/xbl/builtin/mac/jar.mn => dom/xbl/builtin/mac/jar.mn rename : content/xbl/builtin/mac/moz.build => dom/xbl/builtin/mac/moz.build rename : content/xbl/builtin/mac/platformHTMLBindings.xml => dom/xbl/builtin/mac/platformHTMLBindings.xml rename : content/xbl/builtin/moz.build => dom/xbl/builtin/moz.build rename : content/xbl/builtin/textareas-base.inc => dom/xbl/builtin/textareas-base.inc rename : content/xbl/builtin/unix/jar.mn => dom/xbl/builtin/unix/jar.mn rename : content/xbl/builtin/unix/moz.build => dom/xbl/builtin/unix/moz.build rename : content/xbl/builtin/unix/platformHTMLBindings.xml => dom/xbl/builtin/unix/platformHTMLBindings.xml rename : content/xbl/builtin/win/jar.mn => dom/xbl/builtin/win/jar.mn rename : content/xbl/builtin/win/moz.build => dom/xbl/builtin/win/moz.build rename : content/xbl/builtin/win/platformHTMLBindings.xml => dom/xbl/builtin/win/platformHTMLBindings.xml rename : content/xbl/crashtests/205735-1.xhtml => dom/xbl/crashtests/205735-1.xhtml rename : content/xbl/crashtests/223799-1.xul => dom/xbl/crashtests/223799-1.xul rename : content/xbl/crashtests/226744-1.xhtml => dom/xbl/crashtests/226744-1.xhtml rename : content/xbl/crashtests/232095-1.xul => dom/xbl/crashtests/232095-1.xul rename : content/xbl/crashtests/277523-1.xhtml => dom/xbl/crashtests/277523-1.xhtml rename : content/xbl/crashtests/277950-1.xhtml => dom/xbl/crashtests/277950-1.xhtml rename : content/xbl/crashtests/336744-1-inner.html => dom/xbl/crashtests/336744-1-inner.html rename : content/xbl/crashtests/336744-1.html => dom/xbl/crashtests/336744-1.html rename : content/xbl/crashtests/336960-1-inner.xhtml => dom/xbl/crashtests/336960-1-inner.xhtml rename : content/xbl/crashtests/336960-1.html => dom/xbl/crashtests/336960-1.html rename : content/xbl/crashtests/342954-1.xhtml => dom/xbl/crashtests/342954-1.xhtml rename : content/xbl/crashtests/342954-2-xbl.xml => dom/xbl/crashtests/342954-2-xbl.xml rename : content/xbl/crashtests/342954-2.xhtml => dom/xbl/crashtests/342954-2.xhtml rename : content/xbl/crashtests/368276-1.xhtml => dom/xbl/crashtests/368276-1.xhtml rename : content/xbl/crashtests/368641-1.xhtml => dom/xbl/crashtests/368641-1.xhtml rename : content/xbl/crashtests/378521-1.xhtml => dom/xbl/crashtests/378521-1.xhtml rename : content/xbl/crashtests/382376-1.xhtml => dom/xbl/crashtests/382376-1.xhtml rename : content/xbl/crashtests/382376-2.xhtml => dom/xbl/crashtests/382376-2.xhtml rename : content/xbl/crashtests/397596-1.xhtml => dom/xbl/crashtests/397596-1.xhtml rename : content/xbl/crashtests/404125-1.xhtml => dom/xbl/crashtests/404125-1.xhtml rename : content/xbl/crashtests/406900-1.xul => dom/xbl/crashtests/406900-1.xul rename : content/xbl/crashtests/406904-1.xhtml => dom/xbl/crashtests/406904-1.xhtml rename : content/xbl/crashtests/406904-2.xhtml => dom/xbl/crashtests/406904-2.xhtml rename : content/xbl/crashtests/415192-1.xul => dom/xbl/crashtests/415192-1.xul rename : content/xbl/crashtests/415301-1.xul => dom/xbl/crashtests/415301-1.xul rename : content/xbl/crashtests/418133-1.xhtml => dom/xbl/crashtests/418133-1.xhtml rename : content/xbl/crashtests/420233-1.xhtml => dom/xbl/crashtests/420233-1.xhtml rename : content/xbl/crashtests/421997-1.xhtml => dom/xbl/crashtests/421997-1.xhtml rename : content/xbl/crashtests/432813-1-xbl.xml => dom/xbl/crashtests/432813-1-xbl.xml rename : content/xbl/crashtests/432813-1.xhtml => dom/xbl/crashtests/432813-1.xhtml rename : content/xbl/crashtests/454820-1.html => dom/xbl/crashtests/454820-1.html rename : content/xbl/crashtests/460665-1.xhtml => dom/xbl/crashtests/460665-1.xhtml rename : content/xbl/crashtests/463511-1.xhtml => dom/xbl/crashtests/463511-1.xhtml rename : content/xbl/crashtests/464863-1.xhtml => dom/xbl/crashtests/464863-1.xhtml rename : content/xbl/crashtests/472260-1.xhtml => dom/xbl/crashtests/472260-1.xhtml rename : content/xbl/crashtests/477878-1.html => dom/xbl/crashtests/477878-1.html rename : content/xbl/crashtests/492978-1.xul => dom/xbl/crashtests/492978-1.xul rename : content/xbl/crashtests/493123-1.xhtml => dom/xbl/crashtests/493123-1.xhtml rename : content/xbl/crashtests/495354-1.xhtml => dom/xbl/crashtests/495354-1.xhtml rename : content/xbl/crashtests/507628-1.xhtml => dom/xbl/crashtests/507628-1.xhtml rename : content/xbl/crashtests/507991-1.xhtml => dom/xbl/crashtests/507991-1.xhtml rename : content/xbl/crashtests/830614-1.xul => dom/xbl/crashtests/830614-1.xul rename : content/xbl/crashtests/895805-1.xhtml => dom/xbl/crashtests/895805-1.xhtml rename : content/xbl/crashtests/crashtests.list => dom/xbl/crashtests/crashtests.list rename : content/xbl/crashtests/set-field-bad-this.xhtml => dom/xbl/crashtests/set-field-bad-this.xhtml rename : content/xbl/src/moz.build => dom/xbl/moz.build rename : content/xbl/src/nsBindingManager.cpp => dom/xbl/nsBindingManager.cpp rename : content/xbl/src/nsBindingManager.h => dom/xbl/nsBindingManager.h rename : content/xbl/src/nsXBLBinding.cpp => dom/xbl/nsXBLBinding.cpp rename : content/xbl/src/nsXBLBinding.h => dom/xbl/nsXBLBinding.h rename : content/xbl/src/nsXBLContentSink.cpp => dom/xbl/nsXBLContentSink.cpp rename : content/xbl/src/nsXBLContentSink.h => dom/xbl/nsXBLContentSink.h rename : content/xbl/src/nsXBLDocumentInfo.cpp => dom/xbl/nsXBLDocumentInfo.cpp rename : content/xbl/src/nsXBLDocumentInfo.h => dom/xbl/nsXBLDocumentInfo.h rename : content/xbl/src/nsXBLEventHandler.cpp => dom/xbl/nsXBLEventHandler.cpp rename : content/xbl/src/nsXBLEventHandler.h => dom/xbl/nsXBLEventHandler.h rename : content/xbl/src/nsXBLMaybeCompiled.h => dom/xbl/nsXBLMaybeCompiled.h rename : content/xbl/src/nsXBLProtoImpl.cpp => dom/xbl/nsXBLProtoImpl.cpp rename : content/xbl/src/nsXBLProtoImpl.h => dom/xbl/nsXBLProtoImpl.h rename : content/xbl/src/nsXBLProtoImplField.cpp => dom/xbl/nsXBLProtoImplField.cpp rename : content/xbl/src/nsXBLProtoImplField.h => dom/xbl/nsXBLProtoImplField.h rename : content/xbl/src/nsXBLProtoImplMember.h => dom/xbl/nsXBLProtoImplMember.h rename : content/xbl/src/nsXBLProtoImplMethod.cpp => dom/xbl/nsXBLProtoImplMethod.cpp rename : content/xbl/src/nsXBLProtoImplMethod.h => dom/xbl/nsXBLProtoImplMethod.h rename : content/xbl/src/nsXBLProtoImplProperty.cpp => dom/xbl/nsXBLProtoImplProperty.cpp rename : content/xbl/src/nsXBLProtoImplProperty.h => dom/xbl/nsXBLProtoImplProperty.h rename : content/xbl/src/nsXBLPrototypeBinding.cpp => dom/xbl/nsXBLPrototypeBinding.cpp rename : content/xbl/src/nsXBLPrototypeBinding.h => dom/xbl/nsXBLPrototypeBinding.h rename : content/xbl/src/nsXBLPrototypeHandler.cpp => dom/xbl/nsXBLPrototypeHandler.cpp rename : content/xbl/src/nsXBLPrototypeHandler.h => dom/xbl/nsXBLPrototypeHandler.h rename : content/xbl/src/nsXBLPrototypeResources.cpp => dom/xbl/nsXBLPrototypeResources.cpp rename : content/xbl/src/nsXBLPrototypeResources.h => dom/xbl/nsXBLPrototypeResources.h rename : content/xbl/src/nsXBLResourceLoader.cpp => dom/xbl/nsXBLResourceLoader.cpp rename : content/xbl/src/nsXBLResourceLoader.h => dom/xbl/nsXBLResourceLoader.h rename : content/xbl/src/nsXBLSerialize.cpp => dom/xbl/nsXBLSerialize.cpp rename : content/xbl/src/nsXBLSerialize.h => dom/xbl/nsXBLSerialize.h rename : content/xbl/src/nsXBLService.cpp => dom/xbl/nsXBLService.cpp rename : content/xbl/src/nsXBLService.h => dom/xbl/nsXBLService.h rename : content/xbl/src/nsXBLWindowKeyHandler.cpp => dom/xbl/nsXBLWindowKeyHandler.cpp rename : content/xbl/src/nsXBLWindowKeyHandler.h => dom/xbl/nsXBLWindowKeyHandler.h rename : content/xbl/test/bug310107-resource.xhtml => dom/xbl/test/bug310107-resource.xhtml rename : content/xbl/test/chrome.ini => dom/xbl/test/chrome.ini rename : content/xbl/test/file_bug372769.xhtml => dom/xbl/test/file_bug372769.xhtml rename : content/xbl/test/file_bug379959_cross.html => dom/xbl/test/file_bug379959_cross.html rename : content/xbl/test/file_bug379959_data.html => dom/xbl/test/file_bug379959_data.html rename : content/xbl/test/file_bug379959_xbl.xml => dom/xbl/test/file_bug379959_xbl.xml rename : content/xbl/test/file_bug397934.xhtml => dom/xbl/test/file_bug397934.xhtml rename : content/xbl/test/file_bug481558.xbl => dom/xbl/test/file_bug481558.xbl rename : content/xbl/test/file_bug481558css.sjs => dom/xbl/test/file_bug481558css.sjs rename : content/xbl/test/file_bug591198_inner.html => dom/xbl/test/file_bug591198_inner.html rename : content/xbl/test/file_bug591198_xbl.xml => dom/xbl/test/file_bug591198_xbl.xml rename : content/xbl/test/file_bug821850.xhtml => dom/xbl/test/file_bug821850.xhtml rename : content/xbl/test/file_bug844783.xhtml => dom/xbl/test/file_bug844783.xhtml rename : content/xbl/test/file_bug944407.html => dom/xbl/test/file_bug944407.html rename : content/xbl/test/file_bug944407.xml => dom/xbl/test/file_bug944407.xml rename : content/xbl/test/file_bug950909.html => dom/xbl/test/file_bug950909.html rename : content/xbl/test/file_bug950909.xml => dom/xbl/test/file_bug950909.xml rename : content/xbl/test/mochitest.ini => dom/xbl/test/mochitest.ini rename : content/xbl/test/moz.build => dom/xbl/test/moz.build rename : content/xbl/test/test_bug310107.html => dom/xbl/test/test_bug310107.html rename : content/xbl/test/test_bug366770.html => dom/xbl/test/test_bug366770.html rename : content/xbl/test/test_bug371724.xhtml => dom/xbl/test/test_bug371724.xhtml rename : content/xbl/test/test_bug372769.html => dom/xbl/test/test_bug372769.html rename : content/xbl/test/test_bug378518.xul => dom/xbl/test/test_bug378518.xul rename : content/xbl/test/test_bug378866.xhtml => dom/xbl/test/test_bug378866.xhtml rename : content/xbl/test/test_bug379959.html => dom/xbl/test/test_bug379959.html rename : content/xbl/test/test_bug389322.xhtml => dom/xbl/test/test_bug389322.xhtml rename : content/xbl/test/test_bug397934.html => dom/xbl/test/test_bug397934.html rename : content/xbl/test/test_bug398135.xul => dom/xbl/test/test_bug398135.xul rename : content/xbl/test/test_bug398492.xul => dom/xbl/test/test_bug398492.xul rename : content/xbl/test/test_bug400705.xhtml => dom/xbl/test/test_bug400705.xhtml rename : content/xbl/test/test_bug401907.xhtml => dom/xbl/test/test_bug401907.xhtml rename : content/xbl/test/test_bug403162.xhtml => dom/xbl/test/test_bug403162.xhtml rename : content/xbl/test/test_bug468210.xhtml => dom/xbl/test/test_bug468210.xhtml rename : content/xbl/test/test_bug481558.html => dom/xbl/test/test_bug481558.html rename : content/xbl/test/test_bug526178.xhtml => dom/xbl/test/test_bug526178.xhtml rename : content/xbl/test/test_bug542406.xhtml => dom/xbl/test/test_bug542406.xhtml rename : content/xbl/test/test_bug591198.html => dom/xbl/test/test_bug591198.html rename : content/xbl/test/test_bug639338.xhtml => dom/xbl/test/test_bug639338.xhtml rename : content/xbl/test/test_bug721452.xul => dom/xbl/test/test_bug721452.xul rename : content/xbl/test/test_bug723676.xul => dom/xbl/test/test_bug723676.xul rename : content/xbl/test/test_bug772966.xul => dom/xbl/test/test_bug772966.xul rename : content/xbl/test/test_bug790265.xhtml => dom/xbl/test/test_bug790265.xhtml rename : content/xbl/test/test_bug821850.html => dom/xbl/test/test_bug821850.html rename : content/xbl/test/test_bug844783.html => dom/xbl/test/test_bug844783.html rename : content/xbl/test/test_bug944407.xul => dom/xbl/test/test_bug944407.xul rename : content/xbl/test/test_bug950909.xul => dom/xbl/test/test_bug950909.xul extra : rebase_source : 44ab05088f70826c70dee3af30221e628ec1e4e8
499 lines
17 KiB
C++
499 lines
17 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "nsIAtom.h"
|
|
#include "nsIContent.h"
|
|
#include "nsString.h"
|
|
#include "nsJSUtils.h"
|
|
#include "jsapi.h"
|
|
#include "js/CharacterEncoding.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "nsReadableUtils.h"
|
|
#include "nsXBLProtoImplField.h"
|
|
#include "nsIScriptContext.h"
|
|
#include "nsIURI.h"
|
|
#include "nsXBLSerialize.h"
|
|
#include "nsXBLPrototypeBinding.h"
|
|
#include "nsCxPusher.h"
|
|
#include "mozilla/dom/BindingUtils.h"
|
|
#include "xpcpublic.h"
|
|
#include "WrapperFactory.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
nsXBLProtoImplField::nsXBLProtoImplField(const char16_t* aName, const char16_t* aReadOnly)
|
|
: mNext(nullptr),
|
|
mFieldText(nullptr),
|
|
mFieldTextLength(0),
|
|
mLineNumber(0)
|
|
{
|
|
MOZ_COUNT_CTOR(nsXBLProtoImplField);
|
|
mName = NS_strdup(aName); // XXXbz make more sense to use a stringbuffer?
|
|
|
|
mJSAttributes = JSPROP_ENUMERATE;
|
|
if (aReadOnly) {
|
|
nsAutoString readOnly; readOnly.Assign(aReadOnly);
|
|
if (readOnly.LowerCaseEqualsLiteral("true"))
|
|
mJSAttributes |= JSPROP_READONLY;
|
|
}
|
|
}
|
|
|
|
|
|
nsXBLProtoImplField::nsXBLProtoImplField(const bool aIsReadOnly)
|
|
: mNext(nullptr),
|
|
mFieldText(nullptr),
|
|
mFieldTextLength(0),
|
|
mLineNumber(0)
|
|
{
|
|
MOZ_COUNT_CTOR(nsXBLProtoImplField);
|
|
|
|
mJSAttributes = JSPROP_ENUMERATE;
|
|
if (aIsReadOnly)
|
|
mJSAttributes |= JSPROP_READONLY;
|
|
}
|
|
|
|
nsXBLProtoImplField::~nsXBLProtoImplField()
|
|
{
|
|
MOZ_COUNT_DTOR(nsXBLProtoImplField);
|
|
if (mFieldText)
|
|
nsMemory::Free(mFieldText);
|
|
NS_Free(mName);
|
|
NS_CONTENT_DELETE_LIST_MEMBER(nsXBLProtoImplField, this, mNext);
|
|
}
|
|
|
|
void
|
|
nsXBLProtoImplField::AppendFieldText(const nsAString& aText)
|
|
{
|
|
if (mFieldText) {
|
|
nsDependentString fieldTextStr(mFieldText, mFieldTextLength);
|
|
nsAutoString newFieldText = fieldTextStr + aText;
|
|
char16_t* temp = mFieldText;
|
|
mFieldText = ToNewUnicode(newFieldText);
|
|
mFieldTextLength = newFieldText.Length();
|
|
nsMemory::Free(temp);
|
|
}
|
|
else {
|
|
mFieldText = ToNewUnicode(aText);
|
|
mFieldTextLength = aText.Length();
|
|
}
|
|
}
|
|
|
|
// XBL fields are represented on elements inheriting that field a bit trickily.
|
|
// When setting up the XBL prototype object, we install accessors for the fields
|
|
// on the prototype object. Those accessors, when used, will then (via
|
|
// InstallXBLField below) reify a property for the field onto the actual XBL-backed
|
|
// element.
|
|
//
|
|
// The accessor property is a plain old property backed by a getter function and
|
|
// a setter function. These properties are backed by the FieldGetter and
|
|
// FieldSetter natives; they're created by InstallAccessors. The precise field to be
|
|
// reified is identified using two extra slots on the getter/setter functions.
|
|
// XBLPROTO_SLOT stores the XBL prototype object that provides the field.
|
|
// FIELD_SLOT stores the name of the field, i.e. its JavaScript property name.
|
|
//
|
|
// This two-step field installation process -- creating an accessor on the
|
|
// prototype, then have that reify an own property on the actual element -- is
|
|
// admittedly convoluted. Better would be for XBL-backed elements to be proxies
|
|
// that could resolve fields onto themselves. But given that XBL bindings are
|
|
// associated with elements mutably -- you can add/remove/change -moz-binding
|
|
// whenever you want, alas -- doing so would require all elements to be proxies,
|
|
// which isn't performant now. So we do this two-step instead.
|
|
static const uint32_t XBLPROTO_SLOT = 0;
|
|
static const uint32_t FIELD_SLOT = 1;
|
|
|
|
bool
|
|
ValueHasISupportsPrivate(JS::Handle<JS::Value> v)
|
|
{
|
|
if (!v.isObject()) {
|
|
return false;
|
|
}
|
|
|
|
const DOMClass* domClass = GetDOMClass(&v.toObject());
|
|
if (domClass) {
|
|
return domClass->mDOMObjectIsISupports;
|
|
}
|
|
|
|
const JSClass* clasp = ::JS_GetClass(&v.toObject());
|
|
const uint32_t HAS_PRIVATE_NSISUPPORTS =
|
|
JSCLASS_HAS_PRIVATE | JSCLASS_PRIVATE_IS_NSISUPPORTS;
|
|
return (clasp->flags & HAS_PRIVATE_NSISUPPORTS) == HAS_PRIVATE_NSISUPPORTS;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
static bool
|
|
ValueHasISupportsPrivate(JSContext* cx, const JS::Value& aVal)
|
|
{
|
|
JS::Rooted<JS::Value> v(cx, aVal);
|
|
return ValueHasISupportsPrivate(v);
|
|
}
|
|
#endif
|
|
|
|
// Define a shadowing property on |this| for the XBL field defined by the
|
|
// contents of the callee's reserved slots. If the property was defined,
|
|
// *installed will be true, and idp will be set to the property name that was
|
|
// defined.
|
|
static bool
|
|
InstallXBLField(JSContext* cx,
|
|
JS::Handle<JSObject*> callee, JS::Handle<JSObject*> thisObj,
|
|
JS::MutableHandle<jsid> idp, bool* installed)
|
|
{
|
|
*installed = false;
|
|
|
|
// First ensure |this| is a reasonable XBL bound node.
|
|
//
|
|
// FieldAccessorGuard already determined whether |thisObj| was acceptable as
|
|
// |this| in terms of not throwing a TypeError. Assert this for good measure.
|
|
MOZ_ASSERT(ValueHasISupportsPrivate(cx, JS::ObjectValue(*thisObj)));
|
|
|
|
// But there are some cases where we must accept |thisObj| but not install a
|
|
// property on it, or otherwise touch it. Hence this split of |this|-vetting
|
|
// duties.
|
|
nsISupports* native =
|
|
nsContentUtils::XPConnect()->GetNativeOfWrapper(cx, thisObj);
|
|
if (!native) {
|
|
// Looks like whatever |thisObj| is it's not our nsIContent. It might well
|
|
// be the proto our binding installed, however, where the private is the
|
|
// nsXBLDocumentInfo, so just baul out quietly. Do NOT throw an exception
|
|
// here.
|
|
//
|
|
// We could make this stricter by checking the class maybe, but whatever.
|
|
return true;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> xblNode = do_QueryInterface(native);
|
|
if (!xblNode) {
|
|
xpc::Throw(cx, NS_ERROR_UNEXPECTED);
|
|
return false;
|
|
}
|
|
|
|
// Now that |this| is okay, actually install the field.
|
|
|
|
// Because of the possibility (due to XBL binding inheritance, because each
|
|
// XBL binding lives in its own global object) that |this| might be in a
|
|
// different compartment from the callee (not to mention that this method can
|
|
// be called with an arbitrary |this| regardless of how insane XBL is), and
|
|
// because in this method we've entered |this|'s compartment (see in
|
|
// Field[GS]etter where we attempt a cross-compartment call), we must enter
|
|
// the callee's compartment to access its reserved slots.
|
|
nsXBLPrototypeBinding* protoBinding;
|
|
nsDependentJSString fieldName;
|
|
{
|
|
JSAutoCompartment ac(cx, callee);
|
|
|
|
JS::Rooted<JSObject*> xblProto(cx);
|
|
xblProto = &js::GetFunctionNativeReserved(callee, XBLPROTO_SLOT).toObject();
|
|
|
|
JS::Rooted<JS::Value> name(cx, js::GetFunctionNativeReserved(callee, FIELD_SLOT));
|
|
JSFlatString* fieldStr = JS_ASSERT_STRING_IS_FLAT(name.toString());
|
|
fieldName.init(fieldStr);
|
|
|
|
MOZ_ALWAYS_TRUE(JS_ValueToId(cx, name, idp.address()));
|
|
|
|
// If a separate XBL scope is being used, the callee is not same-compartment
|
|
// with the xbl prototype, and the object is a cross-compartment wrapper.
|
|
xblProto = js::UncheckedUnwrap(xblProto);
|
|
JSAutoCompartment ac2(cx, xblProto);
|
|
JS::Value slotVal = ::JS_GetReservedSlot(xblProto, 0);
|
|
protoBinding = static_cast<nsXBLPrototypeBinding*>(slotVal.toPrivate());
|
|
MOZ_ASSERT(protoBinding);
|
|
}
|
|
|
|
nsXBLProtoImplField* field = protoBinding->FindField(fieldName);
|
|
MOZ_ASSERT(field);
|
|
|
|
// This mirrors code in nsXBLProtoImpl::InstallImplementation
|
|
nsCOMPtr<nsIScriptGlobalObject> global =
|
|
do_QueryInterface(xblNode->OwnerDoc()->GetWindow());
|
|
if (!global) {
|
|
return true;
|
|
}
|
|
|
|
nsCOMPtr<nsIScriptContext> context = global->GetContext();
|
|
if (!context) {
|
|
return true;
|
|
}
|
|
|
|
nsresult rv = field->InstallField(context, thisObj, protoBinding->DocURI(),
|
|
installed);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
return true;
|
|
}
|
|
|
|
if (!::JS_IsExceptionPending(cx)) {
|
|
xpc::Throw(cx, rv);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
FieldGetterImpl(JSContext *cx, JS::CallArgs args)
|
|
{
|
|
JS::Handle<JS::Value> thisv = args.thisv();
|
|
MOZ_ASSERT(ValueHasISupportsPrivate(thisv));
|
|
|
|
JS::Rooted<JSObject*> thisObj(cx, &thisv.toObject());
|
|
|
|
// We should be in the compartment of |this|. If we got here via nativeCall,
|
|
// |this| is not same-compartment with |callee|, and it's possible via
|
|
// asymmetric security semantics that |args.calleev()| is actually a security
|
|
// wrapper. In this case, we know we want to do an unsafe unwrap, and
|
|
// InstallXBLField knows how to handle cross-compartment pointers.
|
|
bool installed = false;
|
|
JS::Rooted<JSObject*> callee(cx, js::UncheckedUnwrap(&args.calleev().toObject()));
|
|
JS::Rooted<jsid> id(cx);
|
|
if (!InstallXBLField(cx, callee, thisObj, &id, &installed)) {
|
|
return false;
|
|
}
|
|
|
|
if (!installed) {
|
|
args.rval().setUndefined();
|
|
return true;
|
|
}
|
|
|
|
JS::Rooted<JS::Value> v(cx);
|
|
if (!JS_GetPropertyById(cx, thisObj, id, &v)) {
|
|
return false;
|
|
}
|
|
args.rval().set(v);
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
FieldGetter(JSContext *cx, unsigned argc, JS::Value *vp)
|
|
{
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
return JS::CallNonGenericMethod<ValueHasISupportsPrivate, FieldGetterImpl>
|
|
(cx, args);
|
|
}
|
|
|
|
bool
|
|
FieldSetterImpl(JSContext *cx, JS::CallArgs args)
|
|
{
|
|
JS::Handle<JS::Value> thisv = args.thisv();
|
|
MOZ_ASSERT(ValueHasISupportsPrivate(thisv));
|
|
|
|
JS::Rooted<JSObject*> thisObj(cx, &thisv.toObject());
|
|
|
|
// We should be in the compartment of |this|. If we got here via nativeCall,
|
|
// |this| is not same-compartment with |callee|, and it's possible via
|
|
// asymmetric security semantics that |args.calleev()| is actually a security
|
|
// wrapper. In this case, we know we want to do an unsafe unwrap, and
|
|
// InstallXBLField knows how to handle cross-compartment pointers.
|
|
bool installed = false;
|
|
JS::Rooted<JSObject*> callee(cx, js::UncheckedUnwrap(&args.calleev().toObject()));
|
|
JS::Rooted<jsid> id(cx);
|
|
if (!InstallXBLField(cx, callee, thisObj, &id, &installed)) {
|
|
return false;
|
|
}
|
|
|
|
if (installed) {
|
|
if (!::JS_SetPropertyById(cx, thisObj, id, args.get(0))) {
|
|
return false;
|
|
}
|
|
}
|
|
args.rval().setUndefined();
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
FieldSetter(JSContext *cx, unsigned argc, JS::Value *vp)
|
|
{
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
return JS::CallNonGenericMethod<ValueHasISupportsPrivate, FieldSetterImpl>
|
|
(cx, args);
|
|
}
|
|
|
|
nsresult
|
|
nsXBLProtoImplField::InstallAccessors(JSContext* aCx,
|
|
JS::Handle<JSObject*> aTargetClassObject)
|
|
{
|
|
MOZ_ASSERT(js::IsObjectInContextCompartment(aTargetClassObject, aCx));
|
|
JS::Rooted<JSObject*> globalObject(aCx, JS_GetGlobalForObject(aCx, aTargetClassObject));
|
|
JS::Rooted<JSObject*> scopeObject(aCx, xpc::GetXBLScope(aCx, globalObject));
|
|
NS_ENSURE_TRUE(scopeObject, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
// Don't install it if the field is empty; see also InstallField which also must
|
|
// implement the not-empty requirement.
|
|
if (IsEmpty()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Install a getter/setter pair which will resolve the field onto the actual
|
|
// object, when invoked.
|
|
|
|
// Get the field name as an id.
|
|
JS::Rooted<jsid> id(aCx);
|
|
JS::TwoByteChars chars(mName, NS_strlen(mName));
|
|
if (!JS_CharsToId(aCx, chars, &id))
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
// Properties/Methods have historically taken precendence over fields. We
|
|
// install members first, so just bounce here if the property is already
|
|
// defined.
|
|
bool found = false;
|
|
if (!JS_AlreadyHasOwnPropertyById(aCx, aTargetClassObject, id, &found))
|
|
return NS_ERROR_FAILURE;
|
|
if (found)
|
|
return NS_OK;
|
|
|
|
// FieldGetter and FieldSetter need to run in the XBL scope so that they can
|
|
// see through any SOWs on their targets.
|
|
|
|
// First, enter the XBL scope, and compile the functions there.
|
|
JSAutoCompartment ac(aCx, scopeObject);
|
|
JS::Rooted<JS::Value> wrappedClassObj(aCx, JS::ObjectValue(*aTargetClassObject));
|
|
if (!JS_WrapValue(aCx, &wrappedClassObj) || !JS_WrapId(aCx, id.address()))
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
JS::Rooted<JSObject*> get(aCx,
|
|
JS_GetFunctionObject(js::NewFunctionByIdWithReserved(aCx, FieldGetter,
|
|
0, 0, scopeObject, id)));
|
|
if (!get) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
js::SetFunctionNativeReserved(get, XBLPROTO_SLOT, wrappedClassObj);
|
|
js::SetFunctionNativeReserved(get, FIELD_SLOT,
|
|
JS::StringValue(JSID_TO_STRING(id)));
|
|
|
|
JS::Rooted<JSObject*> set(aCx,
|
|
JS_GetFunctionObject(js::NewFunctionByIdWithReserved(aCx, FieldSetter,
|
|
1, 0, scopeObject, id)));
|
|
if (!set) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
js::SetFunctionNativeReserved(set, XBLPROTO_SLOT, wrappedClassObj);
|
|
js::SetFunctionNativeReserved(set, FIELD_SLOT,
|
|
JS::StringValue(JSID_TO_STRING(id)));
|
|
|
|
// Now, re-enter the class object's scope, wrap the getters/setters, and define
|
|
// them there.
|
|
JSAutoCompartment ac2(aCx, aTargetClassObject);
|
|
if (!JS_WrapObject(aCx, &get) || !JS_WrapObject(aCx, &set) ||
|
|
!JS_WrapId(aCx, id.address()))
|
|
{
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
if (!::JS_DefinePropertyById(aCx, aTargetClassObject, id, JS::UndefinedValue(),
|
|
JS_DATA_TO_FUNC_PTR(JSPropertyOp, get.get()),
|
|
JS_DATA_TO_FUNC_PTR(JSStrictPropertyOp, set.get()),
|
|
AccessorAttributes())) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsXBLProtoImplField::InstallField(nsIScriptContext* aContext,
|
|
JS::Handle<JSObject*> aBoundNode,
|
|
nsIURI* aBindingDocURI,
|
|
bool* aDidInstall) const
|
|
{
|
|
NS_PRECONDITION(aBoundNode,
|
|
"uh-oh, bound node should NOT be null or bad things will "
|
|
"happen");
|
|
|
|
*aDidInstall = false;
|
|
|
|
// Empty fields are treated as not actually present.
|
|
if (IsEmpty()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAutoMicroTask mt;
|
|
|
|
// EvaluateString and JS_DefineUCProperty can both trigger GC, so
|
|
// protect |result| here.
|
|
nsresult rv;
|
|
|
|
nsAutoCString uriSpec;
|
|
aBindingDocURI->GetSpec(uriSpec);
|
|
|
|
AutoPushJSContext cx(aContext->GetNativeContext());
|
|
NS_ASSERTION(!::JS_IsExceptionPending(cx),
|
|
"Shouldn't get here when an exception is pending!");
|
|
|
|
// First, enter the xbl scope, wrap the node, and use that as the scope for
|
|
// the evaluation.
|
|
JS::Rooted<JSObject*> scopeObject(cx, xpc::GetXBLScope(cx, aBoundNode));
|
|
NS_ENSURE_TRUE(scopeObject, NS_ERROR_OUT_OF_MEMORY);
|
|
JSAutoCompartment ac(cx, scopeObject);
|
|
|
|
JS::Rooted<JSObject*> wrappedNode(cx, aBoundNode);
|
|
if (!JS_WrapObject(cx, &wrappedNode))
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
JS::Rooted<JS::Value> result(cx);
|
|
JS::CompileOptions options(cx);
|
|
options.setFileAndLine(uriSpec.get(), mLineNumber)
|
|
.setVersion(JSVERSION_LATEST);
|
|
rv = aContext->EvaluateString(nsDependentString(mFieldText,
|
|
mFieldTextLength),
|
|
wrappedNode, options,
|
|
/* aCoerceToString = */ false,
|
|
result.address());
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
|
|
// Now, enter the node's compartment, wrap the eval result, and define it on
|
|
// the bound node.
|
|
JSAutoCompartment ac2(cx, aBoundNode);
|
|
nsDependentString name(mName);
|
|
if (!JS_WrapValue(cx, &result) ||
|
|
!::JS_DefineUCProperty(cx, aBoundNode,
|
|
reinterpret_cast<const jschar*>(mName),
|
|
name.Length(), result, nullptr, nullptr,
|
|
mJSAttributes)) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
*aDidInstall = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsXBLProtoImplField::Read(nsIObjectInputStream* aStream)
|
|
{
|
|
nsAutoString name;
|
|
nsresult rv = aStream->ReadString(name);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
mName = ToNewUnicode(name);
|
|
|
|
rv = aStream->Read32(&mLineNumber);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
nsAutoString fieldText;
|
|
rv = aStream->ReadString(fieldText);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
mFieldTextLength = fieldText.Length();
|
|
if (mFieldTextLength)
|
|
mFieldText = ToNewUnicode(fieldText);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsXBLProtoImplField::Write(nsIObjectOutputStream* aStream)
|
|
{
|
|
XBLBindingSerializeDetails type = XBLBinding_Serialize_Field;
|
|
|
|
if (mJSAttributes & JSPROP_READONLY) {
|
|
type |= XBLBinding_Serialize_ReadOnly;
|
|
}
|
|
|
|
nsresult rv = aStream->Write8(type);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
rv = aStream->WriteWStringZ(mName);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
rv = aStream->Write32(mLineNumber);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
return aStream->WriteWStringZ(mFieldText ? mFieldText : MOZ_UTF16(""));
|
|
}
|