/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- * vim: set ts=4 sw=4 et tw=80: * * ***** 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 SpiderMonkey E4X code, released August, 2004. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either of 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 "jsstddef.h" #include "jsconfig.h" #if JS_HAS_XML_SUPPORT #include #include #include #include "jstypes.h" #include "jsbit.h" #include "jsprf.h" #include "jsutil.h" #include "jsapi.h" #include "jsarray.h" #include "jsatom.h" #include "jsbool.h" #include "jscntxt.h" #include "jsfun.h" #include "jsgc.h" #include "jsinterp.h" #include "jslock.h" #include "jsnum.h" #include "jsobj.h" #include "jsopcode.h" #include "jsparse.h" #include "jsscan.h" #include "jsscope.h" #include "jsscript.h" #include "jsstr.h" #include "jsxml.h" #ifdef DEBUG #include /* for #ifdef DEBUG memset calls */ #endif /* * NOTES * - in the js shell, you must use the -x command line option, or call * options('xml') before compiling anything that uses XML literals * * TODO * - XXXbe patrol * - Fuse objects and their JSXML* private data into single GC-things * - fix function::foo vs. x.(foo == 42) collision using proper namespacing * - fix the !TCF_HAS_DEFXMLNS optimization in js_FoldConstants * - JSCLASS_DOCUMENT_OBSERVER support -- live two-way binding to Gecko's DOM! * - JS_TypeOfValue sure could use a cleaner interface to "types" */ #ifdef DEBUG_brendan #define METERING 1 #endif #ifdef METERING static struct { jsrefcount qname; jsrefcount qnameobj; jsrefcount liveqname; jsrefcount liveqnameobj; jsrefcount namespace; jsrefcount namespaceobj; jsrefcount livenamespace; jsrefcount livenamespaceobj; jsrefcount xml; jsrefcount xmlobj; jsrefcount livexml; jsrefcount livexmlobj; } xml_stats; #define METER(x) JS_ATOMIC_INCREMENT(&(x)) #define UNMETER(x) JS_ATOMIC_DECREMENT(&(x)) #else #define METER(x) /* nothing */ #define UNMETER(x) /* nothing */ #endif /* * Random utilities and global functions. */ const char js_AnyName_str[] = "AnyName"; const char js_AttributeName_str[] = "AttributeName"; const char js_isXMLName_str[] = "isXMLName"; const char js_XMLList_str[] = "XMLList"; const char js_localName_str[] = "localName"; const char js_xml_parent_str[] = "parent"; const char js_prefix_str[] = "prefix"; const char js_toXMLString_str[] = "toXMLString"; const char js_uri_str[] = "uri"; const char js_amp_entity_str[] = "&"; const char js_gt_entity_str[] = ">"; const char js_lt_entity_str[] = "<"; const char js_quot_entity_str[] = """; #define IS_EMPTY(str) (JSSTRING_LENGTH(str) == 0) #define IS_STAR(str) (JSSTRING_LENGTH(str) == 1 && *JSSTRING_CHARS(str) == '*') static JSBool xml_isXMLName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { *rval = BOOLEAN_TO_JSVAL(js_IsXMLName(cx, argv[0])); return JS_TRUE; } /* * Namespace class and library functions. */ enum namespace_tinyid { NAMESPACE_PREFIX = -1, NAMESPACE_URI = -2 }; static JSBool namespace_getProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSXMLNamespace *ns; if (!JSVAL_IS_INT(id)) return JS_TRUE; ns = (JSXMLNamespace *) JS_GetInstancePrivate(cx, obj, &js_NamespaceClass.base, NULL); if (!ns) return JS_TRUE; switch (JSVAL_TO_INT(id)) { case NAMESPACE_PREFIX: *vp = ns->prefix ? STRING_TO_JSVAL(ns->prefix) : JSVAL_VOID; break; case NAMESPACE_URI: *vp = STRING_TO_JSVAL(ns->uri); break; } return JS_TRUE; } static void namespace_finalize(JSContext *cx, JSObject *obj) { JSXMLNamespace *ns; JSRuntime *rt; ns = (JSXMLNamespace *) JS_GetPrivate(cx, obj); if (!ns) return; JS_ASSERT(ns->object == obj); ns->object = NULL; UNMETER(xml_stats.livenamespaceobj); rt = cx->runtime; if (rt->functionNamespaceObject == obj) rt->functionNamespaceObject = NULL; } static void namespace_mark_vector(JSContext *cx, JSXMLNamespace **vec, uint32 len, void *arg) { uint32 i; JSXMLNamespace *ns; for (i = 0; i < len; i++) { ns = vec[i]; { #ifdef GC_MARK_DEBUG char buf[100]; JS_snprintf(buf, sizeof buf, "%s=%s", ns->prefix ? JS_GetStringBytes(ns->prefix) : "", JS_GetStringBytes(ns->uri)); #else const char *buf = NULL; #endif JS_MarkGCThing(cx, ns, buf, arg); } } } static uint32 namespace_mark(JSContext *cx, JSObject *obj, void *arg) { JSXMLNamespace *ns; ns = (JSXMLNamespace *) JS_GetPrivate(cx, obj); JS_MarkGCThing(cx, ns, js_private_str, arg); return 0; } static JSBool namespace_equality(JSContext *cx, JSObject *obj, jsval v, JSBool *bp) { JSXMLNamespace *ns, *ns2; JSObject *obj2; ns = (JSXMLNamespace *) JS_GetPrivate(cx, obj); JS_ASSERT(JSVAL_IS_OBJECT(v)); obj2 = JSVAL_TO_OBJECT(v); if (!obj2 || OBJ_GET_CLASS(cx, obj2) != &js_NamespaceClass.base) { *bp = JS_FALSE; } else { ns2 = (JSXMLNamespace *) JS_GetPrivate(cx, obj2); *bp = !js_CompareStrings(ns->uri, ns2->uri); } return JS_TRUE; } JS_FRIEND_DATA(JSExtendedClass) js_NamespaceClass = { { "Namespace", JSCLASS_HAS_PRIVATE | JSCLASS_CONSTRUCT_PROTOTYPE | JSCLASS_IS_EXTENDED, JS_PropertyStub, JS_PropertyStub, namespace_getProperty, NULL, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, namespace_finalize, NULL, NULL, NULL, NULL, NULL, NULL, namespace_mark, NULL }, namespace_equality, NULL, NULL, JSCLASS_NO_RESERVED_MEMBERS }; #define NAMESPACE_ATTRS \ (JSPROP_ENUMERATE | JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_SHARED) static JSPropertySpec namespace_props[] = { {js_prefix_str, NAMESPACE_PREFIX, NAMESPACE_ATTRS, 0, 0}, {js_uri_str, NAMESPACE_URI, NAMESPACE_ATTRS, 0, 0}, {0,0,0,0,0} }; static JSBool namespace_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXMLNamespace *ns; ns = (JSXMLNamespace *) JS_GetInstancePrivate(cx, obj, &js_NamespaceClass.base, argv); if (!ns) return JS_FALSE; *rval = STRING_TO_JSVAL(ns->uri); return JS_TRUE; } static JSFunctionSpec namespace_methods[] = { {js_toString_str, namespace_toString, 0,0,0}, {0,0,0,0,0} }; JSXMLNamespace * js_NewXMLNamespace(JSContext *cx, JSString *prefix, JSString *uri, JSBool declared) { JSXMLNamespace *ns; ns = (JSXMLNamespace *) js_NewGCThing(cx, GCX_NAMESPACE, sizeof(JSXMLNamespace)); if (!ns) return NULL; ns->object = NULL; ns->prefix = prefix; ns->uri = uri; ns->declared = declared; METER(xml_stats.namespace); METER(xml_stats.livenamespace); return ns; } void js_MarkXMLNamespace(JSContext *cx, JSXMLNamespace *ns, void *arg) { JS_MarkGCThing(cx, ns->object, js_object_str, arg); JS_MarkGCThing(cx, ns->prefix, js_prefix_str, arg); JS_MarkGCThing(cx, ns->uri, js_uri_str, arg); } void js_FinalizeXMLNamespace(JSContext *cx, JSXMLNamespace *ns) { UNMETER(xml_stats.livenamespace); } JSObject * js_NewXMLNamespaceObject(JSContext *cx, JSString *prefix, JSString *uri, JSBool declared) { JSXMLNamespace *ns; ns = js_NewXMLNamespace(cx, prefix, uri, declared); if (!ns) return NULL; return js_GetXMLNamespaceObject(cx, ns); } JSObject * js_GetXMLNamespaceObject(JSContext *cx, JSXMLNamespace *ns) { JSObject *obj; obj = ns->object; if (obj) { JS_ASSERT(JS_GetPrivate(cx, obj) == ns); return obj; } obj = js_NewObject(cx, &js_NamespaceClass.base, NULL, NULL); if (!obj || !JS_SetPrivate(cx, obj, ns)) { cx->newborn[GCX_OBJECT] = NULL; return NULL; } ns->object = obj; METER(xml_stats.namespaceobj); METER(xml_stats.livenamespaceobj); return obj; } /* * QName class and library functions. */ enum qname_tinyid { QNAME_URI = -1, QNAME_LOCALNAME = -2 }; static JSBool qname_getProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSXMLQName *qn; if (!JSVAL_IS_INT(id)) return JS_TRUE; qn = (JSXMLQName *) JS_GetInstancePrivate(cx, obj, &js_QNameClass.base, NULL); if (!qn) return JS_TRUE; switch (JSVAL_TO_INT(id)) { case QNAME_URI: *vp = qn->uri ? STRING_TO_JSVAL(qn->uri) : JSVAL_NULL; break; case QNAME_LOCALNAME: *vp = STRING_TO_JSVAL(qn->localName); break; } return JS_TRUE; } static void qname_finalize(JSContext *cx, JSObject *obj) { JSXMLQName *qn; qn = (JSXMLQName *) JS_GetPrivate(cx, obj); if (!qn) return; JS_ASSERT(qn->object == obj); qn->object = NULL; UNMETER(xml_stats.liveqnameobj); } static void anyname_finalize(JSContext* cx, JSObject* obj) { JSRuntime *rt; /* Make sure the next call to js_GetAnyName doesn't try to use obj. */ rt = cx->runtime; if (rt->anynameObject == obj) rt->anynameObject = NULL; qname_finalize(cx, obj); } static uint32 qname_mark(JSContext *cx, JSObject *obj, void *arg) { JSXMLQName *qn; qn = (JSXMLQName *) JS_GetPrivate(cx, obj); JS_MarkGCThing(cx, qn, js_private_str, arg); return 0; } static JSBool qname_identity(JSXMLQName *qna, JSXMLQName *qnb) { if (!qna->uri ^ !qnb->uri) return JS_FALSE; if (qna->uri && js_CompareStrings(qna->uri, qnb->uri)) return JS_FALSE; return !js_CompareStrings(qna->localName, qnb->localName); } static JSBool qname_equality(JSContext *cx, JSObject *obj, jsval v, JSBool *bp) { JSXMLQName *qn, *qn2; JSObject *obj2; qn = (JSXMLQName *) JS_GetPrivate(cx, obj); JS_ASSERT(JSVAL_IS_OBJECT(v)); obj2 = JSVAL_TO_OBJECT(v); if (!obj2 || OBJ_GET_CLASS(cx, obj2) != &js_QNameClass.base) { *bp = JS_FALSE; } else { qn2 = (JSXMLQName *) JS_GetPrivate(cx, obj2); *bp = qname_identity(qn, qn2); } return JS_TRUE; } JS_FRIEND_DATA(JSExtendedClass) js_QNameClass = { { "QName", JSCLASS_HAS_PRIVATE | JSCLASS_CONSTRUCT_PROTOTYPE | JSCLASS_IS_EXTENDED, JS_PropertyStub, JS_PropertyStub, qname_getProperty, NULL, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, qname_finalize, NULL, NULL, NULL, NULL, NULL, NULL, qname_mark, NULL }, qname_equality, NULL, NULL, JSCLASS_NO_RESERVED_MEMBERS }; /* * Classes for the ECMA-357-internal types AttributeName and AnyName, which * are like QName, except that they have no property getters. They share the * qname_toString method, and therefore are exposed as constructable objects * in this implementation. */ JS_FRIEND_DATA(JSClass) js_AttributeNameClass = { js_AttributeName_str, JSCLASS_HAS_PRIVATE | JSCLASS_CONSTRUCT_PROTOTYPE, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, qname_finalize, NULL, NULL, NULL, NULL, NULL, NULL, qname_mark, NULL }; JS_FRIEND_DATA(JSClass) js_AnyNameClass = { js_AnyName_str, JSCLASS_HAS_PRIVATE | JSCLASS_CONSTRUCT_PROTOTYPE, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, anyname_finalize, NULL, NULL, NULL, NULL, NULL, NULL, qname_mark, NULL }; #define QNAME_ATTRS \ (JSPROP_ENUMERATE | JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_SHARED) static JSPropertySpec qname_props[] = { {js_uri_str, QNAME_URI, QNAME_ATTRS, 0, 0}, {js_localName_str, QNAME_LOCALNAME, QNAME_ATTRS, 0, 0}, {0,0,0,0,0} }; static JSBool qname_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSClass *clasp; JSXMLQName *qn; JSString *str, *qualstr; size_t length; jschar *chars; clasp = OBJ_GET_CLASS(cx, obj); if (clasp == &js_AttributeNameClass || clasp == &js_AnyNameClass) { qn = (JSXMLQName *) JS_GetPrivate(cx, obj); } else { qn = (JSXMLQName *) JS_GetInstancePrivate(cx, obj, &js_QNameClass.base, argv); if (!qn) return JS_FALSE; } if (!qn->uri) { /* No uri means wildcard qualifier. */ str = ATOM_TO_STRING(cx->runtime->atomState.starQualifierAtom); } else if (IS_EMPTY(qn->uri)) { /* Empty string for uri means localName is in no namespace. */ str = cx->runtime->emptyString; } else { qualstr = ATOM_TO_STRING(cx->runtime->atomState.qualifierAtom); str = js_ConcatStrings(cx, qn->uri, qualstr); if (!str) return JS_FALSE; } str = js_ConcatStrings(cx, str, qn->localName); if (!str) return JS_FALSE; if (str && clasp == &js_AttributeNameClass) { length = JSSTRING_LENGTH(str); chars = (jschar *) JS_malloc(cx, (length + 2) * sizeof(jschar)); if (!chars) return JS_FALSE; *chars = '@'; js_strncpy(chars + 1, JSSTRING_CHARS(str), length); chars[++length] = 0; str = js_NewString(cx, chars, length, 0); if (!str) { JS_free(cx, chars); return JS_FALSE; } } *rval = STRING_TO_JSVAL(str); return JS_TRUE; } static JSFunctionSpec qname_methods[] = { {js_toString_str, qname_toString, 0,0,0}, {0,0,0,0,0} }; JSXMLQName * js_NewXMLQName(JSContext *cx, JSString *uri, JSString *prefix, JSString *localName) { JSXMLQName *qn; qn = (JSXMLQName *) js_NewGCThing(cx, GCX_QNAME, sizeof(JSXMLQName)); if (!qn) return NULL; qn->object = NULL; qn->uri = uri; qn->prefix = prefix; qn->localName = localName; METER(xml_stats.qname); METER(xml_stats.liveqname); return qn; } void js_MarkXMLQName(JSContext *cx, JSXMLQName *qn, void *arg) { JS_MarkGCThing(cx, qn->object, js_object_str, arg); JS_MarkGCThing(cx, qn->uri, js_uri_str, arg); JS_MarkGCThing(cx, qn->prefix, js_prefix_str, arg); JS_MarkGCThing(cx, qn->localName, js_localName_str, arg); } void js_FinalizeXMLQName(JSContext *cx, JSXMLQName *qn) { UNMETER(xml_stats.liveqname); } JSObject * js_NewXMLQNameObject(JSContext *cx, JSString *uri, JSString *prefix, JSString *localName) { JSXMLQName *qn; qn = js_NewXMLQName(cx, uri, prefix, localName); if (!qn) return NULL; return js_GetXMLQNameObject(cx, qn); } JSObject * js_GetXMLQNameObject(JSContext *cx, JSXMLQName *qn) { JSObject *obj; obj = qn->object; if (obj) { JS_ASSERT(JS_GetPrivate(cx, obj) == qn); return obj; } obj = js_NewObject(cx, &js_QNameClass.base, NULL, NULL); if (!obj || !JS_SetPrivate(cx, obj, qn)) { cx->newborn[GCX_OBJECT] = NULL; return NULL; } qn->object = obj; METER(xml_stats.qnameobj); METER(xml_stats.liveqnameobj); return obj; } JSObject * js_GetAttributeNameObject(JSContext *cx, JSXMLQName *qn) { JSObject *obj; obj = qn->object; if (obj) { if (OBJ_GET_CLASS(cx, obj) == &js_AttributeNameClass) return obj; qn = js_NewXMLQName(cx, qn->uri, qn->prefix, qn->localName); if (!qn) return NULL; } obj = js_NewObject(cx, &js_AttributeNameClass, NULL, NULL); if (!obj || !JS_SetPrivate(cx, obj, qn)) { cx->newborn[GCX_OBJECT] = NULL; return NULL; } qn->object = obj; METER(xml_stats.qnameobj); METER(xml_stats.liveqnameobj); return obj; } JSObject * js_ConstructXMLQNameObject(JSContext *cx, jsval nsval, jsval lnval) { jsval argv[2]; /* * ECMA-357 11.1.2, * The _QualifiedIdentifier : PropertySelector :: PropertySelector_ * production, step 2. */ if (!JSVAL_IS_PRIMITIVE(nsval) && OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(nsval)) == &js_AnyNameClass) { nsval = JSVAL_NULL; } argv[0] = nsval; argv[1] = lnval; return js_ConstructObject(cx, &js_QNameClass.base, NULL, NULL, 2, argv); } static JSBool IsXMLName(const jschar *cp, size_t n) { JSBool rv; jschar c; rv = JS_FALSE; if (n != 0 && JS_ISXMLNSSTART(*cp)) { while (--n != 0) { c = *++cp; if (!JS_ISXMLNS(c)) return rv; } rv = JS_TRUE; } return rv; } JSBool js_IsXMLName(JSContext *cx, jsval v) { JSClass *clasp; JSXMLQName *qn; JSString *name; JSErrorReporter older; /* * Inline specialization of the QName constructor called with v passed as * the only argument, to compute the localName for the constructed qname, * without actually allocating the object or computing its uri and prefix. * See ECMA-357 13.1.2.1 step 1 and 13.3.2. */ if (!JSVAL_IS_PRIMITIVE(v) && (clasp = OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(v)), clasp == &js_QNameClass.base || clasp == &js_AttributeNameClass || clasp == &js_AnyNameClass)) { qn = (JSXMLQName *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); name = qn->localName; } else { older = JS_SetErrorReporter(cx, NULL); name = js_ValueToString(cx, v); JS_SetErrorReporter(cx, older); if (!name) { JS_ClearPendingException(cx); return JS_FALSE; } } return IsXMLName(JSSTRING_CHARS(name), JSSTRING_LENGTH(name)); } static JSBool Namespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval urival, prefixval; JSObject *uriobj; JSBool isNamespace, isQName; JSClass *clasp; JSString *empty, *prefix; JSXMLNamespace *ns, *ns2; JSXMLQName *qn; urival = argv[argc > 1]; isNamespace = isQName = JS_FALSE; if (!JSVAL_IS_PRIMITIVE(urival)) { uriobj = JSVAL_TO_OBJECT(urival); clasp = OBJ_GET_CLASS(cx, uriobj); isNamespace = (clasp == &js_NamespaceClass.base); isQName = (clasp == &js_QNameClass.base); } #ifdef __GNUC__ /* suppress bogus gcc warnings */ else uriobj = NULL; #endif if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) { /* Namespace called as function. */ if (argc == 1 && isNamespace) { /* Namespace called with one Namespace argument is identity. */ *rval = urival; return JS_TRUE; } /* Create and return a new QName object exactly as if constructed. */ obj = js_NewObject(cx, &js_NamespaceClass.base, NULL, NULL); if (!obj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); } METER(xml_stats.namespaceobj); METER(xml_stats.livenamespaceobj); /* * Create and connect private data to rooted obj early, so we don't have * to worry about rooting string newborns hanging off of the private data * further below. */ empty = cx->runtime->emptyString; ns = js_NewXMLNamespace(cx, empty, empty, JS_FALSE); if (!ns) return JS_FALSE; if (!JS_SetPrivate(cx, obj, ns)) return JS_FALSE; ns->object = obj; if (argc == 1) { if (isNamespace) { ns2 = (JSXMLNamespace *) JS_GetPrivate(cx, uriobj); ns->uri = ns2->uri; ns->prefix = ns2->prefix; } else if (isQName && (qn = (JSXMLQName *) JS_GetPrivate(cx, uriobj))->uri) { ns->uri = qn->uri; ns->prefix = qn->prefix; } else { ns->uri = js_ValueToString(cx, urival); if (!ns->uri) return JS_FALSE; /* NULL here represents *undefined* in ECMA-357 13.2.2 3(c)iii. */ if (!IS_EMPTY(ns->uri)) ns->prefix = NULL; } } else if (argc == 2) { if (isQName && (qn = (JSXMLQName *) JS_GetPrivate(cx, uriobj))->uri) { ns->uri = qn->uri; } else { ns->uri = js_ValueToString(cx, urival); if (!ns->uri) return JS_FALSE; } prefixval = argv[0]; if (IS_EMPTY(ns->uri)) { if (!JSVAL_IS_VOID(prefixval)) { prefix = js_ValueToString(cx, prefixval); if (!prefix) return JS_FALSE; if (!IS_EMPTY(prefix)) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_NAMESPACE, js_ValueToPrintableString(cx, STRING_TO_JSVAL(prefix))); return JS_FALSE; } } } else if (JSVAL_IS_VOID(prefixval) || !js_IsXMLName(cx, prefixval)) { /* NULL here represents *undefined* in ECMA-357 13.2.2 4(d) etc. */ ns->prefix = NULL; } else { prefix = js_ValueToString(cx, prefixval); if (!prefix) return JS_FALSE; ns->prefix = prefix; } } return JS_TRUE; } static JSBool QName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval nameval, nsval; JSBool isQName, isNamespace; JSXMLQName *qn; JSString *uri, *prefix, *name; JSObject *nsobj; JSClass *clasp; JSXMLNamespace *ns; nameval = argv[argc > 1]; isQName = !JSVAL_IS_PRIMITIVE(nameval) && OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(nameval)) == &js_QNameClass.base; if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) { /* QName called as function. */ if (argc == 1 && isQName) { /* QName called with one QName argument is identity. */ *rval = nameval; return JS_TRUE; } /* * Create and return a new QName object exactly as if constructed. * Use the constructor's clasp so we can be shared by AttributeName * (see below after this function). */ obj = js_NewObject(cx, argv ? JS_ValueToFunction(cx, argv[-2])->clasp : &js_QNameClass.base, NULL, NULL); if (!obj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); } METER(xml_stats.qnameobj); METER(xml_stats.liveqnameobj); if (isQName) { /* If namespace is not specified and name is a QName, clone it. */ qn = (JSXMLQName *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(nameval)); if (argc == 1) { uri = qn->uri; prefix = qn->prefix; name = qn->localName; goto out; } /* Namespace and qname were passed -- use the qname's localName. */ nameval = STRING_TO_JSVAL(qn->localName); } if (argc == 0) { name = cx->runtime->emptyString; } else { name = js_ValueToString(cx, nameval); if (!name) return JS_FALSE; /* Use argv[1] as a local root for name, even if it was not passed. */ argv[1] = STRING_TO_JSVAL(name); } nsval = argv[0]; if (argc == 1 || JSVAL_IS_VOID(nsval)) { if (IS_STAR(name)) { nsval = JSVAL_NULL; } else { if (!js_GetDefaultXMLNamespace(cx, &nsval)) return JS_FALSE; } } if (JSVAL_IS_NULL(nsval)) { /* NULL prefix represents *undefined* in ECMA-357 13.3.2 5(a). */ uri = prefix = NULL; } else { /* * Inline specialization of the Namespace constructor called with * nsval passed as the only argument, to compute the uri and prefix * for the constructed namespace, without actually allocating the * object or computing other members. See ECMA-357 13.3.2 6(a) and * 13.2.2. */ isNamespace = isQName = JS_FALSE; if (!JSVAL_IS_PRIMITIVE(nsval)) { nsobj = JSVAL_TO_OBJECT(nsval); clasp = OBJ_GET_CLASS(cx, nsobj); isNamespace = (clasp == &js_NamespaceClass.base); isQName = (clasp == &js_QNameClass.base); } #ifdef __GNUC__ /* suppress bogus gcc warnings */ else nsobj = NULL; #endif if (isNamespace) { ns = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); uri = ns->uri; prefix = ns->prefix; } else if (isQName && (qn = (JSXMLQName *) JS_GetPrivate(cx, nsobj))->uri) { uri = qn->uri; prefix = qn->prefix; } else { uri = js_ValueToString(cx, nsval); if (!uri) return JS_FALSE; argv[0] = STRING_TO_JSVAL(uri); /* local root */ /* NULL here represents *undefined* in ECMA-357 13.2.2 3(c)iii. */ prefix = IS_EMPTY(uri) ? cx->runtime->emptyString : NULL; } } out: qn = js_NewXMLQName(cx, uri, prefix, name); if (!qn) return JS_FALSE; if (!JS_SetPrivate(cx, obj, qn)) return JS_FALSE; qn->object = obj; return JS_TRUE; } static JSBool AttributeName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { /* * Since js_AttributeNameClass was initialized, obj will have that as its * class, not js_QNameClass. */ return QName(cx, obj, argc, argv, rval); } /* * XMLArray library functions. */ static JSBool namespace_identity(const void *a, const void *b) { const JSXMLNamespace *nsa = (const JSXMLNamespace *) a; const JSXMLNamespace *nsb = (const JSXMLNamespace *) b; if (nsa->prefix && nsb->prefix) { if (js_CompareStrings(nsa->prefix, nsb->prefix)) return JS_FALSE; } else { if (nsa->prefix || nsb->prefix) return JS_FALSE; } return !js_CompareStrings(nsa->uri, nsb->uri); } static JSBool attr_identity(const void *a, const void *b) { const JSXML *xmla = (const JSXML *) a; const JSXML *xmlb = (const JSXML *) b; return qname_identity(xmla->name, xmlb->name); } static void XMLArrayCursorInit(JSXMLArrayCursor *cursor, JSXMLArray *array) { JSXMLArrayCursor *next; cursor->array = array; cursor->index = 0; next = cursor->next = array->cursors; if (next) next->prevp = &cursor->next; cursor->prevp = &array->cursors; array->cursors = cursor; } static void XMLArrayCursorFinish(JSXMLArrayCursor *cursor) { JSXMLArrayCursor *next; if (!cursor->array) return; next = cursor->next; if (next) next->prevp = cursor->prevp; *cursor->prevp = next; cursor->array = NULL; } /* NB: called with null cx from the GC, via xml_mark => XMLArrayTrim. */ static JSBool XMLArraySetCapacity(JSContext *cx, JSXMLArray *array, uint32 capacity) { void **vector; if (capacity == 0) { /* We could let realloc(p, 0) free this, but purify gets confused. */ if (array->vector) free(array->vector); vector = NULL; } else { if ((size_t)capacity > ~(size_t)0 / sizeof(void *) || !(vector = (void **) realloc(array->vector, capacity * sizeof(void *)))) { if (cx) JS_ReportOutOfMemory(cx); return JS_FALSE; } } array->capacity = JSXML_PRESET_CAPACITY | capacity; array->vector = vector; return JS_TRUE; } static void XMLArrayTrim(JSXMLArray *array) { if (array->capacity & JSXML_PRESET_CAPACITY) return; if (array->length < array->capacity) XMLArraySetCapacity(NULL, array, array->length); } static JSBool XMLArrayInit(JSContext *cx, JSXMLArray *array, uint32 capacity) { array->length = array->capacity = 0; array->vector = NULL; array->cursors = NULL; return capacity == 0 || XMLArraySetCapacity(cx, array, capacity); } static void XMLArrayFinish(JSContext *cx, JSXMLArray *array) { JSXMLArrayCursor *cursor; JS_free(cx, array->vector); while ((cursor = array->cursors) != NULL) XMLArrayCursorFinish(cursor); #ifdef DEBUG memset(array, 0xd5, sizeof *array); #endif } #define XML_NOT_FOUND ((uint32) -1) static uint32 XMLArrayFindMember(const JSXMLArray *array, void *elt, JSIdentityOp identity) { void **vector; uint32 i, n; /* The identity op must not reallocate array->vector. */ vector = array->vector; if (identity) { for (i = 0, n = array->length; i < n; i++) { if (identity(vector[i], elt)) return i; } } else { for (i = 0, n = array->length; i < n; i++) { if (vector[i] == elt) return i; } } return XML_NOT_FOUND; } /* * Grow array vector capacity by powers of two to LINEAR_THRESHOLD, and after * that, grow by LINEAR_INCREMENT. Both must be powers of two, and threshold * should be greater than increment. */ #define LINEAR_THRESHOLD 256 #define LINEAR_INCREMENT 32 static JSBool XMLArrayAddMember(JSContext *cx, JSXMLArray *array, uint32 index, void *elt) { uint32 capacity, i; int log2; void **vector; if (index >= array->length) { if (index >= JSXML_CAPACITY(array)) { /* Arrange to clear JSXML_PRESET_CAPACITY from array->capacity. */ capacity = index + 1; if (index >= LINEAR_THRESHOLD) { capacity = JS_ROUNDUP(capacity, LINEAR_INCREMENT); } else { JS_CEILING_LOG2(log2, capacity); capacity = JS_BIT(log2); } if ((size_t)capacity > ~(size_t)0 / sizeof(void *) || !(vector = (void **) realloc(array->vector, capacity * sizeof(void *)))) { JS_ReportOutOfMemory(cx); return JS_FALSE; } array->capacity = capacity; array->vector = vector; for (i = array->length; i < index; i++) vector[i] = NULL; } array->length = index + 1; } array->vector[index] = elt; return JS_TRUE; } static JSBool XMLArrayInsert(JSContext *cx, JSXMLArray *array, uint32 i, uint32 n) { uint32 j; JSXMLArrayCursor *cursor; j = array->length; JS_ASSERT(i <= j); if (!XMLArraySetCapacity(cx, array, j + n)) return JS_FALSE; array->length = j + n; JS_ASSERT(n != (uint32)-1); while (j != i) { --j; array->vector[j + n] = array->vector[j]; } for (cursor = array->cursors; cursor; cursor = cursor->next) { if (cursor->index > i) cursor->index += n; } return JS_TRUE; } static void * XMLArrayDelete(JSContext *cx, JSXMLArray *array, uint32 index, JSBool compress) { uint32 length; void **vector, *elt; JSXMLArrayCursor *cursor; length = array->length; if (index >= length) return NULL; vector = array->vector; elt = vector[index]; if (compress) { while (++index < length) vector[index-1] = vector[index]; array->length = length - 1; array->capacity = JSXML_CAPACITY(array); } else { vector[index] = NULL; } for (cursor = array->cursors; cursor; cursor = cursor->next) { if (cursor->index > index) --cursor->index; } return elt; } static void XMLArrayTruncate(JSContext *cx, JSXMLArray *array, uint32 length) { void **vector; JS_ASSERT(!array->cursors); if (length >= array->length) return; if (length == 0) { if (array->vector) free(array->vector); vector = NULL; } else { vector = realloc(array->vector, length * sizeof(void *)); if (!vector) return; } if (array->length > length) array->length = length; array->capacity = length; array->vector = vector; } #define XMLARRAY_FIND_MEMBER(a,e,f) XMLArrayFindMember(a, (void *)(e), f) #define XMLARRAY_HAS_MEMBER(a,e,f) (XMLArrayFindMember(a, (void *)(e), f) != \ XML_NOT_FOUND) #define XMLARRAY_MEMBER(a,i,t) ((t *) (a)->vector[i]) #define XMLARRAY_SET_MEMBER(a,i,e) ((a)->vector[i] = (void *)(e)) #define XMLARRAY_ADD_MEMBER(x,a,i,e)XMLArrayAddMember(x, a, i, (void *)(e)) #define XMLARRAY_INSERT(x,a,i,n) XMLArrayInsert(x, a, i, n) #define XMLARRAY_APPEND(x,a,e) XMLARRAY_ADD_MEMBER(x, a, (a)->length, (e)) #define XMLARRAY_DELETE(x,a,i,c,t) ((t *) XMLArrayDelete(x, a, i, c)) #define XMLARRAY_TRUNCATE(x,a,n) XMLArrayTruncate(x, a, n) /* * Define XML setting property strings and constants early, so everyone can * use the same names and their magic numbers (tinyids, flags). */ static const char js_ignoreComments_str[] = "ignoreComments"; static const char js_ignoreProcessingInstructions_str[] = "ignoreProcessingInstructions"; static const char js_ignoreWhitespace_str[] = "ignoreWhitespace"; static const char js_prettyPrinting_str[] = "prettyPrinting"; static const char js_prettyIndent_str[] = "prettyIndent"; /* * NB: These XML static property tinyids must * (a) not collide with the generic negative tinyids at the top of jsfun.c; * (b) index their corresponding xml_static_props array elements. * Don't change 'em! */ enum xml_static_tinyid { XML_IGNORE_COMMENTS, XML_IGNORE_PROCESSING_INSTRUCTIONS, XML_IGNORE_WHITESPACE, XML_PRETTY_PRINTING, XML_PRETTY_INDENT }; static JSBool xml_setting_setter(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSBool b; uint8 flag; JS_ASSERT(JSVAL_IS_INT(id)); if (!js_ValueToBoolean(cx, *vp, &b)) return JS_FALSE; flag = JS_BIT(JSVAL_TO_INT(id)); if (b) cx->xmlSettingFlags |= flag; else cx->xmlSettingFlags &= ~flag; return JS_TRUE; } static JSPropertySpec xml_static_props[] = { {js_ignoreComments_str, XML_IGNORE_COMMENTS, JSPROP_PERMANENT, NULL, xml_setting_setter}, {js_ignoreProcessingInstructions_str, XML_IGNORE_PROCESSING_INSTRUCTIONS, JSPROP_PERMANENT, NULL, xml_setting_setter}, {js_ignoreWhitespace_str, XML_IGNORE_WHITESPACE, JSPROP_PERMANENT, NULL, xml_setting_setter}, {js_prettyPrinting_str, XML_PRETTY_PRINTING, JSPROP_PERMANENT, NULL, xml_setting_setter}, {js_prettyIndent_str, XML_PRETTY_INDENT, JSPROP_PERMANENT, NULL, NULL}, {0,0,0,0,0} }; /* Derive cx->xmlSettingFlags bits from xml_static_props tinyids. */ #define XSF_IGNORE_COMMENTS JS_BIT(XML_IGNORE_COMMENTS) #define XSF_IGNORE_PROCESSING_INSTRUCTIONS \ JS_BIT(XML_IGNORE_PROCESSING_INSTRUCTIONS) #define XSF_IGNORE_WHITESPACE JS_BIT(XML_IGNORE_WHITESPACE) #define XSF_PRETTY_PRINTING JS_BIT(XML_PRETTY_PRINTING) #define XSF_CACHE_VALID JS_BIT(XML_PRETTY_INDENT) /* * Extra, unrelated but necessarily disjoint flag used by ParseNodeToXML. * This flag means a couple of things: * * - The top JSXML created for a parse tree must have an object owning it. * * - That the default namespace normally inherited from the temporary * tag that wraps a runtime-concatenated XML source * string must, in the case of a precompiled XML object tree, inherit via * ad-hoc code in ParseNodeToXML. * * Because of the second purpose, we name this flag XSF_PRECOMPILED_ROOT. */ #define XSF_PRECOMPILED_ROOT (XSF_CACHE_VALID << 1) /* Macros for special-casing xml:, xmlns= and xmlns:foo= in ParseNodeToQName. */ #define IS_XML(str) \ (JSSTRING_LENGTH(str) == 3 && IS_XML_CHARS(JSSTRING_CHARS(str))) #define IS_XMLNS(str) \ (JSSTRING_LENGTH(str) == 5 && IS_XMLNS_CHARS(JSSTRING_CHARS(str))) #define IS_XML_CHARS(chars) \ (JS_TOLOWER((chars)[0]) == 'x' && \ JS_TOLOWER((chars)[1]) == 'm' && \ JS_TOLOWER((chars)[2]) == 'l') #define HAS_NS_AFTER_XML(chars) \ (JS_TOLOWER((chars)[3]) == 'n' && \ JS_TOLOWER((chars)[4]) == 's') #define IS_XMLNS_CHARS(chars) \ (IS_XML_CHARS(chars) && HAS_NS_AFTER_XML(chars)) #define STARTS_WITH_XML(chars,length) \ (length >= 3 && IS_XML_CHARS(chars)) static const char xml_namespace_str[] = "http://www.w3.org/XML/1998/namespace"; static const char xmlns_namespace_str[] = "http://www.w3.org/2000/xmlns/"; static JSXMLQName * ParseNodeToQName(JSContext *cx, JSParseNode *pn, JSXMLArray *inScopeNSes, JSBool isAttributeName) { JSString *str, *uri, *prefix, *localName; size_t length, offset; const jschar *start, *limit, *colon; uint32 n; JSXMLNamespace *ns; JS_ASSERT(pn->pn_arity == PN_NULLARY); str = ATOM_TO_STRING(pn->pn_atom); length = JSSTRING_LENGTH(str); start = JSSTRING_CHARS(str); JS_ASSERT(length != 0 && *start != '@'); JS_ASSERT(length != 1 || *start != '*'); uri = cx->runtime->emptyString; limit = start + length; colon = js_strchr_limit(start, ':', limit); if (colon) { offset = PTRDIFF(colon, start, jschar); prefix = js_NewDependentString(cx, str, 0, offset, 0); if (!prefix) return NULL; if (STARTS_WITH_XML(start, offset)) { if (offset == 3) { uri = JS_InternString(cx, xml_namespace_str); if (!uri) return NULL; } else if (offset == 5 && HAS_NS_AFTER_XML(start)) { uri = JS_InternString(cx, xmlns_namespace_str); if (!uri) return NULL; } else { uri = NULL; } } else { uri = NULL; n = inScopeNSes->length; while (n != 0) { ns = XMLARRAY_MEMBER(inScopeNSes, --n, JSXMLNamespace); if (ns->prefix && !js_CompareStrings(ns->prefix, prefix)) { uri = ns->uri; break; } } } if (!uri) { js_ReportCompileErrorNumber(cx, pn, JSREPORT_PN | JSREPORT_ERROR, JSMSG_BAD_XML_NAMESPACE, js_ValueToPrintableString(cx, STRING_TO_JSVAL(prefix))); return NULL; } localName = js_NewStringCopyN(cx, colon + 1, length - (offset + 1), 0); if (!localName) return NULL; } else { if (isAttributeName) { /* * An unprefixed attribute is not in any namespace, so set prefix * as well as uri to the empty string. */ prefix = uri; } else { /* * Loop from back to front looking for the closest declared default * namespace. */ n = inScopeNSes->length; while (n != 0) { ns = XMLARRAY_MEMBER(inScopeNSes, --n, JSXMLNamespace); if (!ns->prefix || IS_EMPTY(ns->prefix)) { uri = ns->uri; break; } } prefix = NULL; } localName = str; } return js_NewXMLQName(cx, uri, prefix, localName); } static JSString * ChompXMLWhitespace(JSContext *cx, JSString *str) { size_t length, newlength, offset; const jschar *cp, *start, *end; jschar c; length = JSSTRING_LENGTH(str); for (cp = start = JSSTRING_CHARS(str), end = cp + length; cp < end; cp++) { c = *cp; if (!JS_ISXMLSPACE(c)) break; } while (end > cp) { c = end[-1]; if (!JS_ISXMLSPACE(c)) break; --end; } newlength = PTRDIFF(end, cp, jschar); if (newlength == length) return str; offset = PTRDIFF(cp, start, jschar); return js_NewDependentString(cx, str, offset, newlength, 0); } static JSXML * ParseNodeToXML(JSContext *cx, JSParseNode *pn, JSXMLArray *inScopeNSes, uintN flags) { JSXML *xml, *kid, *attr, *attrj; JSString *str; uint32 length, n, i, j; JSParseNode *pn2, *pn3, *head, **pnp; JSXMLNamespace *ns; JSXMLQName *qn, *attrjqn; JSXMLClass xml_class; #define PN2X_SKIP_CHILD ((JSXML *) 1) /* * Cases return early to avoid common code that gets an outermost xml's * object, which protects GC-things owned by xml and its descendants from * garbage collection. */ xml = NULL; if (!JS_EnterLocalRootScope(cx)) return NULL; switch (pn->pn_type) { case TOK_XMLELEM: length = inScopeNSes->length; pn2 = pn->pn_head; xml = ParseNodeToXML(cx, pn2, inScopeNSes, flags); if (!xml) goto fail; flags &= ~XSF_PRECOMPILED_ROOT; n = pn->pn_count; JS_ASSERT(n >= 2); n -= 2; if (!XMLArraySetCapacity(cx, &xml->xml_kids, n)) goto fail; i = 0; while ((pn2 = pn2->pn_next) != NULL) { if (!pn2->pn_next) { /* Don't append the end tag! */ JS_ASSERT(pn2->pn_type == TOK_XMLETAGO); break; } if ((flags & XSF_IGNORE_WHITESPACE) && n > 1 && pn2->pn_type == TOK_XMLSPACE) { --n; continue; } kid = ParseNodeToXML(cx, pn2, inScopeNSes, flags); if (kid == PN2X_SKIP_CHILD) { --n; continue; } if (!kid) { xml->xml_kids.length = i; goto fail; } /* Store kid in xml right away, to protect it from GC. */ XMLARRAY_SET_MEMBER(&xml->xml_kids, i, kid); kid->parent = xml; ++i; /* XXX where is this documented in an XML spec, or in E4X? */ if ((flags & XSF_IGNORE_WHITESPACE) && n > 1 && kid->xml_class == JSXML_CLASS_TEXT) { str = ChompXMLWhitespace(cx, kid->xml_value); if (!str) goto fail; kid->xml_value = str; } } JS_ASSERT(i == n); xml->xml_kids.length = n; if (n < pn->pn_count - 2) XMLArrayTrim(&xml->xml_kids); XMLARRAY_TRUNCATE(cx, inScopeNSes, length); break; case TOK_XMLLIST: xml = js_NewXML(cx, JSXML_CLASS_LIST); if (!xml) goto fail; n = pn->pn_count; if (!XMLArraySetCapacity(cx, &xml->xml_kids, n)) goto fail; i = 0; for (pn2 = pn->pn_head; pn2; pn2 = pn2->pn_next) { /* * Always ignore insignificant whitespace in lists -- we shouldn't * condition this on an XML.ignoreWhitespace setting when the list * constructor is XMLList (note XML/XMLList unification hazard). */ if (pn2->pn_type == TOK_XMLSPACE) { --n; continue; } kid = ParseNodeToXML(cx, pn2, inScopeNSes, flags); if (kid == PN2X_SKIP_CHILD) { --n; continue; } if (!kid) { xml->xml_kids.length = i; goto fail; } XMLARRAY_SET_MEMBER(&xml->xml_kids, i, kid); ++i; } xml->xml_kids.length = n; if (n < pn->pn_count) XMLArrayTrim(&xml->xml_kids); break; case TOK_XMLSTAGO: case TOK_XMLPTAGC: length = inScopeNSes->length; pn2 = pn->pn_head; JS_ASSERT(pn2->pn_type = TOK_XMLNAME); if (pn2->pn_arity == PN_LIST) goto syntax; xml = js_NewXML(cx, JSXML_CLASS_ELEMENT); if (!xml) goto fail; /* First pass: check syntax and process namespace declarations. */ JS_ASSERT(pn->pn_count >= 1); n = pn->pn_count - 1; pnp = &pn2->pn_next; head = *pnp; while ((pn2 = *pnp) != NULL) { size_t length; const jschar *chars; if (pn2->pn_type != TOK_XMLNAME || pn2->pn_arity != PN_NULLARY) goto syntax; /* Enforce "Well-formedness constraint: Unique Att Spec". */ for (pn3 = head; pn3 != pn2; pn3 = pn3->pn_next->pn_next) { if (pn3->pn_atom == pn2->pn_atom) { js_ReportCompileErrorNumber(cx, pn2, JSREPORT_PN | JSREPORT_ERROR, JSMSG_DUPLICATE_XML_ATTR, js_ValueToPrintableString(cx, ATOM_KEY(pn2->pn_atom))); goto fail; } } str = ATOM_TO_STRING(pn2->pn_atom); pn2 = pn2->pn_next; JS_ASSERT(pn2); if (pn2->pn_type != TOK_XMLATTR) goto syntax; length = JSSTRING_LENGTH(str); chars = JSSTRING_CHARS(str); if (length >= 5 && IS_XMLNS_CHARS(chars) && (length == 5 || chars[5] == ':')) { JSString *uri, *prefix; uri = ATOM_TO_STRING(pn2->pn_atom); if (length == 5) { /* 10.3.2.1. Step 6(h)(i)(1)(a). */ prefix = cx->runtime->emptyString; } else { prefix = js_NewStringCopyN(cx, chars + 6, length - 6, 0); if (!prefix) goto fail; } /* * Once the new ns is appended to xml->xml_namespaces, it is * protected from GC by the object that owns xml -- which is * either xml->object if outermost, or the object owning xml's * oldest ancestor if !outermost. */ ns = js_NewXMLNamespace(cx, prefix, uri, JS_TRUE); if (!ns) goto fail; /* * Don't add a namespace that's already in scope. If someone * extracts a child property from its parent via [[Get]], then * we enforce the invariant, noted many times in ECMA-357, that * the child's namespaces form a possibly-improper superset of * its ancestors' namespaces. */ if (!XMLARRAY_HAS_MEMBER(inScopeNSes, ns, namespace_identity)) { if (!XMLARRAY_APPEND(cx, inScopeNSes, ns) || !XMLARRAY_APPEND(cx, &xml->xml_namespaces, ns)) { goto fail; } } JS_ASSERT(n >= 2); n -= 2; *pnp = pn2->pn_next; /* XXXbe recycle pn2 */ continue; } pnp = &pn2->pn_next; } /* * If called from js_ParseNodeToXMLObject, emulate the effect of the * ... wrapping done by "ToXML Applied to * the String Type" (ECMA-357 10.3.1). */ if (flags & XSF_PRECOMPILED_ROOT) { JS_ASSERT(length >= 1); ns = XMLARRAY_MEMBER(inScopeNSes, 0, JSXMLNamespace); JS_ASSERT(!XMLARRAY_HAS_MEMBER(&xml->xml_namespaces, ns, namespace_identity)); ns = js_NewXMLNamespace(cx, ns->prefix, ns->uri, JS_FALSE); if (!ns) goto fail; if (!XMLARRAY_APPEND(cx, &xml->xml_namespaces, ns)) goto fail; } XMLArrayTrim(&xml->xml_namespaces); /* Second pass: process tag name and attributes, using namespaces. */ pn2 = pn->pn_head; qn = ParseNodeToQName(cx, pn2, inScopeNSes, JS_FALSE); if (!qn) goto fail; xml->name = qn; JS_ASSERT((n & 1) == 0); n >>= 1; if (!XMLArraySetCapacity(cx, &xml->xml_attrs, n)) goto fail; for (i = 0; (pn2 = pn2->pn_next) != NULL; i++) { qn = ParseNodeToQName(cx, pn2, inScopeNSes, JS_TRUE); if (!qn) { xml->xml_attrs.length = i; goto fail; } /* * Enforce "Well-formedness constraint: Unique Att Spec", part 2: * this time checking local name and namespace URI. */ for (j = 0; j < i; j++) { attrj = XMLARRAY_MEMBER(&xml->xml_attrs, j, JSXML); attrjqn = attrj->name; if (!js_CompareStrings(attrjqn->uri, qn->uri) && !js_CompareStrings(attrjqn->localName, qn->localName)) { js_ReportCompileErrorNumber(cx, pn2, JSREPORT_PN | JSREPORT_ERROR, JSMSG_DUPLICATE_XML_ATTR, js_ValueToPrintableString(cx, ATOM_KEY(pn2->pn_atom))); goto fail; } } pn2 = pn2->pn_next; JS_ASSERT(pn2); JS_ASSERT(pn2->pn_type == TOK_XMLATTR); attr = js_NewXML(cx, JSXML_CLASS_ATTRIBUTE); if (!attr) { xml->xml_attrs.length = i; goto fail; } XMLARRAY_SET_MEMBER(&xml->xml_attrs, i, attr); attr->parent = xml; attr->name = qn; attr->xml_value = ATOM_TO_STRING(pn2->pn_atom); } xml->xml_attrs.length = n; /* Point tag closes its own namespace scope. */ if (pn->pn_type == TOK_XMLPTAGC) XMLARRAY_TRUNCATE(cx, inScopeNSes, length); break; case TOK_XMLSPACE: case TOK_XMLTEXT: case TOK_XMLCDATA: case TOK_XMLCOMMENT: case TOK_XMLPI: str = ATOM_TO_STRING(pn->pn_atom); qn = NULL; if (pn->pn_type == TOK_XMLCOMMENT) { if (flags & XSF_IGNORE_COMMENTS) goto skip_child; xml_class = JSXML_CLASS_COMMENT; } else if (pn->pn_type == TOK_XMLPI) { if (IS_XML(str)) { js_ReportCompileErrorNumber(cx, pn, JSREPORT_PN | JSREPORT_ERROR, JSMSG_RESERVED_ID, js_ValueToPrintableString(cx, STRING_TO_JSVAL(str))); goto fail; } if (flags & XSF_IGNORE_PROCESSING_INSTRUCTIONS) goto skip_child; qn = ParseNodeToQName(cx, pn, inScopeNSes, JS_FALSE); if (!qn) goto fail; str = pn->pn_atom2 ? ATOM_TO_STRING(pn->pn_atom2) : cx->runtime->emptyString; xml_class = JSXML_CLASS_PROCESSING_INSTRUCTION; } else { /* CDATA section content, or element text. */ xml_class = JSXML_CLASS_TEXT; } xml = js_NewXML(cx, xml_class); if (!xml) goto fail; xml->name = qn; if (pn->pn_type == TOK_XMLSPACE) xml->xml_flags |= XMLF_WHITESPACE_TEXT; xml->xml_value = str; break; default: goto syntax; } JS_LeaveLocalRootScope(cx); if ((flags & XSF_PRECOMPILED_ROOT) && !js_GetXMLObject(cx, xml)) return NULL; return xml; skip_child: js_LeaveLocalRootScope(cx); return PN2X_SKIP_CHILD; #undef PN2X_SKIP_CHILD syntax: js_ReportCompileErrorNumber(cx, pn, JSREPORT_PN | JSREPORT_ERROR, JSMSG_BAD_XML_MARKUP); fail: JS_LeaveLocalRootScope(cx); return NULL; } /* * XML helper, object-ops, and library functions. We start with the helpers, * in ECMA-357 order, but merging XML (9.1) and XMLList (9.2) helpers. */ static JSBool GetXMLSetting(JSContext *cx, const char *name, jsval *vp) { jsval v; if (!js_FindConstructor(cx, NULL, js_XML_str, &v)) return JS_FALSE; if (!JSVAL_IS_FUNCTION(cx, v)) { *vp = JSVAL_VOID; return JS_TRUE; } return JS_GetProperty(cx, JSVAL_TO_OBJECT(v), name, vp); } static JSBool GetBooleanXMLSetting(JSContext *cx, const char *name, JSBool *bp) { int i; jsval v; if (cx->xmlSettingFlags & XSF_CACHE_VALID) { for (i = 0; xml_static_props[i].name; i++) { if (!strcmp(xml_static_props[i].name, name)) { *bp = (cx->xmlSettingFlags & JS_BIT(i)) != 0; return JS_TRUE; } } *bp = JS_FALSE; return JS_TRUE; } return GetXMLSetting(cx, name, &v) && js_ValueToBoolean(cx, v, bp); } static JSBool GetUint32XMLSetting(JSContext *cx, const char *name, uint32 *uip) { jsval v; return GetXMLSetting(cx, name, &v) && js_ValueToECMAUint32(cx, v, uip); } static JSBool GetXMLSettingFlags(JSContext *cx, uintN *flagsp) { JSBool flag; /* Just get the first flag to validate the setting flags cache. */ if (!GetBooleanXMLSetting(cx, js_ignoreComments_str, &flag)) return JS_FALSE; *flagsp = cx->xmlSettingFlags; return JS_TRUE; } static JSXML * ParseXMLSource(JSContext *cx, JSString *src) { jsval nsval; JSXMLNamespace *ns; size_t urilen, srclen, length, offset, dstlen; jschar *chars; const jschar *srcp, *endp; void *mark; JSTokenStream *ts; uintN lineno; JSStackFrame *fp; JSOp op; JSParseNode *pn; JSXML *xml; JSXMLArray nsarray; uintN flags; static const char prefix[] = ""; static const char suffix[] = ""; #define constrlen(constr) (sizeof(constr) - 1) if (!js_GetDefaultXMLNamespace(cx, &nsval)) return NULL; ns = (JSXMLNamespace *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(nsval)); urilen = JSSTRING_LENGTH(ns->uri); srclen = JSSTRING_LENGTH(src); length = constrlen(prefix) + urilen + constrlen(middle) + srclen + constrlen(suffix); chars = (jschar *) JS_malloc(cx, (length + 1) * sizeof(jschar)); if (!chars) return NULL; dstlen = length; js_InflateStringToBuffer(cx, prefix, constrlen(prefix), chars, &dstlen); offset = dstlen; js_strncpy(chars + offset, JSSTRING_CHARS(ns->uri), urilen); offset += urilen; dstlen = length - offset + 1; js_InflateStringToBuffer(cx, middle, constrlen(middle), chars + offset, &dstlen); offset += dstlen; srcp = JSSTRING_CHARS(src); js_strncpy(chars + offset, srcp, srclen); offset += srclen; dstlen = length - offset + 1; js_InflateStringToBuffer(cx, suffix, constrlen(suffix), chars + offset, &dstlen); chars [offset + dstlen] = 0; mark = JS_ARENA_MARK(&cx->tempPool); ts = js_NewBufferTokenStream(cx, chars, length); if (!ts) return NULL; for (fp = cx->fp; fp && !fp->pc; fp = fp->down) continue; if (fp) { op = (JSOp) *fp->pc; if (op == JSOP_TOXML || op == JSOP_TOXMLLIST) { ts->filename = fp->script->filename; lineno = js_PCToLineNumber(cx, fp->script, fp->pc); for (endp = srcp + srclen; srcp < endp; srcp++) if (*srcp == '\n') --lineno; ts->lineno = lineno; } } JS_KEEP_ATOMS(cx->runtime); pn = js_ParseXMLTokenStream(cx, cx->fp->scopeChain, ts, JS_FALSE); xml = NULL; if (pn && XMLArrayInit(cx, &nsarray, 1)) { if (GetXMLSettingFlags(cx, &flags)) xml = ParseNodeToXML(cx, pn, &nsarray, flags); XMLArrayFinish(cx, &nsarray); } JS_UNKEEP_ATOMS(cx->runtime); JS_ARENA_RELEASE(&cx->tempPool, mark); JS_free(cx, chars); return xml; #undef constrlen } /* * Errata in 10.3.1, 10.4.1, and 13.4.4.24 (at least). * * 10.3.1 Step 6(a) fails to NOTE that implementations that do not enforce * the constraint: * * for all x belonging to XML: * x.[[InScopeNamespaces]] >= x.[[Parent]].[[InScopeNamespaces]] * * must union x.[[InScopeNamespaces]] into x[0].[[InScopeNamespaces]] here * (in new sub-step 6(a), renumbering the others to (b) and (c)). * * Same goes for 10.4.1 Step 7(a). * * In order for XML.prototype.namespaceDeclarations() to work correctly, the * default namespace thereby unioned into x[0].[[InScopeNamespaces]] must be * flagged as not declared, so that 13.4.4.24 Step 8(a) can exclude all such * undeclared namespaces associated with x not belonging to ancestorNS. */ static JSXML * OrphanXMLChild(JSContext *cx, JSXML *xml, uint32 i) { JSXMLNamespace *ns; ns = XMLARRAY_MEMBER(&xml->xml_namespaces, 0, JSXMLNamespace); xml = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (xml->xml_class == JSXML_CLASS_ELEMENT) { if (!XMLARRAY_APPEND(cx, &xml->xml_namespaces, ns)) return NULL; ns->declared = JS_FALSE; } xml->parent = NULL; return xml; } static JSObject * ToXML(JSContext *cx, jsval v) { JSObject *obj; JSXML *xml; JSClass *clasp; JSString *str; uint32 length; if (JSVAL_IS_PRIMITIVE(v)) { if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) goto bad; } else { obj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, obj)) { xml = (JSXML *) JS_GetPrivate(cx, obj); if (xml->xml_class == JSXML_CLASS_LIST) { if (xml->xml_kids.length != 1) goto bad; xml = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); JS_ASSERT(xml->xml_class != JSXML_CLASS_LIST); return js_GetXMLObject(cx, xml); } return obj; } clasp = OBJ_GET_CLASS(cx, obj); if (clasp->flags & JSCLASS_DOCUMENT_OBSERVER) { JS_ASSERT(0); } if (clasp != &js_StringClass && clasp != &js_NumberClass && clasp != &js_BooleanClass) { goto bad; } } str = js_ValueToString(cx, v); if (!str) return NULL; if (IS_EMPTY(str)) { length = 0; #ifdef __GNUC__ /* suppress bogus gcc warnings */ xml = NULL; #endif } else { xml = ParseXMLSource(cx, str); if (!xml) return NULL; length = JSXML_LENGTH(xml); } if (length == 0) { obj = js_NewXMLObject(cx, JSXML_CLASS_TEXT); if (!obj) return NULL; } else if (length == 1) { xml = OrphanXMLChild(cx, xml, 0); if (!xml) return NULL; obj = js_GetXMLObject(cx, xml); if (!obj) return NULL; } else { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_SYNTAX_ERROR); return NULL; } return obj; bad: str = js_DecompileValueGenerator(cx, JSDVG_IGNORE_STACK, v, NULL); if (str) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_CONVERSION, JS_GetStringBytes(str)); } return NULL; } static JSBool Append(JSContext *cx, JSXML *list, JSXML *kid); static JSObject * ToXMLList(JSContext *cx, jsval v) { JSObject *obj, *listobj; JSXML *xml, *list, *kid; JSClass *clasp; JSString *str; uint32 i, length; if (JSVAL_IS_PRIMITIVE(v)) { if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) goto bad; } else { obj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, obj)) { xml = (JSXML *) JS_GetPrivate(cx, obj); if (xml->xml_class != JSXML_CLASS_LIST) { listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return NULL; list = (JSXML *) JS_GetPrivate(cx, listobj); if (!Append(cx, list, xml)) return NULL; return listobj; } return obj; } clasp = OBJ_GET_CLASS(cx, obj); if (clasp->flags & JSCLASS_DOCUMENT_OBSERVER) { JS_ASSERT(0); } if (clasp != &js_StringClass && clasp != &js_NumberClass && clasp != &js_BooleanClass) { goto bad; } } str = js_ValueToString(cx, v); if (!str) return NULL; if (IS_EMPTY(str)) { xml = NULL; length = 0; } else { if (!JS_EnterLocalRootScope(cx)) return NULL; xml = ParseXMLSource(cx, str); if (!xml) { JS_LeaveLocalRootScope(cx); return NULL; } length = JSXML_LENGTH(xml); } listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (listobj) { list = (JSXML *) JS_GetPrivate(cx, listobj); for (i = 0; i < length; i++) { kid = OrphanXMLChild(cx, xml, i); if (!kid) return NULL; if (!Append(cx, list, kid)) { listobj = NULL; break; } } } if (xml) JS_LeaveLocalRootScope(cx); return listobj; bad: str = js_DecompileValueGenerator(cx, JSDVG_IGNORE_STACK, v, NULL); if (str) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XMLLIST_CONVERSION, JS_GetStringBytes(str)); } return NULL; } /* * ECMA-357 10.2.1 Steps 5-7 pulled out as common subroutines of XMLToXMLString * and their library-public js_* counterparts. The guts of MakeXMLCDataString, * MakeXMLCommentString, and MakeXMLPIString are further factored into a common * MakeXMLSpecialString subroutine. * * These functions take ownership of sb->base, if sb is non-null, in all cases * of success or failure. */ static JSString * MakeXMLSpecialString(JSContext *cx, JSStringBuffer *sb, JSString *str, JSString *str2, const jschar *prefix, size_t prefixlength, const jschar *suffix, size_t suffixlength) { JSStringBuffer localSB; size_t length, length2, newlength; jschar *bp, *base; if (!sb) { sb = &localSB; js_InitStringBuffer(sb); } length = JSSTRING_LENGTH(str); length2 = str2 ? JSSTRING_LENGTH(str2) : 0; newlength = STRING_BUFFER_OFFSET(sb) + prefixlength + length + ((length2 != 0) ? 1 + length2 : 0) + suffixlength; bp = base = (jschar *) JS_realloc(cx, sb->base, (newlength + 1) * sizeof(jschar)); if (!bp) { js_FinishStringBuffer(sb); return NULL; } bp += STRING_BUFFER_OFFSET(sb); js_strncpy(bp, prefix, prefixlength); bp += prefixlength; js_strncpy(bp, JSSTRING_CHARS(str), length); bp += length; if (length2 != 0) { *bp++ = (jschar) ' '; js_strncpy(bp, JSSTRING_CHARS(str2), length2); bp += length2; } js_strncpy(bp, suffix, suffixlength); bp[suffixlength] = 0; str = js_NewString(cx, base, newlength, 0); if (!str) free(base); return str; } static JSString * MakeXMLCDATAString(JSContext *cx, JSStringBuffer *sb, JSString *str) { static const jschar cdata_prefix_ucNstr[] = {'<', '!', '[', 'C', 'D', 'A', 'T', 'A', '['}; static const jschar cdata_suffix_ucNstr[] = {']', ']', '>'}; return MakeXMLSpecialString(cx, sb, str, NULL, cdata_prefix_ucNstr, 9, cdata_suffix_ucNstr, 3); } static JSString * MakeXMLCommentString(JSContext *cx, JSStringBuffer *sb, JSString *str) { static const jschar comment_prefix_ucNstr[] = {'<', '!', '-', '-'}; static const jschar comment_suffix_ucNstr[] = {'-', '-', '>'}; return MakeXMLSpecialString(cx, sb, str, NULL, comment_prefix_ucNstr, 4, comment_suffix_ucNstr, 3); } static JSString * MakeXMLPIString(JSContext *cx, JSStringBuffer *sb, JSString *name, JSString *value) { static const jschar pi_prefix_ucNstr[] = {'<', '?'}; static const jschar pi_suffix_ucNstr[] = {'?', '>'}; return MakeXMLSpecialString(cx, sb, name, value, pi_prefix_ucNstr, 2, pi_suffix_ucNstr, 2); } /* * ECMA-357 10.2.1 17(d-g) pulled out into a common subroutine that appends * equals, a double quote, an attribute value, and a closing double quote. */ static void AppendAttributeValue(JSContext *cx, JSStringBuffer *sb, JSString *valstr) { js_AppendCString(sb, "=\""); valstr = js_EscapeAttributeValue(cx, valstr); if (!valstr) { free(sb->base); sb->base = STRING_BUFFER_ERROR_BASE; return; } js_AppendJSString(sb, valstr); js_AppendChar(sb, '"'); } /* * ECMA-357 10.2.1.1 EscapeElementValue helper method. * * This function takes ownership of sb->base, if sb is non-null, in all cases * of success or failure. */ static JSString * EscapeElementValue(JSContext *cx, JSStringBuffer *sb, JSString *str) { size_t length, newlength; const jschar *cp, *start, *end; jschar c; length = newlength = JSSTRING_LENGTH(str); for (cp = start = JSSTRING_CHARS(str), end = cp + length; cp < end; cp++) { c = *cp; if (c == '<' || c == '>') newlength += 3; else if (c == '&') newlength += 4; } if ((sb && STRING_BUFFER_OFFSET(sb) != 0) || newlength > length) { JSStringBuffer localSB; if (!sb) { sb = &localSB; js_InitStringBuffer(sb); } if (!sb->grow(sb, STRING_BUFFER_OFFSET(sb) + newlength)) { JS_ReportOutOfMemory(cx); return NULL; } for (cp = start; cp < end; cp++) { c = *cp; if (c == '<') js_AppendCString(sb, js_lt_entity_str); else if (c == '>') js_AppendCString(sb, js_gt_entity_str); else if (c == '&') js_AppendCString(sb, js_amp_entity_str); else js_AppendChar(sb, c); } JS_ASSERT(STRING_BUFFER_OK(sb)); str = js_NewString(cx, sb->base, STRING_BUFFER_OFFSET(sb), 0); if (!str) js_FinishStringBuffer(sb); } return str; } /* * ECMA-357 10.2.1.2 EscapeAttributeValue helper method. * This function takes ownership of sb->base, if sb is non-null, in all cases. */ static JSString * EscapeAttributeValue(JSContext *cx, JSStringBuffer *sb, JSString *str) { size_t length, newlength; const jschar *cp, *start, *end; jschar c; length = newlength = JSSTRING_LENGTH(str); for (cp = start = JSSTRING_CHARS(str), end = cp + length; cp < end; cp++) { c = *cp; if (c == '"') newlength += 5; else if (c == '<') newlength += 3; else if (c == '&' || c == '\n' || c == '\r' || c == '\t') newlength += 4; } if ((sb && STRING_BUFFER_OFFSET(sb) != 0) || newlength > length) { JSStringBuffer localSB; if (!sb) { sb = &localSB; js_InitStringBuffer(sb); } if (!sb->grow(sb, STRING_BUFFER_OFFSET(sb) + newlength)) { JS_ReportOutOfMemory(cx); return NULL; } for (cp = start; cp < end; cp++) { c = *cp; if (c == '"') js_AppendCString(sb, js_quot_entity_str); else if (c == '<') js_AppendCString(sb, js_lt_entity_str); else if (c == '&') js_AppendCString(sb, js_amp_entity_str); else if (c == '\n') js_AppendCString(sb, " "); else if (c == '\r') js_AppendCString(sb, " "); else if (c == '\t') js_AppendCString(sb, " "); else js_AppendChar(sb, c); } JS_ASSERT(STRING_BUFFER_OK(sb)); str = js_NewString(cx, sb->base, STRING_BUFFER_OFFSET(sb), 0); if (!str) js_FinishStringBuffer(sb); } return str; } /* 13.3.5.4 [[GetNamespace]]([InScopeNamespaces]) */ static JSXMLNamespace * GetNamespace(JSContext *cx, JSXMLQName *qn, const JSXMLArray *inScopeNSes) { JSXMLNamespace *match, *ns; uint32 i, n; jsval argv[2]; JSObject *nsobj; JS_ASSERT(qn->uri); if (!qn->uri) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_NAMESPACE, qn->prefix ? js_ValueToPrintableString(cx, STRING_TO_JSVAL(qn->prefix)) : js_type_str[JSTYPE_VOID]); return NULL; } /* Look for a matching namespace in inScopeNSes, if provided. */ match = NULL; if (inScopeNSes) { for (i = 0, n = inScopeNSes->length; i < n; i++) { ns = XMLARRAY_MEMBER(inScopeNSes, i, JSXMLNamespace); /* * Erratum, very tricky, and not specified in ECMA-357 13.3.5.4: * If we preserve prefixes, we must match null qn->prefix against * an empty ns->prefix, in order to avoid generating redundant * prefixed and default namespaces for cases such as: * * x = * print(x.toXMLString()); * * Per 10.3.2.1, the namespace attribute in t has an empty string * prefix (*not* a null prefix), per 10.3.2.1 Step 6(h)(i)(1): * * 1. If the [local name] property of a is "xmlns" * a. Map ns.prefix to the empty string * * But t's name has a null prefix in this implementation, meaning * *undefined*, per 10.3.2.1 Step 6(c)'s NOTE (which refers to * the http://www.w3.org/TR/xml-infoset/ spec, item 2.2.3, without * saying how "no value" maps to an ECMA-357 value -- but it must * map to the *undefined* prefix value). * * Since "" != undefined (or null, in the current implementation) * the ECMA-357 spec will fail to match in [[GetNamespace]] called * on t with argument {} U {(prefix="", uri="http://foo.com")}. * This spec bug leads to ToXMLString results that duplicate the * declared namespace. */ if (!js_CompareStrings(ns->uri, qn->uri) && (ns->prefix == qn->prefix || ((ns->prefix && qn->prefix) ? !js_CompareStrings(ns->prefix, qn->prefix) : IS_EMPTY(ns->prefix ? ns->prefix : qn->prefix)))) { match = ns; break; } } } /* If we didn't match, make a new namespace from qn. */ if (!match) { argv[0] = qn->prefix ? STRING_TO_JSVAL(qn->prefix) : JSVAL_VOID; argv[1] = STRING_TO_JSVAL(qn->uri); nsobj = js_ConstructObject(cx, &js_NamespaceClass.base, NULL, NULL, 2, argv); if (!nsobj) return NULL; match = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); } return match; } static JSString * GeneratePrefix(JSContext *cx, JSString *uri, JSXMLArray *decls) { const jschar *cp, *start, *end; size_t length, newlength, offset; uint32 i, n, m, serial; jschar *bp, *dp; JSBool done; JSXMLNamespace *ns; JSString *prefix; JS_ASSERT(!IS_EMPTY(uri)); /* * Try peeling off the last filename suffix or pathname component till * we have a valid XML name. This heuristic will prefer "xul" given * ".../there.is.only.xul", "xbl" given ".../xbl", and "xbl2" given any * likely URI of the form ".../xbl2/2005". */ start = JSSTRING_CHARS(uri); cp = end = start + JSSTRING_LENGTH(uri); while (--cp > start) { if (*cp == '.' || *cp == '/' || *cp == ':') { ++cp; if (IsXMLName(cp, PTRDIFF(end, cp, jschar))) break; end = --cp; } } length = PTRDIFF(end, cp, jschar); /* * Now search through decls looking for a collision. If we collide with * an existing prefix, start tacking on a hyphen and a serial number. */ serial = 0; bp = NULL; #ifdef __GNUC__ /* suppress bogus gcc warnings */ newlength = 0; #endif do { done = JS_TRUE; for (i = 0, n = decls->length; i < n; i++) { ns = XMLARRAY_MEMBER(decls, i, JSXMLNamespace); if (ns->prefix && JSSTRING_LENGTH(ns->prefix) == length && !memcmp(JSSTRING_CHARS(ns->prefix), cp, length * sizeof(jschar))) { if (!bp) { newlength = length + 2 + (size_t) log10(n); bp = (jschar *) JS_malloc(cx, (newlength + 1) * sizeof(jschar)); if (!bp) return NULL; js_strncpy(bp, cp, length); } ++serial; JS_ASSERT(serial <= n); dp = bp + length + 2 + (size_t) log10(serial); *dp = 0; for (m = serial; m != 0; m /= 10) *--dp = (jschar)('0' + m % 10); *--dp = '-'; JS_ASSERT(dp == bp + length); done = JS_FALSE; break; } } } while (!done); if (!bp) { offset = PTRDIFF(cp, start, jschar); prefix = js_NewDependentString(cx, uri, offset, length, 0); } else { prefix = js_NewString(cx, bp, newlength, 0); if (!prefix) JS_free(cx, bp); } return prefix; } static JSBool namespace_match(const void *a, const void *b) { const JSXMLNamespace *nsa = (const JSXMLNamespace *) a; const JSXMLNamespace *nsb = (const JSXMLNamespace *) b; if (nsb->prefix) return nsa->prefix && !js_CompareStrings(nsa->prefix, nsb->prefix); return !js_CompareStrings(nsa->uri, nsb->uri); } /* ECMA-357 10.2.1 and 10.2.2 */ static JSString * XMLToXMLString(JSContext *cx, JSXML *xml, const JSXMLArray *ancestorNSes, uintN indentLevel) { JSBool pretty, indentKids; JSStringBuffer sb; JSString *str, *prefix, *kidstr; uint32 i, n; JSXMLArray empty, decls, ancdecls; JSXMLNamespace *ns, *ns2; uintN nextIndentLevel; JSXML *attr, *kid; if (!GetBooleanXMLSetting(cx, js_prettyPrinting_str, &pretty)) return NULL; js_InitStringBuffer(&sb); if (pretty) js_RepeatChar(&sb, ' ', indentLevel); str = NULL; switch (xml->xml_class) { case JSXML_CLASS_TEXT: /* Step 4. */ if (pretty) { str = ChompXMLWhitespace(cx, xml->xml_value); if (!str) return NULL; } else { str = xml->xml_value; } return EscapeElementValue(cx, &sb, str); case JSXML_CLASS_ATTRIBUTE: /* Step 5. */ return EscapeAttributeValue(cx, &sb, xml->xml_value); case JSXML_CLASS_COMMENT: /* Step 6. */ return MakeXMLCommentString(cx, &sb, xml->xml_value); case JSXML_CLASS_PROCESSING_INSTRUCTION: /* Step 7. */ return MakeXMLPIString(cx, &sb, xml->name->localName, xml->xml_value); case JSXML_CLASS_LIST: /* ECMA-357 10.2.2. */ for (i = 0, n = xml->xml_kids.length; i < n; i++) { if (pretty && i != 0) js_AppendChar(&sb, '\n'); kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); kidstr = XMLToXMLString(cx, kid, ancestorNSes, indentLevel); if (!kidstr) goto list_out; js_AppendJSString(&sb, kidstr); } if (!sb.base) { if (!STRING_BUFFER_OK(&sb)) { JS_ReportOutOfMemory(cx); return NULL; } return cx->runtime->emptyString; } str = js_NewString(cx, sb.base, STRING_BUFFER_OFFSET(&sb), 0); list_out: if (!str) js_FinishStringBuffer(&sb); return str; default:; } /* After this point, control must flow through label out: to exit. */ if (!JS_EnterLocalRootScope(cx)) return NULL; /* ECMA-357 10.2.1 step 8 onward: handle ToXMLString on an XML element. */ if (!ancestorNSes) { XMLArrayInit(cx, &empty, 0); ancestorNSes = ∅ } XMLArrayInit(cx, &decls, 0); ancdecls.capacity = 0; /* Clone in-scope namespaces not in ancestorNSes into decls. */ for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); if (!ns->declared) continue; if (!XMLARRAY_HAS_MEMBER(ancestorNSes, ns, namespace_identity)) { /* NOTE: may want to exclude unused namespaces here. */ ns2 = js_NewXMLNamespace(cx, ns->prefix, ns->uri, JS_TRUE); if (!ns2) goto out; if (!XMLARRAY_APPEND(cx, &decls, ns2)) goto out; } } /* * Union ancestorNSes and decls into ancdecls. Note that ancdecls does * not own its member references. In the spec, ancdecls has no name, but * is always written out as (AncestorNamespaces U namespaceDeclarations). */ if (!XMLArrayInit(cx, &ancdecls, ancestorNSes->length + decls.length)) goto out; for (i = 0, n = ancestorNSes->length; i < n; i++) { ns2 = XMLARRAY_MEMBER(ancestorNSes, i, JSXMLNamespace); JS_ASSERT(!XMLARRAY_HAS_MEMBER(&decls, ns2, namespace_identity)); if (!XMLARRAY_APPEND(cx, &ancdecls, ns2)) goto out; } for (i = 0, n = decls.length; i < n; i++) { ns2 = XMLARRAY_MEMBER(&decls, i, JSXMLNamespace); JS_ASSERT(!XMLARRAY_HAS_MEMBER(&ancdecls, ns2, namespace_identity)); if (!XMLARRAY_APPEND(cx, &ancdecls, ns2)) goto out; } /* Step 11, except we don't clone ns unless its prefix is undefined. */ ns = GetNamespace(cx, xml->name, &ancdecls); if (!ns) goto out; /* Step 12 (NULL means *undefined* here), plus the deferred ns cloning. */ if (!ns->prefix) { /* * Create a namespace prefix that isn't used by any member of decls. * Assign the new prefix to a copy of ns. Flag this namespace as if * it were declared, for assertion-testing's sake later below. * * Erratum: if ns->prefix and xml->name are both null (*undefined* in * ECMA-357), we know that xml was named using the default namespace * (proof: see GetNamespace and the Namespace constructor called with * two arguments). So we ought not generate a new prefix here, when * we can declare ns as the default namespace for xml. * * This helps descendants inherit the namespace instead of redundantly * redeclaring it with generated prefixes in each descendant. */ if (!xml->name->prefix) { prefix = cx->runtime->emptyString; } else { prefix = GeneratePrefix(cx, ns->uri, &ancdecls); if (!prefix) goto out; } ns = js_NewXMLNamespace(cx, prefix, ns->uri, JS_TRUE); if (!ns) goto out; /* * If the xml->name was unprefixed, we must remove any declared default * namespace from decls before appending ns. How can you get a default * namespace in decls that doesn't match the one from name? Apparently * by calling x.setNamespace(ns) where ns has no prefix. The other way * to fix this is to update x's in-scope namespaces when setNamespace * is called, but that's not specified by ECMA-357. * * Likely Erratum here, depending on whether the lack of update to x's * in-scope namespace in XML.prototype.setNamespace (13.4.4.36) is an * erratum or not. Note that changing setNamespace to update the list * of in-scope namespaces will change x.namespaceDeclarations(). */ if (IS_EMPTY(prefix)) { i = XMLArrayFindMember(&decls, ns, namespace_match); if (i != XML_NOT_FOUND) XMLArrayDelete(cx, &decls, i, JS_TRUE); } /* * In the spec, ancdecls has no name, but is always written out as * (AncestorNamespaces U namespaceDeclarations). Since we compute * that union in ancdecls, any time we append a namespace strong * ref to decls, we must also append a weak ref to ancdecls. Order * matters here: code at label out: releases strong refs in decls. */ if (!XMLARRAY_APPEND(cx, &ancdecls, ns) || !XMLARRAY_APPEND(cx, &decls, ns)) { goto out; } } /* Format the element or point-tag into sb. */ js_AppendChar(&sb, '<'); if (ns->prefix && !IS_EMPTY(ns->prefix)) { js_AppendJSString(&sb, ns->prefix); js_AppendChar(&sb, ':'); } js_AppendJSString(&sb, xml->name->localName); /* * Step 16 makes a union to avoid writing two loops in step 17, to share * common attribute value appending spec-code. We prefer two loops for * faster code and less data overhead. */ /* Step 17(c): append XML namespace declarations. */ for (i = 0, n = decls.length; i < n; i++) { ns2 = XMLARRAY_MEMBER(&decls, i, JSXMLNamespace); JS_ASSERT(ns2->declared); js_AppendCString(&sb, " xmlns"); /* 17(c)(ii): NULL means *undefined* here. */ if (!ns2->prefix) { prefix = GeneratePrefix(cx, ns2->uri, &ancdecls); if (!prefix) goto out; ns2->prefix = prefix; } /* 17(c)(iii). */ if (!IS_EMPTY(ns2->prefix)) { js_AppendChar(&sb, ':'); js_AppendJSString(&sb, ns2->prefix); } /* 17(d-g). */ AppendAttributeValue(cx, &sb, ns2->uri); } /* Step 17(b): append attributes. */ for (i = 0, n = xml->xml_attrs.length; i < n; i++) { attr = XMLARRAY_MEMBER(&xml->xml_attrs, i, JSXML); js_AppendChar(&sb, ' '); ns2 = GetNamespace(cx, attr->name, &ancdecls); if (!ns2) goto out; /* 17(b)(ii): NULL means *undefined* here. */ if (!ns2->prefix) { prefix = GeneratePrefix(cx, ns2->uri, &ancdecls); if (!prefix) goto out; /* Again, we avoid copying ns2 until we know it's prefix-less. */ ns2 = js_NewXMLNamespace(cx, prefix, ns2->uri, JS_TRUE); if (!ns2) goto out; /* * In the spec, ancdecls has no name, but is always written out as * (AncestorNamespaces U namespaceDeclarations). Since we compute * that union in ancdecls, any time we append a namespace strong * ref to decls, we must also append a weak ref to ancdecls. Order * matters here: code at label out: releases strong refs in decls. */ if (!XMLARRAY_APPEND(cx, &ancdecls, ns2) || !XMLARRAY_APPEND(cx, &decls, ns2)) { goto out; } } /* 17(b)(iii). */ if (!IS_EMPTY(ns2->prefix)) { js_AppendJSString(&sb, ns2->prefix); js_AppendChar(&sb, ':'); } /* 17(b)(iv). */ js_AppendJSString(&sb, attr->name->localName); /* 17(d-g). */ AppendAttributeValue(cx, &sb, attr->xml_value); } /* Step 18: handle point tags. */ n = xml->xml_kids.length; if (n == 0) { js_AppendCString(&sb, "/>"); } else { /* Steps 19 through 25: handle element content, and open the end-tag. */ js_AppendChar(&sb, '>'); indentKids = n > 1 || (n == 1 && XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML)->xml_class != JSXML_CLASS_TEXT); if (pretty && indentKids) { if (!GetUint32XMLSetting(cx, js_prettyIndent_str, &i)) goto out; nextIndentLevel = indentLevel + i; } else { nextIndentLevel = 0; } for (i = 0; i < n; i++) { if (pretty && indentKids) js_AppendChar(&sb, '\n'); kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); kidstr = XMLToXMLString(cx, kid, &ancdecls, nextIndentLevel); if (!kidstr) goto out; js_AppendJSString(&sb, kidstr); } if (pretty && indentKids) { js_AppendChar(&sb, '\n'); js_RepeatChar(&sb, ' ', indentLevel); } js_AppendCString(&sb, "prefix && !IS_EMPTY(ns->prefix)) { js_AppendJSString(&sb, ns->prefix); js_AppendChar(&sb, ':'); } /* Step 27. */ js_AppendJSString(&sb, xml->name->localName); js_AppendChar(&sb, '>'); } if (!STRING_BUFFER_OK(&sb)) { JS_ReportOutOfMemory(cx); goto out; } str = js_NewString(cx, sb.base, STRING_BUFFER_OFFSET(&sb), 0); out: JS_LeaveLocalRootScope(cx); if (!str) js_FinishStringBuffer(&sb); XMLArrayFinish(cx, &decls); if (ancdecls.capacity != 0) XMLArrayFinish(cx, &ancdecls); return str; } /* ECMA-357 10.2 */ static JSString * ToXMLString(JSContext *cx, jsval v) { JSObject *obj; JSString *str; JSXML *xml; if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_CONVERSION, js_type_str[JSVAL_IS_NULL(v) ? JSTYPE_NULL : JSTYPE_VOID]); return NULL; } if (JSVAL_IS_BOOLEAN(v) || JSVAL_IS_NUMBER(v)) return js_ValueToString(cx, v); if (JSVAL_IS_STRING(v)) return EscapeElementValue(cx, NULL, JSVAL_TO_STRING(v)); obj = JSVAL_TO_OBJECT(v); if (!OBJECT_IS_XML(cx, obj)) { if (!OBJ_DEFAULT_VALUE(cx, obj, JSTYPE_STRING, &v)) return NULL; str = js_ValueToString(cx, v); if (!str) return NULL; return EscapeElementValue(cx, NULL, str); } /* Handle non-element cases in this switch, returning from each case. */ xml = (JSXML *) JS_GetPrivate(cx, obj); return XMLToXMLString(cx, xml, NULL, 0); } static JSXMLQName * ToAttributeName(JSContext *cx, jsval v) { JSString *name, *uri, *prefix; JSObject *obj; JSClass *clasp; JSXMLQName *qn; JSTempValueRooter tvr; if (JSVAL_IS_STRING(v)) { name = JSVAL_TO_STRING(v); uri = prefix = cx->runtime->emptyString; } else { if (JSVAL_IS_PRIMITIVE(v)) { name = js_DecompileValueGenerator(cx, JSDVG_IGNORE_STACK, v, NULL); if (name) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_ATTR_NAME, JS_GetStringBytes(name)); } return NULL; } obj = JSVAL_TO_OBJECT(v); clasp = OBJ_GET_CLASS(cx, obj); if (clasp == &js_AttributeNameClass) return (JSXMLQName *) JS_GetPrivate(cx, obj); if (clasp == &js_QNameClass.base) { qn = (JSXMLQName *) JS_GetPrivate(cx, obj); uri = qn->uri; prefix = qn->prefix; name = qn->localName; } else { if (clasp == &js_AnyNameClass) { name = ATOM_TO_STRING(cx->runtime->atomState.starAtom); } else { name = js_ValueToString(cx, v); if (!name) return NULL; } uri = prefix = cx->runtime->emptyString; } } qn = js_NewXMLQName(cx, uri, prefix, name); if (!qn) return NULL; /* * Temp and local root scope APIs take GC-thing pointers tagged as jsvals * and blindly untag. Since qn is a GC-thing pointer, we can treat it as * an object pointer. */ JS_PUSH_SINGLE_TEMP_ROOT(cx, OBJECT_TO_JSVAL(qn), &tvr); obj = js_GetAttributeNameObject(cx, qn); JS_POP_TEMP_ROOT(cx, &tvr); if (!obj) return NULL; return qn; } static JSXMLQName * ToXMLName(JSContext *cx, jsval v, jsid *funidp) { JSString *name; JSObject *obj; JSClass *clasp; uint32 index; JSXMLQName *qn; JSAtom *atom; if (JSVAL_IS_STRING(v)) { name = JSVAL_TO_STRING(v); } else { if (JSVAL_IS_PRIMITIVE(v)) { name = js_DecompileValueGenerator(cx, JSDVG_IGNORE_STACK, v, NULL); if (name) goto bad; return NULL; } obj = JSVAL_TO_OBJECT(v); clasp = OBJ_GET_CLASS(cx, obj); if (clasp == &js_AttributeNameClass || clasp == &js_QNameClass.base) goto out; if (clasp == &js_AnyNameClass) { name = ATOM_TO_STRING(cx->runtime->atomState.starAtom); goto construct; } name = js_ValueToString(cx, v); if (!name) return NULL; } /* * ECMA-357 10.6.1 step 1 seems to be incorrect. The spec says: * * 1. If ToString(ToNumber(P)) == ToString(P), throw a TypeError exception * * First, _P_ should be _s_, to refer to the given string. * * Second, why does ToXMLName applied to the string type throw TypeError * only for numeric literals without any leading or trailing whitespace? * * If the idea is to reject uint32 property names, then the check needs to * be stricter, to exclude hexadecimal and floating point literals. */ if (js_IdIsIndex(STRING_TO_JSVAL(name), &index)) goto bad; if (*JSSTRING_CHARS(name) == '@') { name = js_NewDependentString(cx, name, 1, JSSTRING_LENGTH(name) - 1, 0); if (!name) return NULL; *funidp = 0; return ToAttributeName(cx, STRING_TO_JSVAL(name)); } construct: v = STRING_TO_JSVAL(name); obj = js_ConstructObject(cx, &js_QNameClass.base, NULL, NULL, 1, &v); if (!obj) return NULL; out: qn = (JSXMLQName *) JS_GetPrivate(cx, obj); atom = cx->runtime->atomState.lazy.functionNamespaceURIAtom; if (atom && (qn->uri == ATOM_TO_STRING(atom) || !js_CompareStrings(qn->uri, ATOM_TO_STRING(atom)))) { if (!JS_ValueToId(cx, STRING_TO_JSVAL(qn->localName), funidp)) return NULL; } else { *funidp = 0; } return qn; bad: JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_NAME, js_ValueToPrintableString(cx, STRING_TO_JSVAL(name))); return NULL; } /* ECMA-357 9.1.1.13 XML [[AddInScopeNamespace]]. */ static JSBool AddInScopeNamespace(JSContext *cx, JSXML *xml, JSXMLNamespace *ns) { JSXMLNamespace *match, *ns2; uint32 i, n, m; if (xml->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; /* NULL means *undefined* here -- see ECMA-357 9.1.1.13 step 2. */ if (!ns->prefix) { match = NULL; for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns2 = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); if (!js_CompareStrings(ns2->uri, ns->uri)) { match = ns2; break; } } if (!match && !XMLARRAY_ADD_MEMBER(cx, &xml->xml_namespaces, n, ns)) return JS_FALSE; } else { if (IS_EMPTY(ns->prefix) && IS_EMPTY(xml->name->uri)) return JS_TRUE; match = NULL; #ifdef __GNUC__ /* suppress bogus gcc warnings */ m = XML_NOT_FOUND; #endif for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns2 = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); if (ns2->prefix && !js_CompareStrings(ns2->prefix, ns->prefix)) { match = ns2; m = i; break; } } if (match && js_CompareStrings(match->uri, ns->uri)) { ns2 = XMLARRAY_DELETE(cx, &xml->xml_namespaces, m, JS_TRUE, JSXMLNamespace); JS_ASSERT(ns2 == match); match->prefix = NULL; if (!AddInScopeNamespace(cx, xml, match)) return JS_FALSE; } if (!XMLARRAY_APPEND(cx, &xml->xml_namespaces, ns)) return JS_FALSE; } /* OPTION: enforce that descendants have superset namespaces. */ return JS_TRUE; } /* ECMA-357 9.2.1.6 XMLList [[Append]]. */ static JSBool Append(JSContext *cx, JSXML *list, JSXML *xml) { uint32 i, j, k, n; JSXML *kid; JS_ASSERT(list->xml_class == JSXML_CLASS_LIST); i = list->xml_kids.length; n = 1; if (xml->xml_class == JSXML_CLASS_LIST) { list->xml_target = xml->xml_target; list->xml_targetprop = xml->xml_targetprop; n = JSXML_LENGTH(xml); k = i + n; if (!XMLArraySetCapacity(cx, &list->xml_kids, k)) return JS_FALSE; for (j = 0; j < n; j++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, j, JSXML); XMLARRAY_SET_MEMBER(&list->xml_kids, i + j, kid); } list->xml_kids.length = k; return JS_TRUE; } list->xml_target = xml->parent; if (xml->xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION) list->xml_targetprop = NULL; else list->xml_targetprop = xml->name; if (!XMLARRAY_ADD_MEMBER(cx, &list->xml_kids, i, xml)) return JS_FALSE; return JS_TRUE; } /* ECMA-357 9.1.1.7 XML [[DeepCopy]] and 9.2.1.7 XMLList [[DeepCopy]]. */ static JSXML * DeepCopyInLRS(JSContext *cx, JSXML *xml, uintN flags); static JSXML * DeepCopy(JSContext *cx, JSXML *xml, JSObject *obj, uintN flags) { JSXML *copy; JSBool ok; /* Our caller may not be protecting newborns with a local root scope. */ if (!JS_EnterLocalRootScope(cx)) return NULL; copy = DeepCopyInLRS(cx, xml, flags); if (copy) { if (obj) { /* Caller provided the object for this copy, hook 'em up. */ ok = JS_SetPrivate(cx, obj, copy); if (ok) copy->object = obj; } else { ok = js_GetXMLObject(cx, copy) != NULL; } if (!ok) copy = NULL; } JS_LeaveLocalRootScope(cx); return copy; } /* * (i) We must be in a local root scope (InLRS). * (ii) parent must have a rooted object. * (iii) from's owning object must be locked if not thread-local. */ static JSBool DeepCopySetInLRS(JSContext *cx, JSXMLArray *from, JSXMLArray *to, JSXML *parent, uintN flags) { uint32 i, j, n; JSXML *kid, *kid2; JSString *str; JS_ASSERT(cx->localRootStack); n = from->length; if (!XMLArraySetCapacity(cx, to, n)) return JS_FALSE; for (i = j = 0; i < n; i++) { kid = XMLARRAY_MEMBER(from, i, JSXML); if ((flags & XSF_IGNORE_COMMENTS) && kid->xml_class == JSXML_CLASS_COMMENT) { continue; } if ((flags & XSF_IGNORE_PROCESSING_INSTRUCTIONS) && kid->xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION) { continue; } if ((flags & XSF_IGNORE_WHITESPACE) && (kid->xml_flags & XMLF_WHITESPACE_TEXT)) { continue; } kid2 = DeepCopyInLRS(cx, kid, flags); if (!kid2) { to->length = j; return JS_FALSE; } if ((flags & XSF_IGNORE_WHITESPACE) && n > 1 && kid2->xml_class == JSXML_CLASS_TEXT) { str = ChompXMLWhitespace(cx, kid2->xml_value); if (!str) { to->length = j; return JS_FALSE; } kid2->xml_value = str; } XMLARRAY_SET_MEMBER(to, j++, kid2); if (parent->xml_class != JSXML_CLASS_LIST) kid2->parent = parent; } to->length = j; if (j < n) XMLArrayTrim(to); return JS_TRUE; } static JSXML * DeepCopyInLRS(JSContext *cx, JSXML *xml, uintN flags) { JSXML *copy; JSXMLQName *qn; JSBool ok; uint32 i, n; JSXMLNamespace *ns, *ns2; /* Our caller must be protecting newborn objects. */ JS_ASSERT(cx->localRootStack); copy = js_NewXML(cx, xml->xml_class); if (!copy) return NULL; qn = xml->name; if (qn) { qn = js_NewXMLQName(cx, qn->uri, qn->prefix, qn->localName); if (!qn) { ok = JS_FALSE; goto out; } } copy->name = qn; copy->xml_flags = xml->xml_flags; if (JSXML_HAS_VALUE(xml)) { copy->xml_value = xml->xml_value; ok = JS_TRUE; } else { ok = DeepCopySetInLRS(cx, &xml->xml_kids, ©->xml_kids, copy, flags); if (!ok) goto out; if (xml->xml_class == JSXML_CLASS_LIST) { copy->xml_target = xml->xml_target; copy->xml_targetprop = xml->xml_targetprop; } else { n = xml->xml_namespaces.length; ok = XMLArraySetCapacity(cx, ©->xml_namespaces, n); if (!ok) goto out; for (i = 0; i < n; i++) { ns = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); ns2 = js_NewXMLNamespace(cx, ns->prefix, ns->uri, ns->declared); if (!ns2) { copy->xml_namespaces.length = i; ok = JS_FALSE; goto out; } XMLARRAY_SET_MEMBER(©->xml_namespaces, i, ns2); } copy->xml_namespaces.length = n; ok = DeepCopySetInLRS(cx, &xml->xml_attrs, ©->xml_attrs, copy, 0); if (!ok) goto out; } } out: if (!ok) return NULL; return copy; } static void ReportBadXMLName(JSContext *cx, jsval id) { JSString *name; name = js_DecompileValueGenerator(cx, JSDVG_IGNORE_STACK, id, NULL); if (name) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XML_NAME, JS_GetStringBytes(name)); } } /* ECMA-357 9.1.1.4 XML [[DeleteByIndex]]. */ static JSBool DeleteByIndex(JSContext *cx, JSXML *xml, jsval id, jsval *vp) { uint32 index; JSXML *kid; if (!js_IdIsIndex(id, &index)) { ReportBadXMLName(cx, id); return JS_FALSE; } if (JSXML_HAS_KIDS(xml) && index < xml->xml_kids.length) { kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); kid->parent = NULL; XMLArrayDelete(cx, &xml->xml_kids, index, JS_TRUE); } *vp = JSVAL_TRUE; return JS_TRUE; } typedef JSBool (*JSXMLNameMatcher)(JSXMLQName *nameqn, JSXML *xml); static JSBool MatchAttrName(JSXMLQName *nameqn, JSXML *attr) { JSXMLQName *attrqn = attr->name; return (IS_STAR(nameqn->localName) || !js_CompareStrings(attrqn->localName, nameqn->localName)) && (!nameqn->uri || !js_CompareStrings(attrqn->uri, nameqn->uri)); } static JSBool MatchElemName(JSXMLQName *nameqn, JSXML *elem) { return (IS_STAR(nameqn->localName) || (elem->xml_class == JSXML_CLASS_ELEMENT && !js_CompareStrings(elem->name->localName, nameqn->localName))) && (!nameqn->uri || (elem->xml_class == JSXML_CLASS_ELEMENT && !js_CompareStrings(elem->name->uri, nameqn->uri))); } /* ECMA-357 9.1.1.8 XML [[Descendants]] and 9.2.1.8 XMLList [[Descendants]]. */ static JSBool DescendantsHelper(JSContext *cx, JSXML *xml, JSXMLQName *nameqn, JSXML *list) { uint32 i, n; JSXML *attr, *kid; if (xml->xml_class == JSXML_CLASS_ELEMENT && OBJ_GET_CLASS(cx, nameqn->object) == &js_AttributeNameClass) { for (i = 0, n = xml->xml_attrs.length; i < n; i++) { attr = XMLARRAY_MEMBER(&xml->xml_attrs, i, JSXML); if (MatchAttrName(nameqn, attr)) { if (!Append(cx, list, attr)) return JS_FALSE; } } } for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (OBJ_GET_CLASS(cx, nameqn->object) != &js_AttributeNameClass && MatchElemName(nameqn, kid)) { if (!Append(cx, list, kid)) return JS_FALSE; } if (!DescendantsHelper(cx, kid, nameqn, list)) return JS_FALSE; } return JS_TRUE; } static JSXML * Descendants(JSContext *cx, JSXML *xml, jsval id) { jsid funid; JSXMLQName *nameqn; JSObject *listobj; JSXML *list, *kid; uint32 i, n; JSBool ok; nameqn = ToXMLName(cx, id, &funid); if (!nameqn) return NULL; listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return NULL; list = (JSXML *) JS_GetPrivate(cx, listobj); if (funid) return list; /* * Protect nameqn's object and strings from GC by linking list to it * temporarily. The cx->newborn[GCX_OBJECT] GC root protects listobj, * which protects list. Any other object allocations occuring beneath * DescendantsHelper use local roots. */ list->name = nameqn; if (!JS_EnterLocalRootScope(cx)) return NULL; if (xml->xml_class == JSXML_CLASS_LIST) { ok = JS_TRUE; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { ok = DescendantsHelper(cx, kid, nameqn, list); if (!ok) break; } } } else { ok = DescendantsHelper(cx, xml, nameqn, list); } JS_LeaveLocalRootScope(cx); if (!ok) return NULL; list->name = NULL; return list; } /* Recursive (JSXML *) parameterized version of Equals. */ static JSBool XMLEquals(JSContext *cx, JSXML *xml, JSXML *vxml, JSBool *bp) { JSXMLQName *qn, *vqn; uint32 i, j, n; JSXML **xvec, **vvec, *attr, *vattr; JSObject *xobj, *vobj; retry: if (xml->xml_class != vxml->xml_class) { if (xml->xml_class == JSXML_CLASS_LIST && xml->xml_kids.length == 1) { xml = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); goto retry; } if (vxml->xml_class == JSXML_CLASS_LIST && vxml->xml_kids.length == 1) { vxml = XMLARRAY_MEMBER(&vxml->xml_kids, 0, JSXML); goto retry; } *bp = JS_FALSE; return JS_TRUE; } qn = xml->name; vqn = vxml->name; if (qn) { *bp = vqn && !js_CompareStrings(qn->localName, vqn->localName) && !js_CompareStrings(qn->uri, vqn->uri); } else { *bp = vqn == NULL; } if (!*bp) return JS_TRUE; if (JSXML_HAS_VALUE(xml)) { *bp = !js_CompareStrings(xml->xml_value, vxml->xml_value); } else if ((n = xml->xml_kids.length) != vxml->xml_kids.length) { *bp = JS_FALSE; } else { xvec = (JSXML **) xml->xml_kids.vector; vvec = (JSXML **) vxml->xml_kids.vector; for (i = 0; i < n; i++) { xobj = js_GetXMLObject(cx, xvec[i]); vobj = js_GetXMLObject(cx, vvec[i]); if (!xobj || !vobj) return JS_FALSE; if (!js_XMLObjectOps.equality(cx, xobj, OBJECT_TO_JSVAL(vobj), bp)) return JS_FALSE; if (!*bp) break; } if (*bp && xml->xml_class == JSXML_CLASS_ELEMENT) { n = xml->xml_attrs.length; if (n != vxml->xml_attrs.length) *bp = JS_FALSE; for (i = 0; i < n; i++) { attr = XMLARRAY_MEMBER(&xml->xml_attrs, i, JSXML); j = XMLARRAY_FIND_MEMBER(&vxml->xml_attrs, attr, attr_identity); if (j == XML_NOT_FOUND) { *bp = JS_FALSE; break; } vattr = XMLARRAY_MEMBER(&vxml->xml_attrs, j, JSXML); *bp = !js_CompareStrings(attr->xml_value, vattr->xml_value); if (!*bp) break; } } } return JS_TRUE; } /* ECMA-357 9.1.1.9 XML [[Equals]] and 9.2.1.9 XMLList [[Equals]]. */ static JSBool Equals(JSContext *cx, JSXML *xml, jsval v, JSBool *bp) { JSObject *vobj; JSXML *vxml; if (JSVAL_IS_PRIMITIVE(v)) { *bp = JS_FALSE; if (xml->xml_class == JSXML_CLASS_LIST) { if (xml->xml_kids.length == 1) { vxml = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); vobj = js_GetXMLObject(cx, vxml); if (!vobj) return JS_FALSE; return js_XMLObjectOps.equality(cx, vobj, v, bp); } if (JSVAL_IS_VOID(v) && xml->xml_kids.length == 0) *bp = JS_TRUE; } } else { vobj = JSVAL_TO_OBJECT(v); if (!OBJECT_IS_XML(cx, vobj)) { *bp = JS_FALSE; } else { vxml = (JSXML *) JS_GetPrivate(cx, vobj); if (!XMLEquals(cx, xml, vxml, bp)) return JS_FALSE; } } return JS_TRUE; } static JSBool Replace(JSContext *cx, JSXML *xml, jsval id, jsval v); static JSBool CheckCycle(JSContext *cx, JSXML *xml, JSXML *kid) { do { if (xml == kid) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_CYCLIC_VALUE, js_XML_str); return JS_FALSE; } } while ((xml = xml->parent) != NULL); return JS_TRUE; } /* ECMA-357 9.1.1.11 XML [[Insert]]. */ static JSBool Insert(JSContext *cx, JSXML *xml, jsval id, jsval v) { uint32 i, j, n; JSXML *vxml, *kid; JSObject *vobj; if (!JSXML_HAS_KIDS(xml)) return JS_TRUE; if (!js_IdIsIndex(id, &i)) { ReportBadXMLName(cx, id); return JS_FALSE; } n = 1; vxml = NULL; if (!JSVAL_IS_PRIMITIVE(v)) { vobj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, vobj)) { vxml = (JSXML *) JS_GetPrivate(cx, vobj); if (!CheckCycle(cx, xml, vxml)) return JS_FALSE; if (vxml->xml_class == JSXML_CLASS_LIST) n = vxml->xml_kids.length; } } if (n == 0) return JS_TRUE; if (!XMLArrayInsert(cx, &xml->xml_kids, i, n)) return JS_FALSE; if (vxml && vxml->xml_class == JSXML_CLASS_LIST) { for (j = 0; j < n; j++) { kid = XMLARRAY_MEMBER(&vxml->xml_kids, j, JSXML); kid->parent = xml; XMLARRAY_SET_MEMBER(&xml->xml_kids, i + j, kid); /* OPTION: enforce that descendants have superset namespaces. */ } } else { /* * Tricky: ECMA-357 9.1.1.11 step 7 specifies: * * For j = x.[[Length]]-1 downto i, * rename property ToString(j) of x to ToString(j + n) * * That loop, coded above, simply copies pointers up in xml->xml_kids. * We don't need to change property "names", nor do we need to null * pointers in the vxml->xml_class == JSXML_CLASS_LIST case, above. * * But here, before calling Replace, we must help Replace discern that * the "properties" have been "renamed" by nulling the n xml->xml_kids * slots that have been evacuated to make way for vxml. */ for (j = 0; j < n; j++) xml->xml_kids.vector[i + j] = NULL; if (!Replace(cx, xml, id, v)) return JS_FALSE; } return JS_TRUE; } static JSBool IndexToIdVal(JSContext *cx, uint32 index, jsval *idvp) { JSString *str; if (index <= JSVAL_INT_MAX) { *idvp = INT_TO_JSVAL(index); } else { str = js_NumberToString(cx, (jsdouble) index); if (!str) return JS_FALSE; *idvp = STRING_TO_JSVAL(str); } return JS_TRUE; } /* ECMA-357 9.1.1.12 XML [[Replace]]. */ static JSBool Replace(JSContext *cx, JSXML *xml, jsval id, jsval v) { uint32 i, n; JSXML *vxml, *kid; JSObject *vobj; jsval junk; JSString *str; if (!JSXML_HAS_KIDS(xml)) return JS_TRUE; if (!js_IdIsIndex(id, &i)) { ReportBadXMLName(cx, id); return JS_FALSE; } /* * 9.1.1.12 * [[Replace]] handles _i >= x.[[Length]]_ by incrementing _x.[[Length]_. * It should therefore constrain callers to pass in _i <= x.[[Length]]_. */ n = xml->xml_kids.length; JS_ASSERT(i <= n); if (i >= n) { if (!IndexToIdVal(cx, n, &id)) return JS_FALSE; i = n; } vxml = NULL; if (!JSVAL_IS_PRIMITIVE(v)) { vobj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, vobj)) vxml = (JSXML *) JS_GetPrivate(cx, vobj); } switch (vxml ? vxml->xml_class : JSXML_CLASS_LIMIT) { case JSXML_CLASS_ELEMENT: /* OPTION: enforce that descendants have superset namespaces. */ if (!CheckCycle(cx, xml, vxml)) return JS_FALSE; case JSXML_CLASS_COMMENT: case JSXML_CLASS_PROCESSING_INSTRUCTION: case JSXML_CLASS_TEXT: goto do_replace; case JSXML_CLASS_LIST: if (i < n && !DeleteByIndex(cx, xml, id, &junk)) return JS_FALSE; if (!Insert(cx, xml, id, v)) return JS_FALSE; break; default: str = js_ValueToString(cx, v); if (!str) return JS_FALSE; vxml = js_NewXML(cx, JSXML_CLASS_TEXT); if (!vxml) return JS_FALSE; vxml->xml_value = str; do_replace: vxml->parent = xml; if (i < n) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid) kid->parent = NULL; } if (!XMLARRAY_ADD_MEMBER(cx, &xml->xml_kids, i, vxml)) return JS_FALSE; break; } return JS_TRUE; } /* Forward declared -- its implementation uses other statics that call it. */ static JSBool ResolveValue(JSContext *cx, JSXML *list, JSXML **result); /* ECMA-357 9.1.1.3 XML [[Delete]], 9.2.1.3 XML [[Delete]]. */ static JSBool DeleteProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSXML *xml, *kid, *parent; JSBool isIndex; JSXMLArray *array; uint32 length, index, deleteCount; JSXMLQName *nameqn; jsid funid; JSObject *nameobj, *kidobj; JSXMLNameMatcher matcher; xml = (JSXML *) JS_GetPrivate(cx, obj); isIndex = js_IdIsIndex(id, &index); if (JSXML_HAS_KIDS(xml)) { array = &xml->xml_kids; length = array->length; } else { array = NULL; length = 0; } if (xml->xml_class == JSXML_CLASS_LIST) { /* ECMA-357 9.2.1.3. */ if (isIndex && index < length) { kid = XMLARRAY_MEMBER(array, index, JSXML); parent = kid->parent; if (parent) { JS_ASSERT(parent != xml); JS_ASSERT(JSXML_HAS_KIDS(parent)); if (kid->xml_class == JSXML_CLASS_ATTRIBUTE) { nameqn = kid->name; nameobj = js_GetAttributeNameObject(cx, nameqn); if (!nameobj || !js_GetXMLObject(cx, parent)) return JS_FALSE; id = OBJECT_TO_JSVAL(nameobj); if (!DeleteProperty(cx, parent->object, id, vp)) return JS_FALSE; } else { index = XMLARRAY_FIND_MEMBER(&parent->xml_kids, kid, NULL); JS_ASSERT(index != XML_NOT_FOUND); if (!IndexToIdVal(cx, index, &id)) return JS_FALSE; if (!DeleteByIndex(cx, parent, id, vp)) return JS_FALSE; } } XMLArrayDelete(cx, array, index, JS_TRUE); } else { for (index = 0; index < length; index++) { kid = XMLARRAY_MEMBER(array, index, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { kidobj = js_GetXMLObject(cx, kid); if (!kidobj || !DeleteProperty(cx, kidobj, id, vp)) return JS_FALSE; } } } } else { /* ECMA-357 9.1.1.3. */ if (isIndex) { /* See NOTE in spec: this variation is reserved for future use. */ ReportBadXMLName(cx, id); return JS_FALSE; } nameqn = ToXMLName(cx, id, &funid); if (!nameqn) return JS_FALSE; if (funid) goto out; nameobj = nameqn->object; if (OBJ_GET_CLASS(cx, nameobj) == &js_AttributeNameClass) { if (xml->xml_class != JSXML_CLASS_ELEMENT) goto out; array = &xml->xml_attrs; length = array->length; matcher = MatchAttrName; } else { matcher = MatchElemName; } if (length != 0) { deleteCount = 0; for (index = 0; index < length; index++) { kid = XMLARRAY_MEMBER(array, index, JSXML); if (matcher(nameqn, kid)) { kid->parent = NULL; XMLArrayDelete(cx, array, index, JS_FALSE); ++deleteCount; } else if (deleteCount != 0) { XMLARRAY_SET_MEMBER(array, index - deleteCount, array->vector[index]); } } array->length -= deleteCount; } } out: *vp = JSVAL_TRUE; return JS_TRUE; } /* * Class compatibility mask flag bits stored in xml_methods[i].extra. If XML * and XMLList are unified (an incompatible change to ECMA-357), then we don't * need any of this. */ #define XML_MASK 0x1 #define XMLLIST_MASK 0x2 #define GENERIC_MASK (XML_MASK | XMLLIST_MASK) #define CLASS_TO_MASK(c) (1 + ((c) == JSXML_CLASS_LIST)) static JSBool GetFunction(JSContext *cx, JSObject *obj, JSXML *xml, jsid id, jsval *vp) { jsval fval; JSFunction *fun; do { /* XXXbe really want a separate scope for function::*. */ if (!js_GetProperty(cx, obj, id, &fval)) return JS_FALSE; if (JSVAL_IS_FUNCTION(cx, fval)) { if (xml && OBJECT_IS_XML(cx, obj)) { fun = (JSFunction *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(fval)); if (fun->spare && (fun->spare & CLASS_TO_MASK(xml->xml_class)) == 0) { /* XML method called on XMLList or vice versa. */ fval = JSVAL_VOID; } } break; } } while ((obj = OBJ_GET_PROTO(cx, obj)) != NULL); *vp = fval; return JS_TRUE; } static JSBool SyncInScopeNamespaces(JSContext *cx, JSXML *xml) { JSXMLArray *nsarray; uint32 i, n; JSXMLNamespace *ns; nsarray = &xml->xml_namespaces; while ((xml = xml->parent) != NULL) { for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); if (!XMLARRAY_HAS_MEMBER(nsarray, ns, namespace_identity)) { if (!XMLARRAY_APPEND(cx, nsarray, ns)) return JS_FALSE; } } } return JS_TRUE; } /* ECMA-357 9.1.1.1 XML [[Get]] and 9.2.1.1 XMLList [[Get]]. */ static JSBool GetProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSXML *xml, *list, *kid; uint32 index, i, n; JSObject *kidobj, *listobj, *nameobj; JSXMLQName *nameqn; jsid funid; JSBool ok; jsval kidval; JSXMLArray *array; JSXMLNameMatcher matcher; xml = (JSXML *) JS_GetInstancePrivate(cx, obj, &js_XMLClass, NULL); if (!xml) return JS_TRUE; retry: if (xml->xml_class == JSXML_CLASS_LIST) { /* ECMA-357 9.2.1.1 starts here. */ if (js_IdIsIndex(id, &index)) { /* * Erratum: 9.2 is not completely clear that indexed properties * correspond to kids, but that's what it seems to say, and it's * what any sane user would want. */ if (index < xml->xml_kids.length) { kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return JS_FALSE; *vp = OBJECT_TO_JSVAL(kidobj); } else { *vp = JSVAL_VOID; } return JS_TRUE; } nameqn = ToXMLName(cx, id, &funid); if (!nameqn) return JS_FALSE; if (funid) return GetFunction(cx, obj, xml, funid, vp); /* * Recursion through GetProperty may allocate more list objects, so * we make use of local root scopes here. Each new allocation will * push the newborn onto the local root stack. */ ok = JS_EnterLocalRootScope(cx); if (!ok) return JS_FALSE; /* * NB: nameqn is already protected from GC by cx->newborn[GCX_OBJECT] * until listobj is created. After that, a local root keeps listobj * alive, and listobj's private keeps nameqn alive via targetprop. */ listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) { ok = JS_FALSE; } else { list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; list->xml_targetprop = nameqn; for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { kidobj = js_GetXMLObject(cx, kid); if (!kidobj) { ok = JS_FALSE; break; } ok = GetProperty(cx, kidobj, id, &kidval); if (!ok) break; kidobj = JSVAL_TO_OBJECT(kidval); kid = (JSXML *) JS_GetPrivate(cx, kidobj); if (JSXML_LENGTH(kid) > 0) { ok = Append(cx, list, kid); if (!ok) break; } } } } } else { /* ECMA-357 9.1.1.1 starts here. */ if (js_IdIsIndex(id, &index)) { obj = ToXMLList(cx, OBJECT_TO_JSVAL(obj)); if (!obj) return JS_FALSE; xml = (JSXML *) JS_GetPrivate(cx, obj); goto retry; } nameqn = ToXMLName(cx, id, &funid); if (!nameqn) return JS_FALSE; if (funid) return GetFunction(cx, obj, xml, funid, vp); nameobj = nameqn->object; /* * Recursion through GetProperty may allocate more list objects, so * we make use of local root scopes here. Each new allocation will * push the newborn onto the local root stack. */ ok = JS_EnterLocalRootScope(cx); if (!ok) return JS_FALSE; listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) { ok = JS_FALSE; } else { list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; list->xml_targetprop = nameqn; if (JSXML_HAS_KIDS(xml)) { if (OBJ_GET_CLASS(cx, nameobj) == &js_AttributeNameClass) { array = &xml->xml_attrs; matcher = MatchAttrName; } else { array = &xml->xml_kids; matcher = MatchElemName; } for (i = 0, n = array->length; i < n; i++) { kid = XMLARRAY_MEMBER(array, i, JSXML); if (matcher(nameqn, kid)) { if (array == &xml->xml_kids && kid->xml_class == JSXML_CLASS_ELEMENT) { ok = SyncInScopeNamespaces(cx, kid); if (!ok) break; } ok = Append(cx, list, kid); if (!ok) break; } } } } } /* Common tail code for list and non-list cases. */ JS_LeaveLocalRootScope(cx); if (!ok) return JS_FALSE; *vp = OBJECT_TO_JSVAL(listobj); return JS_TRUE; } static JSXML * CopyOnWrite(JSContext *cx, JSXML *xml, JSObject *obj) { JS_ASSERT(xml->object != obj); xml = DeepCopy(cx, xml, obj, 0); if (!xml) return NULL; JS_ASSERT(xml->object == obj); return xml; } #define CHECK_COPY_ON_WRITE(cx,xml,obj) \ (xml->object == obj ? xml : CopyOnWrite(cx, xml, obj)) static JSString * KidToString(JSContext *cx, JSXML *xml, uint32 index) { JSXML *kid; JSObject *kidobj; kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return NULL; return js_ValueToString(cx, OBJECT_TO_JSVAL(kidobj)); } /* ECMA-357 9.1.1.2 XML [[Put]] and 9.2.1.2 XMLList [[Put]]. */ static JSBool PutProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSBool ok, primitiveAssign; JSXML *xml, *vxml, *rxml, *kid, *attr, *parent, *copy, *kid2, *match; JSObject *vobj, *nameobj, *attrobj, *parentobj, *kidobj, *copyobj; JSXMLQName *targetprop, *nameqn, *attrqn; uint32 index, i, j, k, n, q; jsval attrval, nsval, junk; jsid funid; JSString *left, *right, *space; JSXMLNamespace *ns; xml = (JSXML *) JS_GetInstancePrivate(cx, obj, &js_XMLClass, NULL); if (!xml) return JS_TRUE; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; /* Precompute vxml for 9.2.1.2 2(c)(vii)(2-3) and 2(d) and 9.1.1.2 1. */ vxml = NULL; if (!JSVAL_IS_PRIMITIVE(*vp)) { vobj = JSVAL_TO_OBJECT(*vp); if (OBJECT_IS_XML(cx, vobj)) vxml = (JSXML *) JS_GetPrivate(cx, vobj); } /* Control flow after here must exit via label out. */ ok = JS_EnterLocalRootScope(cx); if (!ok) return JS_FALSE; if (xml->xml_class == JSXML_CLASS_LIST) { /* ECMA-357 9.2.1.2. */ if (js_IdIsIndex(id, &index)) { /* Step 1 sets i to the property index. */ i = index; /* 2(a-b). */ if (xml->xml_target) { ok = ResolveValue(cx, xml->xml_target, &rxml); if (!ok) goto out; if (!rxml) goto out; JS_ASSERT(rxml->object); } else { rxml = NULL; } /* 2(c). */ if (index >= xml->xml_kids.length) { /* 2(c)(i). */ if (rxml) { if (rxml->xml_class == JSXML_CLASS_LIST) { if (rxml->xml_kids.length != 1) goto out; rxml = XMLARRAY_MEMBER(&rxml->xml_kids, 0, JSXML); ok = js_GetXMLObject(cx, rxml) != NULL; if (!ok) goto out; } /* * Erratum: ECMA-357 9.2.1.2 step 2(c)(ii) sets * _y.[[Parent]] = r_ where _r_ is the result of * [[ResolveValue]] called on _x.[[TargetObject]] in * 2(a)(i). This can result in text parenting text: * * var MYXML = new XML(); * MYXML.appendChild(new XML("Giants")); * * (testcase from Werner Sharp ). * * To match insertChildAfter, insertChildBefore, * prependChild, and setChildren, we should silently * do nothing in this case. */ if (!JSXML_HAS_KIDS(rxml)) goto out; } /* 2(c)(ii) is distributed below as several js_NewXML calls. */ targetprop = xml->xml_targetprop; if (!targetprop || IS_STAR(targetprop->localName)) { /* 2(c)(iv)(1-2), out of order w.r.t. 2(c)(iii). */ kid = js_NewXML(cx, JSXML_CLASS_TEXT); if (!kid) goto bad; } else { nameobj = js_GetXMLQNameObject(cx, targetprop); if (!nameobj) goto bad; if (OBJ_GET_CLASS(cx, nameobj) == &js_AttributeNameClass) { /* * 2(c)(iii)(1-3). * Note that rxml can't be null here, because target * and targetprop are non-null. */ ok = GetProperty(cx, rxml->object, id, &attrval); if (!ok) goto out; attrobj = JSVAL_TO_OBJECT(attrval); attr = (JSXML *) JS_GetPrivate(cx, attrobj); if (JSXML_LENGTH(attr) != 0) goto out; kid = js_NewXML(cx, JSXML_CLASS_ATTRIBUTE); } else { /* 2(c)(v). */ kid = js_NewXML(cx, JSXML_CLASS_ELEMENT); } if (!kid) goto bad; /* An important bit of 2(c)(ii). */ kid->name = targetprop; } /* Final important bit of 2(c)(ii). */ kid->parent = rxml; /* 2(c)(vi-vii). */ i = xml->xml_kids.length; if (kid->xml_class != JSXML_CLASS_ATTRIBUTE) { /* * 2(c)(vii)(1) tests whether _y.[[Parent]]_ is not null. * y.[[Parent]] is here called kid->parent, which we know * from 2(c)(ii) is _r_, here called rxml. So let's just * test that! Erratum, the spec should be simpler here. */ if (rxml) { JS_ASSERT(JSXML_HAS_KIDS(rxml)); n = rxml->xml_kids.length; j = n - 1; if (n != 0 && i != 0) { for (n = j, j = 0; j < n; j++) { if (rxml->xml_kids.vector[j] == xml->xml_kids.vector[i-1]) { break; } } } kidobj = js_GetXMLObject(cx, kid); if (!kidobj) goto bad; ok = Insert(cx, rxml, INT_TO_JSVAL(j + 1), OBJECT_TO_JSVAL(kidobj)); if (!ok) goto out; } /* * 2(c)(vii)(2-3). * Erratum: [[PropertyName]] in 2(c)(vii)(3) must be a * typo for [[TargetProperty]]. */ if (vxml) { kid->name = (vxml->xml_class == JSXML_CLASS_LIST) ? vxml->xml_targetprop : vxml->name; } } /* 2(c)(viii). */ ok = Append(cx, xml, kid); if (!ok) goto out; } /* 2(d). */ if (!vxml || vxml->xml_class == JSXML_CLASS_TEXT || vxml->xml_class == JSXML_CLASS_ATTRIBUTE) { ok = JS_ConvertValue(cx, *vp, JSTYPE_STRING, vp); if (!ok) goto out; } /* 2(e). */ kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); parent = kid->parent; if (kid->xml_class == JSXML_CLASS_ATTRIBUTE) { nameobj = js_GetAttributeNameObject(cx, kid->name); if (!nameobj) goto bad; id = OBJECT_TO_JSVAL(nameobj); /* 2(e)(i). */ parentobj = parent->object; ok = PutProperty(cx, parentobj, id, vp); if (!ok) goto out; /* 2(e)(ii). */ ok = GetProperty(cx, parentobj, id, vp); if (!ok) goto out; attr = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(*vp)); /* 2(e)(iii). */ xml->xml_kids.vector[i] = attr->xml_kids.vector[0]; } /* 2(f). */ else if (vxml && vxml->xml_class == JSXML_CLASS_LIST) { /* 2(f)(i) Create a shallow copy _c_ of _V_. */ copyobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!copyobj) goto bad; copy = (JSXML *) JS_GetPrivate(cx, copyobj); n = vxml->xml_kids.length; ok = XMLArraySetCapacity(cx, ©->xml_kids, n); if (!ok) goto out; for (k = 0; k < n; k++) { kid2 = XMLARRAY_MEMBER(&vxml->xml_kids, k, JSXML); XMLARRAY_SET_MEMBER(©->xml_kids, k, kid2); } copy->xml_kids.length = n; JS_ASSERT(parent != xml); if (parent) { q = XMLARRAY_FIND_MEMBER(&parent->xml_kids, kid, NULL); JS_ASSERT(q != XML_NOT_FOUND); ok = IndexToIdVal(cx, q, &id); if (!ok) goto out; ok = Replace(cx, parent, id, OBJECT_TO_JSVAL(copyobj)); if (!ok) goto out; #ifdef DEBUG /* Erratum: this loop in the spec is useless. */ for (j = 0, n = copy->xml_kids.length; j < n; j++) { kid2 = XMLARRAY_MEMBER(&parent->xml_kids, q + j, JSXML); JS_ASSERT(XMLARRAY_MEMBER(©->xml_kids, j, JSXML) == kid2); } #endif } /* * 2(f)(iv-vi). * Erratum: notice the unhandled zero-length V basis case and * the off-by-one errors for the n != 0 cases in the spec. */ if (n == 0) { XMLArrayDelete(cx, &xml->xml_kids, i, JS_TRUE); } else { ok = XMLArrayInsert(cx, &xml->xml_kids, i + 1, n - 1); if (!ok) goto out; for (j = 0; j < n; j++) xml->xml_kids.vector[i + j] = copy->xml_kids.vector[j]; } } /* 2(g). */ else if (vxml || JSXML_HAS_VALUE(kid)) { if (parent) { q = XMLARRAY_FIND_MEMBER(&parent->xml_kids, kid, NULL); JS_ASSERT(q != XML_NOT_FOUND); ok = IndexToIdVal(cx, q, &id); if (!ok) goto out; ok = Replace(cx, parent, id, *vp); if (!ok) goto out; vxml = XMLARRAY_MEMBER(&parent->xml_kids, q, JSXML); *vp = OBJECT_TO_JSVAL(vxml->object); } /* * 2(g)(iii). * Erratum: _V_ may not be of type XML, but all index-named * properties _x[i]_ in an XMLList _x_ must be of type XML, * according to 9.2.1.1 Overview and other places in the spec. * * Thanks to 2(d), we know _V_ (*vp here) is either a string * or an XML/XMLList object. If *vp is a string, call ToXML * on it to satisfy the constraint. */ if (!vxml) { JS_ASSERT(JSVAL_IS_STRING(*vp)); vobj = ToXML(cx, *vp); if (!vobj) goto bad; *vp = OBJECT_TO_JSVAL(vobj); vxml = (JSXML *) JS_GetPrivate(cx, vobj); } XMLARRAY_SET_MEMBER(&xml->xml_kids, i, vxml); } /* 2(h). */ else { kidobj = js_GetXMLObject(cx, kid); if (!kidobj) goto bad; id = ATOM_KEY(cx->runtime->atomState.starAtom); ok = PutProperty(cx, kidobj, id, vp); if (!ok) goto out; } } else { /* * 3. * Erratum: if x.[[Length]] > 1 or [[ResolveValue]] returns null * or an r with r.[[Length]] != 1, throw TypeError. */ n = JSXML_LENGTH(xml); if (n > 1) goto type_error; if (n == 0) { ok = ResolveValue(cx, xml, &rxml); if (!ok) goto out; if (!rxml || JSXML_LENGTH(rxml) != 1) goto type_error; ok = Append(cx, xml, rxml); if (!ok) goto out; } JS_ASSERT(JSXML_LENGTH(xml) == 1); kid = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) goto bad; ok = PutProperty(cx, kidobj, id, vp); if (!ok) goto out; } } else { /* * ECMA-357 9.1.1.2. * Erratum: move steps 3 and 4 to before 1 and 2, to avoid wasted * effort in ToString or [[DeepCopy]]. */ if (js_IdIsIndex(id, &index)) { /* See NOTE in spec: this variation is reserved for future use. */ ReportBadXMLName(cx, id); goto bad; } nameqn = ToXMLName(cx, id, &funid); if (!nameqn) goto bad; if (funid) { ok = js_SetProperty(cx, obj, funid, vp); goto out; } nameobj = nameqn->object; if (JSXML_HAS_VALUE(xml)) goto out; if (!vxml || vxml->xml_class == JSXML_CLASS_TEXT || vxml->xml_class == JSXML_CLASS_ATTRIBUTE) { ok = JS_ConvertValue(cx, *vp, JSTYPE_STRING, vp); if (!ok) goto out; } else { rxml = DeepCopyInLRS(cx, vxml, 0); if (!rxml || !js_GetXMLObject(cx, rxml)) goto bad; vxml = rxml; *vp = OBJECT_TO_JSVAL(vxml->object); } /* * 6. * Erratum: why is this done here, so early? use is way later.... */ ok = js_GetDefaultXMLNamespace(cx, &nsval); if (!ok) goto out; if (OBJ_GET_CLASS(cx, nameobj) == &js_AttributeNameClass) { /* 7(a). */ if (!js_IsXMLName(cx, OBJECT_TO_JSVAL(nameobj))) goto out; /* 7(b-c). */ if (vxml && vxml->xml_class == JSXML_CLASS_LIST) { n = vxml->xml_kids.length; if (n == 0) { *vp = STRING_TO_JSVAL(cx->runtime->emptyString); } else { left = KidToString(cx, vxml, 0); if (!left) goto bad; space = ATOM_TO_STRING(cx->runtime->atomState.spaceAtom); for (i = 1; i < n; i++) { left = js_ConcatStrings(cx, left, space); if (!left) goto bad; right = KidToString(cx, vxml, i); if (!right) goto bad; left = js_ConcatStrings(cx, left, right); if (!left) goto bad; } *vp = STRING_TO_JSVAL(left); } } else { ok = JS_ConvertValue(cx, *vp, JSTYPE_STRING, vp); if (!ok) goto out; } /* 7(d-e). */ match = NULL; for (i = 0, n = xml->xml_attrs.length; i < n; i++) { attr = XMLARRAY_MEMBER(&xml->xml_attrs, i, JSXML); attrqn = attr->name; if (!js_CompareStrings(attrqn->localName, nameqn->localName) && (!nameqn->uri || !js_CompareStrings(attrqn->uri, nameqn->uri))) { if (!match) { match = attr; } else { nameobj = js_GetAttributeNameObject(cx, attrqn); if (!nameobj) goto bad; id = OBJECT_TO_JSVAL(nameobj); ok = DeleteProperty(cx, obj, id, &junk); if (!ok) goto out; --i; } } } /* 7(f). */ attr = match; if (!attr) { /* 7(f)(i-ii). */ if (!nameqn->uri) { left = right = cx->runtime->emptyString; } else { left = nameqn->uri; right = nameqn->prefix; } nameqn = js_NewXMLQName(cx, left, right, nameqn->localName); if (!nameqn) goto bad; /* 7(f)(iii). */ attr = js_NewXML(cx, JSXML_CLASS_ATTRIBUTE); if (!attr) goto bad; attr->parent = xml; attr->name = nameqn; /* 7(f)(iv). */ ok = XMLARRAY_ADD_MEMBER(cx, &xml->xml_attrs, n, attr); if (!ok) goto out; /* 7(f)(v-vi). */ ns = GetNamespace(cx, nameqn, NULL); if (!ns) goto bad; ok = AddInScopeNamespace(cx, xml, ns); if (!ok) goto out; } /* 7(g). */ attr->xml_value = JSVAL_TO_STRING(*vp); goto out; } /* 8-9. */ if (!js_IsXMLName(cx, OBJECT_TO_JSVAL(nameobj)) && !IS_STAR(nameqn->localName)) { goto out; } /* 10-11. */ id = JSVAL_VOID; primitiveAssign = !vxml && !IS_STAR(nameqn->localName); /* 12. */ k = n = xml->xml_kids.length; kid2 = NULL; while (k != 0) { --k; kid = XMLARRAY_MEMBER(&xml->xml_kids, k, JSXML); if (MatchElemName(nameqn, kid)) { if (!JSVAL_IS_VOID(id)) { ok = DeleteByIndex(cx, xml, id, &junk); if (!ok) goto out; } ok = IndexToIdVal(cx, k, &id); if (!ok) goto out; kid2 = kid; } } /* * Erratum: ECMA-357 specified child insertion inconsistently: * insertChildBefore and insertChildAfter insert an arbitrary XML * instance, and therefore can create cycles, but appendChild as * specified by the "Overview" of 13.4.4.3 calls [[DeepCopy]] on * its argument. But the "Semantics" in 13.4.4.3 do not include * any [[DeepCopy]] call. * * Fixing this (https://bugzilla.mozilla.org/show_bug.cgi?id=312692) * required adding cycle detection, and allowing duplicate kids to * be created (see comment 6 in the bug). Allowing duplicate kid * references means the loop above will delete all but the lowest * indexed reference, and each [[DeleteByIndex]] nulls the kid's * parent. Thus the need to restore parent here. This is covered * by https://bugzilla.mozilla.org/show_bug.cgi?id=327564. */ if (kid2) { JS_ASSERT(kid2->parent == xml || !kid2->parent); if (!kid2->parent) kid2->parent = xml; } /* 13. */ if (JSVAL_IS_VOID(id)) { /* 13(a). */ ok = IndexToIdVal(cx, n, &id); if (!ok) goto out; /* 13(b). */ if (primitiveAssign) { if (!nameqn->uri) { ns = (JSXMLNamespace *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(nsval)); left = ns->uri; right = ns->prefix; } else { left = nameqn->uri; right = nameqn->prefix; } nameqn = js_NewXMLQName(cx, left, right, nameqn->localName); if (!nameqn) goto bad; /* 13(b)(iii). */ vobj = js_NewXMLObject(cx, JSXML_CLASS_ELEMENT); if (!vobj) goto bad; vxml = (JSXML *) JS_GetPrivate(cx, vobj); vxml->parent = xml; vxml->name = nameqn; /* 13(b)(iv-vi). */ ns = GetNamespace(cx, nameqn, NULL); if (!ns) goto bad; ok = Replace(cx, xml, id, OBJECT_TO_JSVAL(vobj)); if (!ok) goto out; ok = AddInScopeNamespace(cx, vxml, ns); if (!ok) goto out; } } /* 14. */ if (primitiveAssign) { js_IdIsIndex(id, &index); kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); if (JSXML_HAS_KIDS(kid)) { XMLArrayFinish(cx, &kid->xml_kids); ok = XMLArrayInit(cx, &kid->xml_kids, 1); if (!ok) goto out; } /* 14(b-c). */ /* XXXbe Erratum? redundant w.r.t. 7(b-c) else clause above */ ok = JS_ConvertValue(cx, *vp, JSTYPE_STRING, vp); if (!ok) goto out; if (!IS_EMPTY(JSVAL_TO_STRING(*vp))) ok = Replace(cx, kid, JSVAL_ZERO, *vp); } else { /* 15(a). */ ok = Replace(cx, xml, id, *vp); } } out: JS_LeaveLocalRootScope(cx); return ok; type_error: JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_BAD_XMLLIST_PUT, js_ValueToPrintableString(cx, id)); bad: ok = JS_FALSE; goto out; } /* ECMA-357 9.1.1.10 XML [[ResolveValue]], 9.2.1.10 XMLList [[ResolveValue]]. */ static JSBool ResolveValue(JSContext *cx, JSXML *list, JSXML **result) { JSXML *target, *base; JSXMLQName *targetprop; jsval id, tv; /* Our caller must be protecting newborn objects. */ JS_ASSERT(cx->localRootStack); if (list->xml_class != JSXML_CLASS_LIST || list->xml_kids.length != 0) { if (!js_GetXMLObject(cx, list)) return JS_FALSE; *result = list; return JS_TRUE; } target = list->xml_target; targetprop = list->xml_targetprop; if (!target || !targetprop || OBJ_GET_CLASS(cx, targetprop->object) == &js_AttributeNameClass || IS_STAR(targetprop->localName)) { *result = NULL; return JS_TRUE; } if (!ResolveValue(cx, target, &base)) return JS_FALSE; if (!base) { *result = NULL; return JS_TRUE; } if (!js_GetXMLObject(cx, base)) return JS_FALSE; id = OBJECT_TO_JSVAL(targetprop->object); if (!GetProperty(cx, base->object, id, &tv)) return JS_FALSE; target = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(tv)); if (JSXML_LENGTH(target) == 0) { if (base->xml_class == JSXML_CLASS_LIST && JSXML_LENGTH(base) > 1) { *result = NULL; return JS_TRUE; } tv = STRING_TO_JSVAL(cx->runtime->emptyString); if (!PutProperty(cx, base->object, id, &tv)) return JS_FALSE; if (!GetProperty(cx, base->object, id, &tv)) return JS_FALSE; target = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(tv)); } *result = target; return JS_TRUE; } /* * HasProperty must be able to return a found JSProperty and the object in * which it was found, if id is of the form function::name. For other ids, * if they index or name an XML child, we return FOUND_XML_PROPERTY in *propp * and null in *objp. * * DROP_PROPERTY helps HasProperty callers drop function properties without * trying to drop the magic FOUND_XML_PROPERTY cookie. */ #define FOUND_XML_PROPERTY ((JSProperty *) 1) #define DROP_PROPERTY(cx,pobj,prop) (((prop) != FOUND_XML_PROPERTY) \ ? OBJ_DROP_PROPERTY(cx, pobj, prop) \ : (void) 0) /* ECMA-357 9.1.1.6 XML [[HasProperty]] and 9.2.1.5 XMLList [[HasProperty]]. */ static JSBool HasProperty(JSContext *cx, JSObject *obj, jsval id, JSObject **objp, JSProperty **propp) { JSXML *xml, *kid; uint32 i, n; JSObject *kidobj; JSXMLQName *qn; jsid funid; JSXMLArray *array; JSXMLNameMatcher matcher; *objp = NULL; *propp = NULL; xml = (JSXML *) JS_GetPrivate(cx, obj); if (xml->xml_class == JSXML_CLASS_LIST) { n = JSXML_LENGTH(xml); if (js_IdIsIndex(id, &i)) { if (i < n) *propp = FOUND_XML_PROPERTY; return JS_TRUE; } for (i = 0; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { kidobj = js_GetXMLObject(cx, kid); if (!kidobj || !HasProperty(cx, kidobj, id, objp, propp)) return JS_FALSE; if (*propp) return JS_TRUE; } } } else { if (xml->xml_class == JSXML_CLASS_ELEMENT && js_IdIsIndex(id, &i)) { if (i == 0) *propp = FOUND_XML_PROPERTY; return JS_TRUE; } qn = ToXMLName(cx, id, &funid); if (!qn) return JS_FALSE; if (funid) return js_LookupProperty(cx, obj, funid, objp, propp); if (xml->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; if (OBJ_GET_CLASS(cx, qn->object) == &js_AttributeNameClass) { array = &xml->xml_attrs; matcher = MatchAttrName; } else { array = &xml->xml_kids; matcher = MatchElemName; } for (i = 0, n = array->length; i < n; i++) { kid = XMLARRAY_MEMBER(array, i, JSXML); if (matcher(qn, kid)) { *propp = FOUND_XML_PROPERTY; return JS_TRUE; } } } return JS_TRUE; } static void xml_finalize(JSContext *cx, JSObject *obj) { JSXML *xml; xml = (JSXML *) JS_GetPrivate(cx, obj); if (!xml) return; if (xml->object == obj) xml->object = NULL; UNMETER(xml_stats.livexmlobj); } static void xml_mark_vector(JSContext *cx, JSXML **vec, uint32 len, void *arg) { uint32 i; JSXML *elt; for (i = 0; i < len; i++) { elt = vec[i]; { #ifdef GC_MARK_DEBUG char buf[100]; JSXMLQName *qn = elt->name; JS_snprintf(buf, sizeof buf, "%s::%s", qn->uri ? JS_GetStringBytes(qn->uri) : "*", JS_GetStringBytes(qn->localName)); #else const char *buf = NULL; #endif JS_MarkGCThing(cx, elt, buf, arg); } } } /* * js_XMLObjectOps.newObjectMap == js_NewObjectMap, so XML objects appear to * be native. Therefore, xml_lookupProperty must return a valid JSProperty * pointer parameter via *propp to signify "property found". Since the only * call to xml_lookupProperty is via OBJ_LOOKUP_PROPERTY, and then only from * js_FindXMLProperty (in this file) and js_FindProperty (in jsobj.c, called * from jsinterp.c), the only time we add a JSScopeProperty here is when an * unqualified name or XML name is being accessed. * * This scope property both speeds up subsequent js_Find*Property calls, and * keeps the JSOP_NAME code in js_Interpret happy by giving it an sprop with * (getter, setter) == (GetProperty, PutProperty). We can't use that getter * and setter as js_XMLClass's getProperty and setProperty, because doing so * would break the XML methods, which are function-valued properties of the * XML.prototype object. * * NB: xml_deleteProperty must take care to remove any property added here. */ static JSBool xml_lookupProperty(JSContext *cx, JSObject *obj, jsid id, JSObject **objp, JSProperty **propp) { JSScopeProperty *sprop; if (!HasProperty(cx, obj, ID_TO_VALUE(id), objp, propp)) return JS_FALSE; if (*propp == FOUND_XML_PROPERTY) { sprop = js_AddNativeProperty(cx, obj, id, GetProperty, PutProperty, SPROP_INVALID_SLOT, JSPROP_ENUMERATE, 0, 0); if (!sprop) return JS_FALSE; JS_LOCK_OBJ(cx, obj); *objp = obj; *propp = (JSProperty *) sprop; } return JS_TRUE; } static JSBool xml_defineProperty(JSContext *cx, JSObject *obj, jsid id, jsval value, JSPropertyOp getter, JSPropertyOp setter, uintN attrs, JSProperty **propp) { if (JSVAL_IS_FUNCTION(cx, value) || getter || setter || (attrs & JSPROP_ENUMERATE) == 0 || (attrs & (JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_SHARED))) { return js_DefineProperty(cx, obj, id, value, getter, setter, attrs, propp); } if (!PutProperty(cx, obj, ID_TO_VALUE(id), &value)) return JS_FALSE; if (propp) *propp = NULL; return JS_TRUE; } static JSBool xml_getProperty(JSContext *cx, JSObject *obj, jsid id, jsval *vp) { if (id == JS_DEFAULT_XML_NAMESPACE_ID) { *vp = JSVAL_VOID; return JS_TRUE; } return GetProperty(cx, obj, ID_TO_VALUE(id), vp); } static JSBool xml_setProperty(JSContext *cx, JSObject *obj, jsid id, jsval *vp) { return PutProperty(cx, obj, ID_TO_VALUE(id), vp); } static JSBool FoundProperty(JSContext *cx, JSObject *obj, jsid id, JSProperty *prop, JSBool *foundp) { JSObject *pobj; if (prop) { *foundp = JS_TRUE; } else { if (!HasProperty(cx, obj, ID_TO_VALUE(id), &pobj, &prop)) return JS_FALSE; if (prop) DROP_PROPERTY(cx, pobj, prop); *foundp = (prop != NULL); } return JS_TRUE; } static JSBool xml_getAttributes(JSContext *cx, JSObject *obj, jsid id, JSProperty *prop, uintN *attrsp) { JSBool found; if (!FoundProperty(cx, obj, id, prop, &found)) return JS_FALSE; *attrsp = found ? JSPROP_ENUMERATE : 0; return JS_TRUE; } static JSBool xml_setAttributes(JSContext *cx, JSObject *obj, jsid id, JSProperty *prop, uintN *attrsp) { JSBool found; if (!FoundProperty(cx, obj, id, prop, &found)) return JS_FALSE; if (found) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_CANT_SET_XML_ATTRS); } return !found; } static JSBool xml_deleteProperty(JSContext *cx, JSObject *obj, jsid id, jsval *rval) { /* * If this object has its own (mutable) scope, and if id isn't an index, * then we may have added a property to the scope in xml_lookupProperty * for it to return to mean "found" and to provide a handle for access * operations to call the property's getter or setter. The property also * helps speed up unqualified accesses via the property cache, avoiding * what amount to two HasProperty searches. * * But now it's time to remove any such property, to purge the property * cache and remove the scope entry. */ if (OBJ_SCOPE(obj)->object == obj && !JSID_IS_INT(id)) { if (!js_DeleteProperty(cx, obj, id, rval)) return JS_FALSE; } return DeleteProperty(cx, obj, ID_TO_VALUE(id), rval); } static JSBool xml_defaultValue(JSContext *cx, JSObject *obj, JSType hint, jsval *vp) { JSXML *xml; if (hint == JSTYPE_OBJECT) { /* Called from for..in code in js_Interpret: return an XMLList. */ xml = (JSXML *) JS_GetPrivate(cx, obj); if (xml->xml_class != JSXML_CLASS_LIST) { obj = ToXMLList(cx, OBJECT_TO_JSVAL(obj)); if (!obj) return JS_FALSE; } *vp = OBJECT_TO_JSVAL(obj); return JS_TRUE; } return JS_CallFunctionName(cx, obj, js_toString_str, 0, NULL, vp); } static JSBool xml_enumerate(JSContext *cx, JSObject *obj, JSIterateOp enum_op, jsval *statep, jsid *idp) { JSXML *xml; uint32 length, index; JSXMLArrayCursor *cursor; xml = (JSXML *) JS_GetPrivate(cx, obj); length = JSXML_LENGTH(xml); switch (enum_op) { case JSENUMERATE_INIT: if (length == 0) { cursor = NULL; } else { cursor = (JSXMLArrayCursor *) JS_malloc(cx, sizeof *cursor); if (!cursor) return JS_FALSE; XMLArrayCursorInit(cursor, &xml->xml_kids); } *statep = PRIVATE_TO_JSVAL(cursor); if (idp) *idp = INT_TO_JSID(length); break; case JSENUMERATE_NEXT: cursor = JSVAL_TO_PRIVATE(*statep); if (cursor && cursor->array && (index = cursor->index) < length) { *idp = INT_TO_JSID(index); cursor->index = index + 1; break; } /* FALL THROUGH */ case JSENUMERATE_DESTROY: cursor = JSVAL_TO_PRIVATE(*statep); if (cursor) { XMLArrayCursorFinish(cursor); JS_free(cx, cursor); } *statep = JSVAL_NULL; break; } return JS_TRUE; } static JSBool xml_hasInstance(JSContext *cx, JSObject *obj, jsval v, JSBool *bp) { return JS_TRUE; } static uint32 xml_mark(JSContext *cx, JSObject *obj, void *arg) { JSXML *xml; xml = (JSXML *) JS_GetPrivate(cx, obj); JS_MarkGCThing(cx, xml, js_private_str, arg); return js_Mark(cx, obj, arg); } static void xml_clear(JSContext *cx, JSObject *obj) { } static JSBool HasSimpleContent(JSXML *xml) { JSXML *kid; JSBool simple; uint32 i, n; again: switch (xml->xml_class) { case JSXML_CLASS_COMMENT: case JSXML_CLASS_PROCESSING_INSTRUCTION: return JS_FALSE; case JSXML_CLASS_LIST: if (xml->xml_kids.length == 0) return JS_TRUE; if (xml->xml_kids.length == 1) { kid = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); xml = kid; goto again; } /* FALL THROUGH */ default: simple = JS_TRUE; for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { simple = JS_FALSE; break; } } return simple; } } static JSObject * xml_getMethod(JSContext *cx, JSObject *obj, jsid id, jsval *vp) { JSXML *xml; jsval fval; JS_ASSERT(JS_InstanceOf(cx, obj, &js_XMLClass, NULL)); xml = (JSXML *) JS_GetPrivate(cx, obj); retry: /* 11.2.2.1 Step 3(d) onward. */ if (!GetFunction(cx, obj, xml, id, &fval)) return NULL; if (JSVAL_IS_VOID(fval) && OBJECT_IS_XML(cx, obj)) { if (xml->xml_class == JSXML_CLASS_LIST) { if (xml->xml_kids.length == 1) { xml = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); obj = js_GetXMLObject(cx, xml); if (!obj) return NULL; goto retry; } } else if (HasSimpleContent(xml)) { JSString *str; JSObject *tmp; str = js_ValueToString(cx, OBJECT_TO_JSVAL(obj)); if (!str || !js_ValueToObject(cx, STRING_TO_JSVAL(str), &tmp)) return NULL; if (!js_GetProperty(cx, tmp, id, &fval)) return NULL; if (!JSVAL_IS_VOID(fval)) obj = tmp; } } *vp = fval; return obj; } static JSBool xml_setMethod(JSContext *cx, JSObject *obj, jsid id, jsval *vp) { return js_SetProperty(cx, obj, id, vp); } static JSBool xml_enumerateValues(JSContext *cx, JSObject *obj, JSIterateOp enum_op, jsval *statep, jsid *idp, jsval *vp) { JSXML *xml, *kid; uint32 length, index; JSXMLArrayCursor *cursor; JSObject *kidobj; xml = (JSXML *) JS_GetPrivate(cx, obj); length = JSXML_LENGTH(xml); JS_ASSERT(INT_FITS_IN_JSVAL(length)); switch (enum_op) { case JSENUMERATE_INIT: if (length == 0) { cursor = NULL; } else { cursor = (JSXMLArrayCursor *) JS_malloc(cx, sizeof *cursor); if (!cursor) return JS_FALSE; XMLArrayCursorInit(cursor, &xml->xml_kids); } *statep = PRIVATE_TO_JSVAL(cursor); if (idp) *idp = INT_TO_JSID(length); if (vp) *vp = JSVAL_VOID; break; case JSENUMERATE_NEXT: cursor = JSVAL_TO_PRIVATE(*statep); if (cursor && cursor->array && (index = cursor->index) < length) { kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return JS_FALSE; JS_ASSERT(INT_FITS_IN_JSVAL(index)); *idp = INT_TO_JSID(index); *vp = OBJECT_TO_JSVAL(kidobj); cursor->index = index + 1; break; } /* FALL THROUGH */ case JSENUMERATE_DESTROY: cursor = JSVAL_TO_PRIVATE(*statep); if (cursor) { XMLArrayCursorFinish(cursor); JS_free(cx, cursor); } *statep = JSVAL_NULL; break; } return JS_TRUE; } static JSBool xml_equality(JSContext *cx, JSObject *obj, jsval v, JSBool *bp) { JSXML *xml, *vxml; JSObject *vobj; JSBool ok; JSString *str, *vstr; jsdouble d, d2; xml = (JSXML *) JS_GetPrivate(cx, obj); vxml = NULL; if (!JSVAL_IS_PRIMITIVE(v)) { vobj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, vobj)) vxml = (JSXML *) JS_GetPrivate(cx, vobj); } if (xml->xml_class == JSXML_CLASS_LIST) { ok = Equals(cx, xml, v, bp); } else if (vxml) { if (vxml->xml_class == JSXML_CLASS_LIST) { ok = Equals(cx, vxml, OBJECT_TO_JSVAL(obj), bp); } else { if (((xml->xml_class == JSXML_CLASS_TEXT || xml->xml_class == JSXML_CLASS_ATTRIBUTE) && HasSimpleContent(vxml)) || ((vxml->xml_class == JSXML_CLASS_TEXT || vxml->xml_class == JSXML_CLASS_ATTRIBUTE) && HasSimpleContent(xml))) { ok = JS_EnterLocalRootScope(cx); if (ok) { str = js_ValueToString(cx, OBJECT_TO_JSVAL(obj)); vstr = js_ValueToString(cx, v); ok = str && vstr; if (ok) *bp = !js_CompareStrings(str, vstr); JS_LeaveLocalRootScope(cx); } } else { ok = XMLEquals(cx, xml, vxml, bp); } } } else { ok = JS_EnterLocalRootScope(cx); if (ok) { if (HasSimpleContent(xml)) { str = js_ValueToString(cx, OBJECT_TO_JSVAL(obj)); vstr = js_ValueToString(cx, v); ok = str && vstr; if (ok) *bp = !js_CompareStrings(str, vstr); } else if (JSVAL_IS_STRING(v) || JSVAL_IS_NUMBER(v)) { str = js_ValueToString(cx, OBJECT_TO_JSVAL(obj)); if (!str) { ok = JS_FALSE; } else if (JSVAL_IS_STRING(v)) { *bp = !js_CompareStrings(str, JSVAL_TO_STRING(v)); } else { ok = js_ValueToNumber(cx, STRING_TO_JSVAL(str), &d); if (ok) { d2 = JSVAL_IS_INT(v) ? JSVAL_TO_INT(v) : *JSVAL_TO_DOUBLE(v); *bp = JSDOUBLE_COMPARE(d, ==, d2, JS_FALSE); } } } else { *bp = JS_FALSE; } JS_LeaveLocalRootScope(cx); } } return ok; } static JSBool xml_concatenate(JSContext *cx, JSObject *obj, jsval v, jsval *vp) { JSBool ok; JSObject *listobj, *robj; JSXML *list, *lxml, *rxml; ok = JS_EnterLocalRootScope(cx); if (!ok) return JS_FALSE; listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) { ok = JS_FALSE; goto out; } list = (JSXML *) JS_GetPrivate(cx, listobj); lxml = (JSXML *) JS_GetPrivate(cx, obj); ok = Append(cx, list, lxml); if (!ok) goto out; if (VALUE_IS_XML(cx, v)) { rxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); } else { robj = ToXML(cx, v); if (!robj) { ok = JS_FALSE; goto out; } rxml = (JSXML *) JS_GetPrivate(cx, robj); } ok = Append(cx, list, rxml); if (!ok) goto out; *vp = OBJECT_TO_JSVAL(listobj); out: JS_LeaveLocalRootScope(cx); return ok; } /* Use js_NewObjectMap so XML objects satisfy OBJ_IS_NATIVE tests. */ JS_FRIEND_DATA(JSXMLObjectOps) js_XMLObjectOps = { { js_NewObjectMap, js_DestroyObjectMap, xml_lookupProperty, xml_defineProperty, xml_getProperty, xml_setProperty, xml_getAttributes, xml_setAttributes, xml_deleteProperty, xml_defaultValue, xml_enumerate, js_CheckAccess, NULL, NULL, NULL, NULL, NULL, xml_hasInstance, js_SetProtoOrParent, js_SetProtoOrParent, xml_mark, xml_clear, NULL, NULL }, xml_getMethod, xml_setMethod, xml_enumerateValues, xml_equality, xml_concatenate }; static JSObjectOps * xml_getObjectOps(JSContext *cx, JSClass *clasp) { return &js_XMLObjectOps.base; } JS_FRIEND_DATA(JSClass) js_XMLClass = { js_XML_str, JSCLASS_HAS_PRIVATE, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, xml_finalize, xml_getObjectOps, NULL, NULL, NULL, NULL, NULL, NULL, NULL }; static JSObject * CallConstructorFunction(JSContext *cx, JSObject *obj, JSClass *clasp, uintN argc, jsval *argv) { JSObject *tmp; jsval rval; while ((tmp = OBJ_GET_PARENT(cx, obj)) != NULL) obj = tmp; if (!JS_CallFunctionName(cx, obj, clasp->name, argc, argv, &rval)) return NULL; JS_ASSERT(!JSVAL_IS_PRIMITIVE(rval)); return JSVAL_TO_OBJECT(rval); } #define XML_METHOD_PROLOG \ JS_BEGIN_MACRO \ xml = (JSXML *) JS_GetInstancePrivate(cx, obj, &js_XMLClass, argv); \ if (!xml) \ return JS_FALSE; \ JS_END_MACRO static JSBool xml_addNamespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSObject *nsobj; JSXMLNamespace *ns; XML_METHOD_PROLOG; if (xml->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; nsobj = CallConstructorFunction(cx, obj, &js_NamespaceClass.base, 1, argv); if (!nsobj) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(nsobj); ns = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); if (!AddInScopeNamespace(cx, xml, ns)) return JS_FALSE; ns->declared = JS_TRUE; *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } static JSBool xml_appendChild(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *vxml; jsval name, v; JSObject *vobj; XML_METHOD_PROLOG; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; if (!js_GetAnyName(cx, &name)) return JS_FALSE; if (!GetProperty(cx, obj, name, &v)) return JS_FALSE; JS_ASSERT(!JSVAL_IS_PRIMITIVE(v)); vobj = JSVAL_TO_OBJECT(v); JS_ASSERT(OBJECT_IS_XML(cx, vobj)); vxml = (JSXML *) JS_GetPrivate(cx, vobj); JS_ASSERT(vxml->xml_class == JSXML_CLASS_LIST); if (!IndexToIdVal(cx, vxml->xml_kids.length, &name)) return JS_FALSE; if (!PutProperty(cx, JSVAL_TO_OBJECT(v), name, &argv[0])) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_attribute(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXMLQName *qn; qn = ToAttributeName(cx, argv[0]); if (!qn) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(qn->object); /* local root */ return GetProperty(cx, obj, argv[0], rval); } /* XML and XMLList */ static JSBool xml_attributes(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval name; JSXMLQName *qn; JSTempValueRooter tvr; JSBool ok; name = ATOM_KEY(cx->runtime->atomState.starAtom); qn = ToAttributeName(cx, name); if (!qn) return JS_FALSE; name = OBJECT_TO_JSVAL(qn->object); JS_PUSH_SINGLE_TEMP_ROOT(cx, name, &tvr); ok = GetProperty(cx, obj, name, rval); JS_POP_TEMP_ROOT(cx, &tvr); return ok; } /* XML and XMLList */ static JSBool xml_child_helper(JSContext *cx, JSObject *obj, JSXML *xml, jsval name, jsval *rval) { uint32 index; JSXML *kid; JSObject *kidobj; /* ECMA-357 13.4.4.6 */ JS_ASSERT(xml->xml_class != JSXML_CLASS_LIST); if (js_IdIsIndex(name, &index)) { if (index >= JSXML_LENGTH(xml)) { *rval = JSVAL_VOID; } else { kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(kidobj); } return JS_TRUE; } return GetProperty(cx, obj, name, rval); } static JSBool xml_child(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list, *kid, *vxml; jsval name, v; uint32 i, n; JSObject *listobj, *kidobj; XML_METHOD_PROLOG; name = argv[0]; if (xml->xml_class == JSXML_CLASS_LIST) { /* ECMA-357 13.5.4.4 */ listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return JS_FALSE; if (!xml_child_helper(cx, kidobj, kid, name, &v)) return JS_FALSE; if (JSVAL_IS_VOID(v)) { /* The property didn't exist in this kid. */ continue; } JS_ASSERT(!JSVAL_IS_PRIMITIVE(v)); vxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if ((!JSXML_HAS_KIDS(vxml) || vxml->xml_kids.length != 0) && !Append(cx, list, vxml)) { return JS_FALSE; } } return JS_TRUE; } return xml_child_helper(cx, obj, xml, name, rval); } static JSBool xml_childIndex(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *parent; uint32 i, n; XML_METHOD_PROLOG; parent = xml->parent; if (!parent || xml->xml_class == JSXML_CLASS_ATTRIBUTE) { *rval = DOUBLE_TO_JSVAL(cx->runtime->jsNaN); return JS_TRUE; } for (i = 0, n = JSXML_LENGTH(parent); i < n; i++) { if (XMLARRAY_MEMBER(&parent->xml_kids, i, JSXML) == xml) break; } JS_ASSERT(i < n); return js_NewNumberValue(cx, i, rval); } /* XML and XMLList */ static JSBool xml_children(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval name; name = ATOM_KEY(cx->runtime->atomState.starAtom); return GetProperty(cx, obj, name, rval); } /* XML and XMLList */ static JSBool xml_comments(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list, *kid, *vxml; JSObject *listobj, *kidobj; JSBool ok; uint32 i, n; jsval v; XML_METHOD_PROLOG; listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; ok = JS_TRUE; if (xml->xml_class == JSXML_CLASS_LIST) { /* 13.5.4.6 Step 2. */ for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { ok = JS_EnterLocalRootScope(cx); if (!ok) break; kidobj = js_GetXMLObject(cx, kid); ok = kidobj ? xml_comments(cx, kidobj, argc, argv, &v) : JS_FALSE; JS_LeaveLocalRootScope(cx); if (!ok) break; vxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if (JSXML_LENGTH(vxml) != 0) { ok = Append(cx, list, vxml); if (!ok) break; } } } } else { /* 13.4.4.9 Step 2. */ for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_COMMENT) { ok = Append(cx, list, kid); if (!ok) break; } } } return ok; } /* XML and XMLList */ static JSBool xml_contains(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *kid; jsval value; JSBool eq; JSObject *kidobj; uint32 i, n; XML_METHOD_PROLOG; value = argv[0]; if (xml->xml_class == JSXML_CLASS_LIST) { eq = JS_FALSE; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj || !xml_equality(cx, kidobj, value, &eq)) return JS_FALSE; if (eq) break; } } else { if (!xml_equality(cx, obj, value, &eq)) return JS_FALSE; } *rval = BOOLEAN_TO_JSVAL(eq); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_copy(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *copy; XML_METHOD_PROLOG; copy = DeepCopy(cx, xml, NULL, 0); if (!copy) return JS_FALSE; *rval = OBJECT_TO_JSVAL(copy->object); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_descendants(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list; jsval name; XML_METHOD_PROLOG; name = (argc == 0) ? ATOM_KEY(cx->runtime->atomState.starAtom) : argv[0]; list = Descendants(cx, xml, name); if (!list) return JS_FALSE; *rval = OBJECT_TO_JSVAL(list->object); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_elements(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list, *kid, *vxml; jsval name, v; JSXMLQName *nameqn; jsid funid; JSObject *listobj, *kidobj; JSBool ok; uint32 i, n; XML_METHOD_PROLOG; name = (argc == 0) ? ATOM_KEY(cx->runtime->atomState.starAtom) : argv[0]; nameqn = ToXMLName(cx, name, &funid); if (!nameqn) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(nameqn->object); listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); if (funid) return JS_TRUE; list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; list->xml_targetprop = nameqn; ok = JS_TRUE; if (xml->xml_class == JSXML_CLASS_LIST) { /* 13.5.4.6 */ for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { ok = JS_EnterLocalRootScope(cx); if (!ok) break; kidobj = js_GetXMLObject(cx, kid); ok = kidobj ? xml_elements(cx, kidobj, argc, argv, &v) : JS_FALSE; JS_LeaveLocalRootScope(cx); if (!ok) break; vxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if (JSXML_LENGTH(vxml) != 0) { ok = Append(cx, list, vxml); if (!ok) break; } } } } else { for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT && MatchElemName(nameqn, kid)) { ok = Append(cx, list, kid); if (!ok) break; } } } return ok; } /* XML and XMLList */ static JSBool xml_hasOwnProperty(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval name; JSObject *pobj; JSProperty *prop; name = argv[0]; if (!HasProperty(cx, obj, name, &pobj, &prop)) return JS_FALSE; if (!prop) { return js_HasOwnPropertyHelper(cx, obj, js_LookupProperty, argc, argv, rval); } DROP_PROPERTY(cx, pobj, prop); *rval = JSVAL_TRUE; return JS_TRUE; } /* XML and XMLList */ static JSBool xml_hasComplexContent(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *kid; JSObject *kidobj; uint32 i, n; XML_METHOD_PROLOG; again: switch (xml->xml_class) { case JSXML_CLASS_ATTRIBUTE: case JSXML_CLASS_COMMENT: case JSXML_CLASS_PROCESSING_INSTRUCTION: case JSXML_CLASS_TEXT: *rval = JSVAL_FALSE; break; case JSXML_CLASS_LIST: if (xml->xml_kids.length == 0) { *rval = JSVAL_TRUE; } else if (xml->xml_kids.length == 1) { kid = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) return JS_FALSE; obj = kidobj; xml = (JSXML *) JS_GetPrivate(cx, obj); goto again; } /* FALL THROUGH */ default: *rval = JSVAL_FALSE; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { *rval = JSVAL_TRUE; break; } } break; } return JS_TRUE; } /* XML and XMLList */ static JSBool xml_hasSimpleContent(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; XML_METHOD_PROLOG; *rval = BOOLEAN_TO_JSVAL(HasSimpleContent(xml)); return JS_TRUE; } static JSBool xml_inScopeNamespaces(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSObject *arrayobj, *nsobj; JSXML *xml; uint32 length, i, j, n; JSXMLNamespace *ns, *ns2; jsval v; arrayobj = js_NewArrayObject(cx, 0, NULL); if (!arrayobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(arrayobj); length = 0; XML_METHOD_PROLOG; do { if (xml->xml_class != JSXML_CLASS_ELEMENT) continue; for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); for (j = 0; j < length; j++) { if (!JS_GetElement(cx, arrayobj, j, &v)) return JS_FALSE; nsobj = JSVAL_TO_OBJECT(v); ns2 = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); if ((ns2->prefix && ns->prefix) ? !js_CompareStrings(ns2->prefix, ns->prefix) : !js_CompareStrings(ns2->uri, ns->uri)) { break; } } if (j == length) { nsobj = js_GetXMLNamespaceObject(cx, ns); if (!nsobj) return JS_FALSE; v = OBJECT_TO_JSVAL(nsobj); if (!JS_SetElement(cx, arrayobj, length, &v)) return JS_FALSE; ++length; } } } while ((xml = xml->parent) != NULL); return JS_TRUE; } static JSBool xml_insertChildAfter(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *kid; jsval arg; uint32 i; XML_METHOD_PROLOG; if (!JSXML_HAS_KIDS(xml)) return JS_TRUE; arg = argv[0]; if (JSVAL_IS_NULL(arg)) { kid = NULL; i = 0; } else { if (!VALUE_IS_XML(cx, arg)) return JS_TRUE; kid = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(arg)); i = XMLARRAY_FIND_MEMBER(&xml->xml_kids, kid, NULL); if (i == XML_NOT_FOUND) return JS_TRUE; ++i; } xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; if (!Insert(cx, xml, INT_TO_JSID(i), argv[1])) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } static JSBool xml_insertChildBefore(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *kid; jsval arg; uint32 i; XML_METHOD_PROLOG; if (!JSXML_HAS_KIDS(xml)) return JS_TRUE; arg = argv[0]; if (JSVAL_IS_NULL(arg)) { kid = NULL; i = xml->xml_kids.length; } else { if (!VALUE_IS_XML(cx, arg)) return JS_TRUE; kid = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(arg)); i = XMLARRAY_FIND_MEMBER(&xml->xml_kids, kid, NULL); if (i == XML_NOT_FOUND) return JS_TRUE; } xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; if (!Insert(cx, xml, INT_TO_JSID(i), argv[1])) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_length(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; XML_METHOD_PROLOG; if (xml->xml_class != JSXML_CLASS_LIST) { *rval = JSVAL_ONE; } else { if (!js_NewNumberValue(cx, xml->xml_kids.length, rval)) return JS_FALSE; } return JS_TRUE; } static JSBool xml_localName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; XML_METHOD_PROLOG; *rval = xml->name ? STRING_TO_JSVAL(xml->name->localName) : JSVAL_NULL; return JS_TRUE; } static JSBool xml_name(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSObject *nameobj; XML_METHOD_PROLOG; if (!xml->name) { *rval = JSVAL_NULL; } else { nameobj = js_GetXMLQNameObject(cx, xml->name); if (!nameobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(nameobj); } return JS_TRUE; } static JSBool xml_namespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSObject *arrayobj; JSBool ok; jsuint i, length; jsval v; JSXMLArray inScopeNSes; JSXMLNamespace *ns; JSString *prefix; XML_METHOD_PROLOG; if (argc == 0 && (xml->xml_class == JSXML_CLASS_TEXT || xml->xml_class == JSXML_CLASS_COMMENT || xml->xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION)) { *rval = JSVAL_NULL; return JS_TRUE; } if (!xml_inScopeNamespaces(cx, obj, 0, NULL, rval)) return JS_FALSE; arrayobj = JSVAL_TO_OBJECT(*rval); ok = js_GetLengthProperty(cx, arrayobj, &length); if (!ok) return JS_FALSE; if (argc == 0) { if (!XMLArrayInit(cx, &inScopeNSes, length)) return JS_FALSE; for (i = 0; i < length; i++) { ok = OBJ_GET_PROPERTY(cx, arrayobj, INT_TO_JSID(i), &v); if (!ok) break; JS_ASSERT(!JSVAL_IS_PRIMITIVE(v)); ns = (JSXMLNamespace *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); XMLARRAY_SET_MEMBER(&inScopeNSes, i, ns); } inScopeNSes.length = i; ns = ok ? GetNamespace(cx, xml->name, &inScopeNSes) : NULL; XMLArrayFinish(cx, &inScopeNSes); if (!ns) return JS_FALSE; *rval = OBJECT_TO_JSVAL(ns->object); } else { prefix = js_ValueToString(cx, argv[0]); if (!prefix) return JS_FALSE; argv[0] = STRING_TO_JSVAL(prefix); /* local root */ for (i = 0; i < length; i++) { if (!OBJ_GET_PROPERTY(cx, arrayobj, INT_TO_JSID(i), &v)) return JS_FALSE; JS_ASSERT(!JSVAL_IS_PRIMITIVE(v)); ns = (JSXMLNamespace *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if (ns->prefix && !js_CompareStrings(ns->prefix, prefix)) break; } *rval = (i < length) ? OBJECT_TO_JSVAL(ns->object) : JSVAL_VOID; } return JS_TRUE; } static JSBool xml_namespaceDeclarations(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSObject *arrayobj, *nsobj; JSXML *xml, *yml; JSBool ok; JSXMLArray ancestors, declared; uint32 i, n; JSXMLNamespace *ns; jsval v; arrayobj = js_NewArrayObject(cx, 0, NULL); if (!arrayobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(arrayobj); XML_METHOD_PROLOG; if (JSXML_HAS_VALUE(xml) || xml->xml_class == JSXML_CLASS_LIST) return JS_TRUE; /* From here, control flow must goto out to finish these arrays. */ ok = JS_TRUE; XMLArrayInit(cx, &ancestors, 0); XMLArrayInit(cx, &declared, 0); yml = xml; while ((yml = yml->parent) != NULL) { JS_ASSERT(yml->xml_class == JSXML_CLASS_ELEMENT); for (i = 0, n = yml->xml_namespaces.length; i < n; i++) { ns = XMLARRAY_MEMBER(&yml->xml_namespaces, i, JSXMLNamespace); if (!XMLARRAY_HAS_MEMBER(&ancestors, ns, namespace_match)) { ok = XMLARRAY_APPEND(cx, &ancestors, ns); if (!ok) goto out; } } } for (i = 0, n = xml->xml_namespaces.length; i < n; i++) { ns = XMLARRAY_MEMBER(&xml->xml_namespaces, i, JSXMLNamespace); if (!ns->declared) continue; if (!XMLARRAY_HAS_MEMBER(&ancestors, ns, namespace_match)) { ok = XMLARRAY_APPEND(cx, &declared, ns); if (!ok) goto out; } } for (i = 0, n = declared.length; i < n; i++) { ns = XMLARRAY_MEMBER(&declared, i, JSXMLNamespace); nsobj = js_GetXMLNamespaceObject(cx, ns); if (!nsobj) { ok = JS_FALSE; goto out; } v = OBJECT_TO_JSVAL(nsobj); ok = OBJ_SET_PROPERTY(cx, arrayobj, INT_TO_JSID(i), &v); if (!ok) goto out; } out: XMLArrayFinish(cx, &ancestors); XMLArrayFinish(cx, &declared); return ok; } static const char js_attribute_str[] = "attribute"; static const char js_text_str[] = "text"; /* Exported to jsgc.c #ifdef GC_MARK_DEBUG. */ const char *js_xml_class_str[] = { "list", "element", js_attribute_str, "processing-instruction", js_text_str, "comment" }; static JSBool xml_nodeKind(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSString *str; XML_METHOD_PROLOG; str = JS_InternString(cx, js_xml_class_str[xml->xml_class]); if (!str) return JS_FALSE; *rval = STRING_TO_JSVAL(str); return JS_TRUE; } static JSBool NormalizingDelete(JSContext *cx, JSObject *obj, JSXML *xml, jsval id) { jsval junk; if (xml->xml_class == JSXML_CLASS_LIST) return DeleteProperty(cx, obj, id, &junk); return DeleteByIndex(cx, xml, id, &junk); } /* * Erratum? the testcase js/tests/e4x/XML/13.4.4.26.js wants all-whitespace * text between tags to be removed by normalize. */ static JSBool IsXMLSpace(JSString *str) { const jschar *cp, *end; cp = JSSTRING_CHARS(str); end = cp + JSSTRING_LENGTH(str); while (cp < end) { if (!JS_ISXMLSPACE(*cp)) return JS_FALSE; ++cp; } return JS_TRUE; } /* XML and XMLList */ static JSBool xml_normalize(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *kid, *kid2; uint32 i, n; JSObject *kidobj; JSString *str; jsval junk; XML_METHOD_PROLOG; *rval = OBJECT_TO_JSVAL(obj); if (!JSXML_HAS_KIDS(xml)) return JS_TRUE; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { kidobj = js_GetXMLObject(cx, kid); if (!kidobj || !xml_normalize(cx, kidobj, argc, argv, &junk)) return JS_FALSE; } else if (kid->xml_class == JSXML_CLASS_TEXT) { while (i + 1 < n && (kid2 = XMLARRAY_MEMBER(&xml->xml_kids, i + 1, JSXML)) ->xml_class == JSXML_CLASS_TEXT) { str = js_ConcatStrings(cx, kid->xml_value, kid2->xml_value); if (!str) return JS_FALSE; if (!NormalizingDelete(cx, obj, xml, INT_TO_JSVAL(i + 1))) return JS_FALSE; n = xml->xml_kids.length; kid->xml_value = str; } if (IS_EMPTY(kid->xml_value) || IsXMLSpace(kid->xml_value)) { if (!NormalizingDelete(cx, obj, xml, INT_TO_JSVAL(i))) return JS_FALSE; n = xml->xml_kids.length; --i; } } } return JS_TRUE; } /* XML and XMLList */ static JSBool xml_parent(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *parent, *kid; uint32 i, n; JSObject *parentobj; XML_METHOD_PROLOG; parent = xml->parent; if (xml->xml_class == JSXML_CLASS_LIST) { *rval = JSVAL_VOID; n = xml->xml_kids.length; if (n == 0) return JS_TRUE; kid = XMLARRAY_MEMBER(&xml->xml_kids, 0, JSXML); parent = kid->parent; for (i = 1; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->parent != parent) return JS_TRUE; } } if (!parent) { *rval = JSVAL_NULL; return JS_TRUE; } parentobj = js_GetXMLObject(cx, parent); if (!parentobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(parentobj); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_processingInstructions(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list, *kid, *vxml; jsval name, v; JSXMLQName *nameqn; jsid funid; JSObject *listobj, *kidobj; JSBool ok; uint32 i, n; XML_METHOD_PROLOG; name = (argc == 0) ? ATOM_KEY(cx->runtime->atomState.starAtom) : argv[0]; nameqn = ToXMLName(cx, name, &funid); if (!nameqn) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(nameqn->object); listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); if (funid) return JS_TRUE; list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; list->xml_targetprop = nameqn; ok = JS_TRUE; if (xml->xml_class == JSXML_CLASS_LIST) { /* 13.5.4.17 Step 4 (misnumbered 9 -- Erratum?). */ for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { ok = JS_EnterLocalRootScope(cx); if (!ok) break; kidobj = js_GetXMLObject(cx, kid); ok = kidobj ? xml_processingInstructions(cx, kidobj, argc, argv, &v) : JS_FALSE; JS_LeaveLocalRootScope(cx); if (!ok) break; vxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if (JSXML_LENGTH(vxml) != 0) { ok = Append(cx, list, vxml); if (!ok) break; } } } } else { /* 13.4.4.28 Step 4. */ for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION && (IS_STAR(nameqn->localName) || !js_CompareStrings(nameqn->localName, kid->name->localName))) { ok = Append(cx, list, kid); if (!ok) break; } } } return ok; } static JSBool xml_prependChild(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; XML_METHOD_PROLOG; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; *rval = OBJECT_TO_JSVAL(obj); return Insert(cx, xml, JSVAL_ZERO, argv[0]); } /* XML and XMLList */ static JSBool xml_propertyIsEnumerable(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; jsval name; uint32 index; XML_METHOD_PROLOG; name = argv[0]; *rval = JSVAL_FALSE; if (js_IdIsIndex(name, &index)) { if (xml->xml_class == JSXML_CLASS_LIST) { /* 13.5.4.18. */ *rval = BOOLEAN_TO_JSVAL(index < xml->xml_kids.length); } else { /* 13.4.4.30. */ *rval = BOOLEAN_TO_JSVAL(index == 0); } } return JS_TRUE; } static JSBool namespace_full_match(const void *a, const void *b) { const JSXMLNamespace *nsa = (const JSXMLNamespace *) a; const JSXMLNamespace *nsb = (const JSXMLNamespace *) b; if (nsa->prefix && nsb->prefix && js_CompareStrings(nsa->prefix, nsb->prefix)) { return JS_FALSE; } return !js_CompareStrings(nsa->uri, nsb->uri); } static JSBool xml_removeNamespace_helper(JSContext *cx, JSXML *xml, JSXMLNamespace *ns) { JSXMLNamespace *thisns, *attrns; uint32 i, n; JSXML *attr, *kid; thisns = GetNamespace(cx, xml->name, &xml->xml_namespaces); JS_ASSERT(thisns); if (thisns == ns) return JS_TRUE; for (i = 0, n = xml->xml_attrs.length; i < n; i++) { attr = XMLARRAY_MEMBER(&xml->xml_attrs, i, JSXML); attrns = GetNamespace(cx, attr->name, &xml->xml_namespaces); JS_ASSERT(attrns); if (attrns == ns) return JS_TRUE; } i = XMLARRAY_FIND_MEMBER(&xml->xml_namespaces, ns, namespace_full_match); if (i != XML_NOT_FOUND) XMLArrayDelete(cx, &xml->xml_namespaces, i, JS_TRUE); for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { if (!xml_removeNamespace_helper(cx, kid, ns)) return JS_FALSE; } } return JS_TRUE; } static JSBool xml_removeNamespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSObject *nsobj; JSXMLNamespace *ns; XML_METHOD_PROLOG; *rval = OBJECT_TO_JSVAL(obj); if (xml->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; nsobj = CallConstructorFunction(cx, obj, &js_NamespaceClass.base, 1, argv); if (!nsobj) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(nsobj); ns = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); /* NOTE: remove ns from each ancestor if not used by that ancestor. */ return xml_removeNamespace_helper(cx, xml, ns); } static JSBool xml_replace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *vxml, *kid; jsval name, value, id, junk; uint32 index; JSObject *nameobj; JSXMLQName *nameqn; XML_METHOD_PROLOG; *rval = OBJECT_TO_JSVAL(obj); if (xml->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; value = argv[1]; vxml = VALUE_IS_XML(cx, value) ? (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(value)) : NULL; if (!vxml) { if (!JS_ConvertValue(cx, value, JSTYPE_STRING, &argv[1])) return JS_FALSE; value = argv[1]; } else { vxml = DeepCopy(cx, vxml, NULL, 0); if (!vxml) return JS_FALSE; value = argv[1] = OBJECT_TO_JSVAL(vxml->object); } xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; name = argv[0]; if (js_IdIsIndex(name, &index)) return Replace(cx, xml, name, value); /* Call function QName per spec, not ToXMLName, to avoid attribute names. */ nameobj = CallConstructorFunction(cx, obj, &js_QNameClass.base, 1, &name); if (!nameobj) return JS_FALSE; argv[0] = OBJECT_TO_JSVAL(nameobj); nameqn = (JSXMLQName *) JS_GetPrivate(cx, nameobj); id = JSVAL_VOID; index = xml->xml_kids.length; while (index != 0) { --index; kid = XMLARRAY_MEMBER(&xml->xml_kids, index, JSXML); if (MatchElemName(nameqn, kid)) { if (!JSVAL_IS_VOID(id) && !DeleteByIndex(cx, xml, id, &junk)) return JS_FALSE; if (!IndexToIdVal(cx, index, &id)) return JS_FALSE; } } if (JSVAL_IS_VOID(id)) return JS_TRUE; return Replace(cx, xml, id, value); } static JSBool xml_setChildren(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { if (!PutProperty(cx, obj, ATOM_KEY(cx->runtime->atomState.starAtom), &argv[0])) { return JS_FALSE; } *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } static JSBool xml_setLocalName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; jsval name; JSXMLQName *nameqn; JSString *namestr; XML_METHOD_PROLOG; if (!JSXML_HAS_NAME(xml)) return JS_TRUE; name = argv[0]; if (!JSVAL_IS_PRIMITIVE(name) && OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(name)) == &js_QNameClass.base) { nameqn = (JSXMLQName *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(name)); namestr = nameqn->localName; } else { if (!JS_ConvertValue(cx, name, JSTYPE_STRING, &argv[0])) return JS_FALSE; name = argv[0]; namestr = JSVAL_TO_STRING(name); } xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; xml->name->localName = namestr; return JS_TRUE; } static JSBool xml_setName(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *nsowner; jsval name; JSXMLQName *nameqn; JSObject *nameobj; JSXMLArray *nsarray; uint32 i, n; JSXMLNamespace *ns; XML_METHOD_PROLOG; if (!JSXML_HAS_NAME(xml)) return JS_TRUE; name = argv[0]; if (!JSVAL_IS_PRIMITIVE(name) && OBJ_GET_CLASS(cx, JSVAL_TO_OBJECT(name)) == &js_QNameClass.base && !(nameqn = (JSXMLQName *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(name))) ->uri) { name = argv[0] = STRING_TO_JSVAL(nameqn->localName); } nameobj = js_ConstructObject(cx, &js_QNameClass.base, NULL, NULL, 1, &name); if (!nameobj) return JS_FALSE; nameqn = (JSXMLQName *) JS_GetPrivate(cx, nameobj); /* ECMA-357 13.4.4.35 Step 4. */ if (xml->xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION) nameqn->uri = cx->runtime->emptyString; xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml) return JS_FALSE; xml->name = nameqn; /* * Erratum: nothing in 13.4.4.35 talks about making the name match the * in-scope namespaces, either by finding an in-scope namespace with a * matching uri and setting the new name's prefix to that namespace's * prefix, or by extending the in-scope namespaces for xml (which are in * xml->parent if xml is an attribute or a PI). */ if (xml->xml_class == JSXML_CLASS_ELEMENT) { nsowner = xml; } else { if (!xml->parent || xml->parent->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; nsowner = xml->parent; } if (nameqn->prefix) { /* * The name being set has a prefix, which originally came from some * namespace object (which may be the null namespace, where both the * prefix and uri are the empty string). We must go through a full * GetNamespace in case that namespace is in-scope in nsowner. * * If we find such an in-scope namespace, we return true right away, * in this block. Otherwise, we fall through to the final return of * AddInScopeNamespace(cx, nsowner, ns). */ ns = GetNamespace(cx, nameqn, &nsowner->xml_namespaces); if (!ns) return JS_FALSE; /* XXXbe have to test membership to see whether GetNamespace added */ if (XMLARRAY_HAS_MEMBER(&nsowner->xml_namespaces, ns, NULL)) return JS_TRUE; } else { /* * At this point, we know nameqn->prefix is null, so nameqn->uri can't * be the empty string (the null namespace always uses the empty string * for both prefix and uri). * * This means we must inline GetNamespace and specialize it to match * uri only, never prefix. If we find a namespace with nameqn's uri * already in nsowner->xml_namespaces, then all that we need do is set * nameqn->prefix to that namespace's prefix. * * If no such namespace exists, we can create one without going through * the constructor, because we know nameqn->uri is non-empty (so prefix * does not need to be converted from null to empty by QName). */ JS_ASSERT(!IS_EMPTY(nameqn->uri)); nsarray = &nsowner->xml_namespaces; for (i = 0, n = nsarray->length; i < n; i++) { ns = XMLARRAY_MEMBER(nsarray, i, JSXMLNamespace); if (!js_CompareStrings(ns->uri, nameqn->uri)) { nameqn->prefix = ns->prefix; return JS_TRUE; } } ns = js_NewXMLNamespace(cx, NULL, nameqn->uri, JS_TRUE); if (!ns) return JS_FALSE; } return AddInScopeNamespace(cx, nsowner, ns); } static JSBool xml_setNamespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *nsowner; JSObject *nsobj, *qnobj; JSXMLNamespace *ns; jsval qnargv[2]; XML_METHOD_PROLOG; if (xml->xml_class != JSXML_CLASS_ELEMENT && xml->xml_class != JSXML_CLASS_ATTRIBUTE) { return JS_TRUE; } xml = CHECK_COPY_ON_WRITE(cx, xml, obj); if (!xml || !js_GetXMLQNameObject(cx, xml->name)) return JS_FALSE; nsobj = js_ConstructObject(cx, &js_NamespaceClass.base, NULL, obj, 1, argv); if (!nsobj) return JS_FALSE; ns = (JSXMLNamespace *) JS_GetPrivate(cx, nsobj); ns->declared = JS_TRUE; qnargv[0] = argv[0] = OBJECT_TO_JSVAL(nsobj); qnargv[1] = OBJECT_TO_JSVAL(xml->name->object); qnobj = js_ConstructObject(cx, &js_QNameClass.base, NULL, NULL, 2, qnargv); if (!qnobj) return JS_FALSE; xml->name = (JSXMLQName *) JS_GetPrivate(cx, qnobj); /* * Erratum: the spec fails to update the governing in-scope namespaces. * See the erratum noted in xml_setName, above. */ if (xml->xml_class == JSXML_CLASS_ELEMENT) { nsowner = xml; } else { if (!xml->parent || xml->parent->xml_class != JSXML_CLASS_ELEMENT) return JS_TRUE; nsowner = xml->parent; } return AddInScopeNamespace(cx, nsowner, ns); } /* XML and XMLList */ static JSBool xml_text(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml, *list, *kid, *vxml; JSObject *listobj, *kidobj; uint32 i, n; JSBool ok; jsval v; XML_METHOD_PROLOG; listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); list = (JSXML *) JS_GetPrivate(cx, listobj); list->xml_target = xml; if (xml->xml_class == JSXML_CLASS_LIST) { ok = JS_TRUE; for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_ELEMENT) { ok = JS_EnterLocalRootScope(cx); if (!ok) break; kidobj = js_GetXMLObject(cx, kid); ok = kidobj ? xml_text(cx, kidobj, argc, argv, &v) : JS_FALSE; JS_LeaveLocalRootScope(cx); if (!ok) return JS_FALSE; vxml = (JSXML *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(v)); if (JSXML_LENGTH(vxml) != 0 && !Append(cx, list, vxml)) return JS_FALSE; } } } else { for (i = 0, n = JSXML_LENGTH(xml); i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class == JSXML_CLASS_TEXT && !Append(cx, list, kid)) return JS_FALSE; } } return JS_TRUE; } /* XML and XMLList */ static JSBool xml_toXMLString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSString *str; str = ToXMLString(cx, OBJECT_TO_JSVAL(obj)); if (!str) return JS_FALSE; *rval = STRING_TO_JSVAL(str); return JS_TRUE; } /* XML and XMLList */ static JSString * xml_toString_helper(JSContext *cx, JSXML *xml) { JSString *str, *kidstr; JSXML *kid; uint32 i, n; if (xml->xml_class == JSXML_CLASS_ATTRIBUTE || xml->xml_class == JSXML_CLASS_TEXT) { return xml->xml_value; } if (!HasSimpleContent(xml)) return ToXMLString(cx, OBJECT_TO_JSVAL(xml->object)); str = cx->runtime->emptyString; JS_EnterLocalRootScope(cx); for (i = 0, n = xml->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); if (kid->xml_class != JSXML_CLASS_COMMENT && kid->xml_class != JSXML_CLASS_PROCESSING_INSTRUCTION) { kidstr = xml_toString_helper(cx, kid); if (!kidstr) { str = NULL; break; } str = js_ConcatStrings(cx, str, kidstr); if (!str) break; } } JS_LeaveLocalRootScope(cx); return str; } static JSBool xml_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSXML *xml; JSString *str; XML_METHOD_PROLOG; str = xml_toString_helper(cx, xml); if (!str) return JS_FALSE; *rval = STRING_TO_JSVAL(str); return JS_TRUE; } /* XML and XMLList */ static JSBool xml_valueOf(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } static JSFunctionSpec xml_methods[] = { {"addNamespace", xml_addNamespace, 1,0,XML_MASK}, {"appendChild", xml_appendChild, 1,0,XML_MASK}, {js_attribute_str, xml_attribute, 1,0,GENERIC_MASK}, {"attributes", xml_attributes, 0,0,GENERIC_MASK}, {"child", xml_child, 1,0,GENERIC_MASK}, {"childIndex", xml_childIndex, 0,0,XML_MASK}, {"children", xml_children, 0,0,GENERIC_MASK}, {"comments", xml_comments, 0,0,GENERIC_MASK}, {"contains", xml_contains, 1,0,GENERIC_MASK}, {"copy", xml_copy, 0,0,GENERIC_MASK}, {"descendants", xml_descendants, 1,0,GENERIC_MASK}, {"elements", xml_elements, 1,0,GENERIC_MASK}, {"hasOwnProperty", xml_hasOwnProperty, 1,0,GENERIC_MASK}, {"hasComplexContent", xml_hasComplexContent, 1,0,GENERIC_MASK}, {"hasSimpleContent", xml_hasSimpleContent, 1,0,GENERIC_MASK}, {"inScopeNamespaces", xml_inScopeNamespaces, 0,0,XML_MASK}, {"insertChildAfter", xml_insertChildAfter, 2,0,XML_MASK}, {"insertChildBefore", xml_insertChildBefore, 2,0,XML_MASK}, {js_length_str, xml_length, 0,0,GENERIC_MASK}, {js_localName_str, xml_localName, 0,0,XML_MASK}, {js_name_str, xml_name, 0,0,XML_MASK}, {js_namespace_str, xml_namespace, 1,0,XML_MASK}, {"namespaceDeclarations", xml_namespaceDeclarations, 0,0,XML_MASK}, {"nodeKind", xml_nodeKind, 0,0,XML_MASK}, {"normalize", xml_normalize, 0,0,GENERIC_MASK}, {js_xml_parent_str, xml_parent, 0,0,GENERIC_MASK}, {"processingInstructions",xml_processingInstructions,1,0,GENERIC_MASK}, {"prependChild", xml_prependChild, 1,0,XML_MASK}, {"propertyIsEnumerable", xml_propertyIsEnumerable, 1,0,GENERIC_MASK}, {"removeNamespace", xml_removeNamespace, 1,0,XML_MASK}, {"replace", xml_replace, 2,0,XML_MASK}, {"setChildren", xml_setChildren, 1,0,XML_MASK}, {"setLocalName", xml_setLocalName, 1,0,XML_MASK}, {"setName", xml_setName, 1,0,XML_MASK}, {"setNamespace", xml_setNamespace, 1,0,XML_MASK}, {js_text_str, xml_text, 0,0,GENERIC_MASK}, {js_toString_str, xml_toString, 0,0,GENERIC_MASK}, {js_toXMLString_str, xml_toXMLString, 0,0,GENERIC_MASK}, {js_valueOf_str, xml_valueOf, 0,0,GENERIC_MASK}, {0,0,0,0,0} }; static JSBool CopyXMLSettings(JSContext *cx, JSObject *from, JSObject *to) { int i; const char *name; jsval v; for (i = XML_IGNORE_COMMENTS; i < XML_PRETTY_INDENT; i++) { name = xml_static_props[i].name; if (!JS_GetProperty(cx, from, name, &v)) return JS_FALSE; if (JSVAL_IS_BOOLEAN(v) && !JS_SetProperty(cx, to, name, &v)) return JS_FALSE; } name = xml_static_props[i].name; if (!JS_GetProperty(cx, from, name, &v)) return JS_FALSE; if (JSVAL_IS_NUMBER(v) && !JS_SetProperty(cx, to, name, &v)) return JS_FALSE; return JS_TRUE; } static JSBool SetDefaultXMLSettings(JSContext *cx, JSObject *obj) { int i; jsval v; for (i = XML_IGNORE_COMMENTS; i < XML_PRETTY_INDENT; i++) { v = JSVAL_TRUE; if (!JS_SetProperty(cx, obj, xml_static_props[i].name, &v)) return JS_FALSE; } v = INT_TO_JSVAL(2); return JS_SetProperty(cx, obj, xml_static_props[i].name, &v); } static JSBool xml_settings(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSObject *settings; settings = JS_NewObject(cx, NULL, NULL, NULL); if (!settings) return JS_FALSE; *rval = OBJECT_TO_JSVAL(settings); return CopyXMLSettings(cx, obj, settings); } static JSBool xml_setSettings(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval v; JSBool ok; JSObject *settings; v = argv[0]; if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) { cx->xmlSettingFlags = 0; ok = SetDefaultXMLSettings(cx, obj); } else { if (JSVAL_IS_PRIMITIVE(v)) return JS_TRUE; settings = JSVAL_TO_OBJECT(v); cx->xmlSettingFlags = 0; ok = CopyXMLSettings(cx, settings, obj); } if (ok) cx->xmlSettingFlags |= XSF_CACHE_VALID; return ok; } static JSBool xml_defaultSettings(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { JSObject *settings; settings = JS_NewObject(cx, NULL, NULL, NULL); if (!settings) return JS_FALSE; *rval = OBJECT_TO_JSVAL(settings); return SetDefaultXMLSettings(cx, settings); } static JSFunctionSpec xml_static_methods[] = { {"settings", xml_settings, 0,0,0}, {"setSettings", xml_setSettings, 1,0,0}, {"defaultSettings", xml_defaultSettings, 0,0,0}, {0,0,0,0,0} }; static JSBool XML(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval v; JSXML *xml, *copy; JSObject *xobj, *vobj; JSClass *clasp; v = argv[0]; if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) v = STRING_TO_JSVAL(cx->runtime->emptyString); xobj = ToXML(cx, v); if (!xobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(xobj); xml = (JSXML *) JS_GetPrivate(cx, xobj); if ((cx->fp->flags & JSFRAME_CONSTRUCTING) && !JSVAL_IS_PRIMITIVE(v)) { vobj = JSVAL_TO_OBJECT(v); clasp = OBJ_GET_CLASS(cx, vobj); if (clasp == &js_XMLClass || (clasp->flags & JSCLASS_DOCUMENT_OBSERVER)) { /* No need to lock obj, it's newly constructed and thread local. */ copy = DeepCopy(cx, xml, obj, 0); if (!copy) return JS_FALSE; JS_ASSERT(copy->object == obj); *rval = OBJECT_TO_JSVAL(obj); return JS_TRUE; } } return JS_TRUE; } static JSBool XMLList(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { jsval v; JSObject *vobj, *listobj; JSXML *xml, *list; v = argv[0]; if (JSVAL_IS_NULL(v) || JSVAL_IS_VOID(v)) v = STRING_TO_JSVAL(cx->runtime->emptyString); if ((cx->fp->flags & JSFRAME_CONSTRUCTING) && !JSVAL_IS_PRIMITIVE(v)) { vobj = JSVAL_TO_OBJECT(v); if (OBJECT_IS_XML(cx, vobj)) { xml = (JSXML *) JS_GetPrivate(cx, vobj); if (xml->xml_class == JSXML_CLASS_LIST) { listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); list = (JSXML *) JS_GetPrivate(cx, listobj); if (!Append(cx, list, xml)) return JS_FALSE; return JS_TRUE; } } } /* Toggle on XML support since the script has explicitly requested it. */ listobj = ToXMLList(cx, v); if (!listobj) return JS_FALSE; *rval = OBJECT_TO_JSVAL(listobj); return JS_TRUE; } #define JSXML_LIST_SIZE (offsetof(JSXML, u) + sizeof(struct JSXMLListVar)) #define JSXML_ELEMENT_SIZE (offsetof(JSXML, u) + sizeof(struct JSXMLVar)) #define JSXML_LEAF_SIZE (offsetof(JSXML, u) + sizeof(JSString *)) static size_t sizeof_JSXML[JSXML_CLASS_LIMIT] = { JSXML_LIST_SIZE, /* JSXML_CLASS_LIST */ JSXML_ELEMENT_SIZE, /* JSXML_CLASS_ELEMENT */ JSXML_LEAF_SIZE, /* JSXML_CLASS_ATTRIBUTE */ JSXML_LEAF_SIZE, /* JSXML_CLASS_PROCESSING_INSTRUCTION */ JSXML_LEAF_SIZE, /* JSXML_CLASS_TEXT */ JSXML_LEAF_SIZE /* JSXML_CLASS_COMMENT */ }; #ifdef DEBUG_notme JSCList xml_leaks = JS_INIT_STATIC_CLIST(&xml_leaks); uint32 xml_serial; #endif JSXML * js_NewXML(JSContext *cx, JSXMLClass xml_class) { JSXML *xml; xml = (JSXML *) js_NewGCThing(cx, GCX_XML, sizeof_JSXML[xml_class]); if (!xml) return NULL; xml->object = NULL; xml->domnode = NULL; xml->parent = NULL; xml->name = NULL; xml->xml_class = xml_class; xml->xml_flags = 0; if (JSXML_CLASS_HAS_VALUE(xml_class)) { xml->xml_value = cx->runtime->emptyString; } else { XMLArrayInit(cx, &xml->xml_kids, 0); if (xml_class == JSXML_CLASS_LIST) { xml->xml_target = NULL; xml->xml_targetprop = NULL; } else { XMLArrayInit(cx, &xml->xml_namespaces, 0); XMLArrayInit(cx, &xml->xml_attrs, 0); } } #ifdef DEBUG_notme JS_APPEND_LINK(&xml->links, &xml_leaks); xml->serial = xml_serial++; #endif METER(xml_stats.xml); METER(xml_stats.livexml); return xml; } /* * Code factored from js_MarkXML for use by xml_DeutschScorrWaite, see below. * All things marked here cannot lead to overlong lists (mark stack overflow). */ static void xml_mark_tail(JSContext *cx, JSXML *xml, void *arg) { XMLArrayTrim(&xml->xml_kids); if (xml->xml_class == JSXML_CLASS_LIST) { if (xml->xml_target) js_MarkXML(cx, xml->xml_target, arg); if (xml->xml_targetprop) js_MarkXMLQName(cx, xml->xml_targetprop, arg); } else { namespace_mark_vector(cx, (JSXMLNamespace **) xml->xml_namespaces.vector, xml->xml_namespaces.length, arg); XMLArrayTrim(&xml->xml_namespaces); xml_mark_vector(cx, (JSXML **) xml->xml_attrs.vector, xml->xml_attrs.length, arg); XMLArrayTrim(&xml->xml_attrs); } } static void xml_DeutschSchorrWaite(JSContext *cx, JSXML *xml, void *arg) { JSXML *top, *kid; uint8 *flagp; uint32 i, n; #ifdef JS_GCMETER JSRuntime *rt = cx->runtime; # define GCMETER(x) x #else # define GCMETER(x) /* nothing */ #endif top = NULL; flagp = js_GetGCThingFlags(xml); down: GCMETER(if (++rt->gcStats.dswdepth > rt->gcStats.maxdswdepth) rt->gcStats.maxdswdepth = rt->gcStats.dswdepth); *flagp |= GCF_MARK; i = 0; for (;;) { /* * Let (i == n) index xml->parent, not any child in xml->xml_kids. * Use JSXML_LENGTH here and below in case xml is a leaf node whose * parent we are marking non-recursively. In the case where parent * is being marked, the "kid/down" sense is backwards -- humor me. */ for (n = JSXML_LENGTH(xml); i <= n; i++) { if (i < n) { kid = XMLARRAY_MEMBER(&xml->xml_kids, i, JSXML); } else { kid = xml->parent; if (!kid) continue; } flagp = js_GetGCThingFlags(kid); if (*flagp & GCF_MARK) continue; /* * Don't descend if a for..in loop is enumerating xml's kids, i.e. * if xml has kids and its xml_kids.cursors member is non-null. */ if (JSXML_HAS_KIDS(kid) && (JSXML_HAS_VALUE(xml) || !xml->xml_kids.cursors)) { if (i < n) XMLARRAY_SET_MEMBER(&xml->xml_kids, i, top); else xml->parent = top; if (JSXML_HAS_KIDS(xml)) xml->xml_kids.cursors = JS_UINT32_TO_PTR(i); top = xml; xml = kid; goto down; } js_MarkXML(cx, kid, arg); } /* If we are back at the root (or we never left it), we're done. */ GCMETER(rt->gcStats.dswdepth--); xml->xml_kids.cursors = NULL; if (!top) return; /* Time to go back up the spanning tree. */ GCMETER(rt->gcStats.dswup++); i = JSXML_HAS_KIDS(top) ? JS_PTR_TO_UINT32(top->xml_kids.cursors) : 0; if (i < JSXML_LENGTH(top)) { kid = XMLARRAY_MEMBER(&top->xml_kids, i, JSXML); XMLARRAY_SET_MEMBER(&top->xml_kids, i, xml); } else { JS_ASSERT(i == JSXML_LENGTH(top)); kid = top->parent; top->parent = xml; } xml = top; top = kid; i++; } #undef GCMETER } void js_MarkXML(JSContext *cx, JSXML *xml, void *arg) { int stackDummy; JS_MarkGCThing(cx, xml->object, js_object_str, arg); JS_MarkGCThing(cx, xml->name, js_name_str, arg); if (!JS_CHECK_STACK_SIZE(cx, stackDummy)) { xml_DeutschSchorrWaite(cx, xml, arg); } else { JS_MarkGCThing(cx, xml->parent, js_xml_parent_str, arg); if (JSXML_HAS_VALUE(xml)) { JS_MarkGCThing(cx, xml->xml_value, "value", arg); } else { xml_mark_vector(cx, (JSXML **) xml->xml_kids.vector, xml->xml_kids.length, arg); xml_mark_tail(cx, xml, arg); } } } void js_FinalizeXML(JSContext *cx, JSXML *xml) { if (JSXML_HAS_KIDS(xml)) { XMLArrayFinish(cx, &xml->xml_kids); if (xml->xml_class == JSXML_CLASS_ELEMENT) { XMLArrayFinish(cx, &xml->xml_namespaces); XMLArrayFinish(cx, &xml->xml_attrs); } } #ifdef DEBUG_notme JS_REMOVE_LINK(&xml->links); #endif UNMETER(xml_stats.livexml); } JSObject * js_ParseNodeToXMLObject(JSContext *cx, JSParseNode *pn) { jsval nsval; JSXMLNamespace *ns; JSXMLArray nsarray; JSXML *xml; if (!js_GetDefaultXMLNamespace(cx, &nsval)) return NULL; JS_ASSERT(!JSVAL_IS_PRIMITIVE(nsval)); ns = (JSXMLNamespace *) JS_GetPrivate(cx, JSVAL_TO_OBJECT(nsval)); if (!XMLArrayInit(cx, &nsarray, 1)) return NULL; XMLARRAY_APPEND(cx, &nsarray, ns); xml = ParseNodeToXML(cx, pn, &nsarray, XSF_PRECOMPILED_ROOT); XMLArrayFinish(cx, &nsarray); if (!xml) return NULL; return xml->object; } JSObject * js_NewXMLObject(JSContext *cx, JSXMLClass xml_class) { JSXML *xml; xml = js_NewXML(cx, xml_class); if (!xml) return NULL; return js_GetXMLObject(cx, xml); } static JSObject * NewXMLObject(JSContext *cx, JSXML *xml) { JSObject *obj; obj = js_NewObject(cx, &js_XMLClass, NULL, NULL); if (!obj || !JS_SetPrivate(cx, obj, xml)) { cx->newborn[GCX_OBJECT] = NULL; return NULL; } METER(xml_stats.xmlobj); METER(xml_stats.livexmlobj); return obj; } JSObject * js_GetXMLObject(JSContext *cx, JSXML *xml) { JSObject *obj; obj = xml->object; if (obj) { JS_ASSERT(JS_GetPrivate(cx, obj) == xml); return obj; } /* * A JSXML cannot be shared among threads unless it has an object. * A JSXML cannot be given an object unless: * (a) it has no parent; or * (b) its parent has no object (therefore is thread-private); or * (c) its parent's object is locked. * * Once given an object, a JSXML is immutable. */ JS_ASSERT(!xml->parent || !xml->parent->object || JS_IS_OBJ_LOCKED(cx, xml->parent->object)); obj = NewXMLObject(cx, xml); if (!obj) return NULL; xml->object = obj; return obj; } JSObject * js_InitNamespaceClass(JSContext *cx, JSObject *obj) { return JS_InitClass(cx, obj, NULL, &js_NamespaceClass.base, Namespace, 2, namespace_props, namespace_methods, NULL, NULL); } JSObject * js_InitQNameClass(JSContext *cx, JSObject *obj) { return JS_InitClass(cx, obj, NULL, &js_QNameClass.base, QName, 2, qname_props, qname_methods, NULL, NULL); } JSObject * js_InitAttributeNameClass(JSContext *cx, JSObject *obj) { return JS_InitClass(cx, obj, NULL, &js_AttributeNameClass, AttributeName, 2, qname_props, qname_methods, NULL, NULL); } JSObject * js_InitAnyNameClass(JSContext *cx, JSObject *obj) { jsval v; if (!js_GetAnyName(cx, &v)) return NULL; return JSVAL_TO_OBJECT(v); } JSObject * js_InitXMLClass(JSContext *cx, JSObject *obj) { JSObject *proto, *pobj, *ctor; JSFunctionSpec *fs; JSFunction *fun; JSXML *xml; JSProperty *prop; JSScopeProperty *sprop; jsval cval, argv[1], junk; /* Define the isXMLName function. */ if (!JS_DefineFunction(cx, obj, js_isXMLName_str, xml_isXMLName, 1, 0)) return NULL; /* Define the XML class constructor and prototype. */ proto = JS_InitClass(cx, obj, NULL, &js_XMLClass, XML, 1, NULL, NULL, xml_static_props, xml_static_methods); if (!proto) return NULL; /* * XXX Hack alert: expand JS_DefineFunctions here to copy fs->extra into * fun->spare, clearing fun->extra. No xml_methods require extra local GC * roots allocated after actual arguments on the VM stack, but we need a * way to tell which methods work only on XML objects, which work only on * XMLList objects, and which work on either. */ for (fs = xml_methods; fs->name; fs++) { fun = JS_DefineFunction(cx, proto, fs->name, fs->call, fs->nargs, fs->flags); if (!fun) return NULL; fun->extra = 0; fun->spare = fs->extra; } xml = js_NewXML(cx, JSXML_CLASS_TEXT); if (!xml || !JS_SetPrivate(cx, proto, xml)) return NULL; xml->object = proto; METER(xml_stats.xmlobj); METER(xml_stats.livexmlobj); /* * Prepare to set default settings on the XML constructor we just made. * NB: We can't use JS_GetConstructor, because it calls OBJ_GET_PROPERTY, * which is xml_getProperty, which creates a new XMLList every time! We * must instead call js_LookupProperty directly. */ if (!js_LookupProperty(cx, proto, ATOM_TO_JSID(cx->runtime->atomState.constructorAtom), &pobj, &prop)) { return NULL; } JS_ASSERT(prop); sprop = (JSScopeProperty *) prop; JS_ASSERT(SPROP_HAS_VALID_SLOT(sprop, OBJ_SCOPE(pobj))); cval = OBJ_GET_SLOT(cx, pobj, sprop->slot); OBJ_DROP_PROPERTY(cx, pobj, prop); JS_ASSERT(JSVAL_IS_FUNCTION(cx, cval)); /* Set default settings. */ ctor = JSVAL_TO_OBJECT(cval); argv[0] = JSVAL_VOID; if (!xml_setSettings(cx, ctor, 1, argv, &junk)) return NULL; /* Define the XMLList function and give it the same prototype as XML. */ fun = JS_DefineFunction(cx, obj, js_XMLList_str, XMLList, 1, 0); if (!fun) return NULL; if (!js_SetClassPrototype(cx, fun->object, proto, JSPROP_READONLY | JSPROP_PERMANENT)) { return NULL; } return proto; } JSObject * js_InitXMLClasses(JSContext *cx, JSObject *obj) { if (!js_InitNamespaceClass(cx, obj)) return NULL; if (!js_InitQNameClass(cx, obj)) return NULL; if (!js_InitAttributeNameClass(cx, obj)) return NULL; if (!js_InitAnyNameClass(cx, obj)) return NULL; return js_InitXMLClass(cx, obj); } JSBool js_GetFunctionNamespace(JSContext *cx, jsval *vp) { JSRuntime *rt; JSObject *obj; JSAtom *atom; JSString *prefix, *uri; /* An invalid URI, for internal use only, guaranteed not to collide. */ static const char anti_uri[] = "@mozilla.org/js/function"; rt = cx->runtime; obj = rt->functionNamespaceObject; if (!obj) { atom = js_Atomize(cx, js_function_str, 8, 0); JS_ASSERT(atom); prefix = ATOM_TO_STRING(atom); atom = js_Atomize(cx, anti_uri, sizeof anti_uri - 1, ATOM_PINNED); if (!atom) return JS_FALSE; rt->atomState.lazy.functionNamespaceURIAtom = atom; uri = ATOM_TO_STRING(atom); obj = js_NewXMLNamespaceObject(cx, prefix, uri, JS_FALSE); if (!obj) return JS_FALSE; /* * Avoid entraining any in-scope Object.prototype. The loss of * Namespace.prototype is not detectable, as there is no way to * refer to this instance in scripts. When used to qualify method * names, its prefix and uri references are copied to the QName. */ OBJ_SET_PROTO(cx, obj, NULL); OBJ_SET_PARENT(cx, obj, NULL); rt->functionNamespaceObject = obj; } *vp = OBJECT_TO_JSVAL(obj); return JS_TRUE; } /* * Note the asymmetry between js_GetDefaultXMLNamespace and js_SetDefaultXML- * Namespace. Get searches fp->scopeChain for JS_DEFAULT_XML_NAMESPACE_ID, * while Set sets JS_DEFAULT_XML_NAMESPACE_ID in fp->varobj (unless fp is a * lightweight function activation). There's no requirement that fp->varobj * lie directly on fp->scopeChain, although it should be reachable using the * prototype chain from a scope object (cf. JSOPTION_VAROBJFIX in jsapi.h). * * If Get can't find JS_DEFAULT_XML_NAMESPACE_ID along the scope chain, it * creates a default namespace via 'new Namespace()'. In contrast, Set uses * its v argument as the uri of a new Namespace, with "" as the prefix. See * ECMA-357 12.1 and 12.1.1. Note that if Set is called with a Namespace n, * the default XML namespace will be set to ("", n.uri). So the uri string * is really the only usefully stored value of the default namespace. */ JSBool js_GetDefaultXMLNamespace(JSContext *cx, jsval *vp) { JSStackFrame *fp; JSObject *nsobj, *obj, *tmp; jsval v; fp = cx->fp; nsobj = fp->xmlNamespace; if (nsobj) { *vp = OBJECT_TO_JSVAL(nsobj); return JS_TRUE; } obj = NULL; for (tmp = fp->scopeChain; tmp; tmp = OBJ_GET_PARENT(cx, obj)) { obj = tmp; if (!OBJ_GET_PROPERTY(cx, obj, JS_DEFAULT_XML_NAMESPACE_ID, &v)) return JS_FALSE; if (!JSVAL_IS_PRIMITIVE(v)) { fp->xmlNamespace = JSVAL_TO_OBJECT(v); *vp = v; return JS_TRUE; } } nsobj = js_ConstructObject(cx, &js_NamespaceClass.base, NULL, obj, 0, NULL); if (!nsobj) return JS_FALSE; v = OBJECT_TO_JSVAL(nsobj); if (obj && !OBJ_DEFINE_PROPERTY(cx, obj, JS_DEFAULT_XML_NAMESPACE_ID, v, JS_PropertyStub, JS_PropertyStub, JSPROP_PERMANENT, NULL)) { return JS_FALSE; } fp->xmlNamespace = nsobj; *vp = v; return JS_TRUE; } JSBool js_SetDefaultXMLNamespace(JSContext *cx, jsval v) { jsval argv[2]; JSObject *nsobj, *varobj; JSStackFrame *fp; argv[0] = STRING_TO_JSVAL(cx->runtime->emptyString); argv[1] = v; nsobj = js_ConstructObject(cx, &js_NamespaceClass.base, NULL, NULL, 2, argv); if (!nsobj) return JS_FALSE; v = OBJECT_TO_JSVAL(nsobj); fp = cx->fp; varobj = fp->varobj; if (varobj) { if (!OBJ_DEFINE_PROPERTY(cx, varobj, JS_DEFAULT_XML_NAMESPACE_ID, v, JS_PropertyStub, JS_PropertyStub, JSPROP_PERMANENT, NULL)) { return JS_FALSE; } } else { JS_ASSERT(fp->fun && !(fp->fun->flags & JSFUN_HEAVYWEIGHT)); } fp->xmlNamespace = JSVAL_TO_OBJECT(v); return JS_TRUE; } JSBool js_ToAttributeName(JSContext *cx, jsval *vp) { JSXMLQName *qn; qn = ToAttributeName(cx, *vp); if (!qn) return JS_FALSE; *vp = OBJECT_TO_JSVAL(qn->object); return JS_TRUE; } JSString * js_EscapeAttributeValue(JSContext *cx, JSString *str) { return EscapeAttributeValue(cx, NULL, str); } JSString * js_AddAttributePart(JSContext *cx, JSBool isName, JSString *str, JSString *str2) { size_t len, len2, newlen; jschar *chars; if (JSSTRING_IS_DEPENDENT(str) || !(*js_GetGCThingFlags(str) & GCF_MUTABLE)) { str = js_NewStringCopyN(cx, JSSTRING_CHARS(str), JSSTRING_LENGTH(str), 0); if (!str) return NULL; } len = str->length; len2 = JSSTRING_LENGTH(str2); newlen = (isName) ? len + 1 + len2 : len + 2 + len2 + 1; chars = (jschar *) JS_realloc(cx, str->chars, (newlen+1) * sizeof(jschar)); if (!chars) return NULL; /* * Reallocating str (because we know it has no other references) requires * purging any deflated string cached for it. */ js_PurgeDeflatedStringCache(str); str->chars = chars; str->length = newlen; chars += len; if (isName) { *chars++ = ' '; js_strncpy(chars, JSSTRING_CHARS(str2), len2); chars += len2; } else { *chars++ = '='; *chars++ = '"'; js_strncpy(chars, JSSTRING_CHARS(str2), len2); chars += len2; *chars++ = '"'; } *chars = 0; return str; } JSString * js_EscapeElementValue(JSContext *cx, JSString *str) { return EscapeElementValue(cx, NULL, str); } JSString * js_ValueToXMLString(JSContext *cx, jsval v) { return ToXMLString(cx, v); } static JSBool anyname_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) { *rval = ATOM_KEY(cx->runtime->atomState.starAtom); return JS_TRUE; } JSBool js_GetAnyName(JSContext *cx, jsval *vp) { JSRuntime *rt; JSObject *obj; JSXMLQName *qn; rt = cx->runtime; obj = rt->anynameObject; if (!obj) { qn = js_NewXMLQName(cx, rt->emptyString, rt->emptyString, ATOM_TO_STRING(rt->atomState.starAtom)); if (!qn) return JS_FALSE; obj = js_NewObject(cx, &js_AnyNameClass, NULL, NULL); if (!obj || !JS_SetPrivate(cx, obj, qn)) { cx->newborn[GCX_OBJECT] = NULL; return JS_FALSE; } qn->object = obj; METER(xml_stats.qnameobj); METER(xml_stats.liveqnameobj); /* * Avoid entraining any in-scope Object.prototype. This loses the * default toString inheritance, but no big deal: we want a better * custom one for clearer diagnostics. */ if (!JS_DefineFunction(cx, obj, js_toString_str, anyname_toString, 0, 0)) { return JS_FALSE; } OBJ_SET_PROTO(cx, obj, NULL); JS_ASSERT(!OBJ_GET_PARENT(cx, obj)); rt->anynameObject = obj; } *vp = OBJECT_TO_JSVAL(obj); return JS_TRUE; } JSBool js_FindXMLProperty(JSContext *cx, jsval name, JSObject **objp, jsval *namep) { JSXMLQName *qn; jsid funid, id; JSObject *obj, *pobj, *lastobj; JSProperty *prop; const char *printable; qn = ToXMLName(cx, name, &funid); if (!qn) return JS_FALSE; id = OBJECT_TO_JSID(qn->object); obj = cx->fp->scopeChain; do { if (!OBJ_LOOKUP_PROPERTY(cx, obj, id, &pobj, &prop)) return JS_FALSE; if (prop) { OBJ_DROP_PROPERTY(cx, pobj, prop); /* * Call OBJ_THIS_OBJECT to skip any With object that wraps an XML * object to carry scope chain linkage in js_FilterXMLList. */ pobj = OBJ_THIS_OBJECT(cx, obj); if (OBJECT_IS_XML(cx, pobj)) { *objp = pobj; *namep = ID_TO_VALUE(id); return JS_TRUE; } } lastobj = obj; } while ((obj = OBJ_GET_PARENT(cx, obj)) != NULL); printable = js_ValueToPrintableString(cx, name); if (printable) { JS_ReportErrorFlagsAndNumber(cx, JSREPORT_ERROR, js_GetErrorMessage, NULL, JSMSG_UNDEFINED_XML_NAME, printable); } return JS_FALSE; } JSBool js_GetXMLProperty(JSContext *cx, JSObject *obj, jsval name, jsval *vp) { return GetProperty(cx, obj, name, vp); } JSBool js_SetXMLProperty(JSContext *cx, JSObject *obj, jsval name, jsval *vp) { return PutProperty(cx, obj, name, vp); } static JSXML * GetPrivate(JSContext *cx, JSObject *obj, const char *method) { JSXML *xml; xml = (JSXML *) JS_GetInstancePrivate(cx, obj, &js_XMLClass, NULL); if (!xml) { JS_ReportErrorNumber(cx, js_GetErrorMessage, NULL, JSMSG_INCOMPATIBLE_METHOD, js_XML_str, method, OBJ_GET_CLASS(cx, obj)->name); } return xml; } JSBool js_GetXMLDescendants(JSContext *cx, JSObject *obj, jsval id, jsval *vp) { JSXML *xml, *list; xml = GetPrivate(cx, obj, "descendants internal method"); if (!xml) return JS_FALSE; list = Descendants(cx, xml, id); if (!list) return JS_FALSE; *vp = OBJECT_TO_JSVAL(list->object); return JS_TRUE; } JSBool js_DeleteXMLListElements(JSContext *cx, JSObject *listobj) { JSXML *list; uint32 n; jsval junk; list = (JSXML *) JS_GetPrivate(cx, listobj); for (n = list->xml_kids.length; n != 0; --n) { if (!DeleteProperty(cx, listobj, INT_TO_JSID(0), &junk)) return JS_FALSE; } return JS_TRUE; } JSBool js_FilterXMLList(JSContext *cx, JSObject *obj, jsbytecode *pc, jsval *vp) { JSBool ok, match; JSStackFrame *fp; JSObject *scobj, *listobj, *resobj, *withobj, *kidobj; JSXML *xml, *list, *result, *kid; uint32 i, n; ok = JS_EnterLocalRootScope(cx); if (!ok) return JS_FALSE; /* All control flow after this point must exit via label out or bad. */ fp = cx->fp; scobj = fp->scopeChain; xml = GetPrivate(cx, obj, "filtering predicate operator"); if (!xml) goto bad; if (xml->xml_class == JSXML_CLASS_LIST) { list = xml; } else { listobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!listobj) goto bad; list = (JSXML *) JS_GetPrivate(cx, listobj); ok = Append(cx, list, xml); if (!ok) goto out; } resobj = js_NewXMLObject(cx, JSXML_CLASS_LIST); if (!resobj) goto bad; result = (JSXML *) JS_GetPrivate(cx, resobj); /* Hoist the scope chain update out of the loop over kids. */ withobj = js_NewObject(cx, &js_WithClass, NULL, scobj); if (!withobj) goto bad; fp->scopeChain = withobj; for (i = 0, n = list->xml_kids.length; i < n; i++) { kid = XMLARRAY_MEMBER(&list->xml_kids, i, JSXML); kidobj = js_GetXMLObject(cx, kid); if (!kidobj) goto bad; OBJ_SET_PROTO(cx, withobj, kidobj); ok = js_Interpret(cx, pc, vp) && js_ValueToBoolean(cx, *vp, &match); if (!ok) goto out; if (match) { ok = Append(cx, result, kid); if (!ok) goto out; } } *vp = OBJECT_TO_JSVAL(resobj); out: fp->scopeChain = scobj; JS_LeaveLocalRootScope(cx); return ok; bad: ok = JS_FALSE; goto out; } JSObject * js_ValueToXMLObject(JSContext *cx, jsval v) { return ToXML(cx, v); } JSObject * js_ValueToXMLListObject(JSContext *cx, jsval v) { return ToXMLList(cx, v); } JSObject * js_CloneXMLObject(JSContext *cx, JSObject *obj) { uintN flags; JSXML *xml; if (!GetXMLSettingFlags(cx, &flags)) return NULL; xml = (JSXML *) JS_GetPrivate(cx, obj); if (flags & (XSF_IGNORE_COMMENTS | XSF_IGNORE_PROCESSING_INSTRUCTIONS | XSF_IGNORE_WHITESPACE)) { xml = DeepCopy(cx, xml, NULL, flags); if (!xml) return NULL; return xml->object; } return NewXMLObject(cx, xml); } JSObject * js_NewXMLSpecialObject(JSContext *cx, JSXMLClass xml_class, JSString *name, JSString *value) { uintN flags; JSObject *obj; JSXML *xml; JSXMLQName *qn; if (!GetXMLSettingFlags(cx, &flags)) return NULL; if ((xml_class == JSXML_CLASS_COMMENT && (flags & XSF_IGNORE_COMMENTS)) || (xml_class == JSXML_CLASS_PROCESSING_INSTRUCTION && (flags & XSF_IGNORE_PROCESSING_INSTRUCTIONS))) { return js_NewXMLObject(cx, JSXML_CLASS_TEXT); } obj = js_NewXMLObject(cx, xml_class); if (!obj) return NULL; xml = (JSXML *) JS_GetPrivate(cx, obj); if (name) { qn = js_NewXMLQName(cx, cx->runtime->emptyString, NULL, name); if (!qn) return NULL; xml->name = qn; } xml->xml_value = value; return obj; } JSString * js_MakeXMLCDATAString(JSContext *cx, JSString *str) { return MakeXMLCDATAString(cx, NULL, str); } JSString * js_MakeXMLCommentString(JSContext *cx, JSString *str) { return MakeXMLCommentString(cx, NULL, str); } JSString * js_MakeXMLPIString(JSContext *cx, JSString *name, JSString *str) { return MakeXMLPIString(cx, NULL, name, str); } #endif /* JS_HAS_XML_SUPPORT */