From 6f2dea22eebac012da24d8dbd29e8470ce1c38c6 Mon Sep 17 00:00:00 2001 From: Steve Fink <sfink@mozilla.com> Date: Wed, 18 Jul 2012 14:32:39 -0700 Subject: [PATCH] Bug 776635 - Allow use of JSRootedObject using only external headers. r=billm Currently, JS::Rooted<JSObject*> requires the full definition of JSObject to be available (due to RootMethods<JSObject*>::kind() calling JSObject::rootKind()). This patch switches to using a RootKind<T> template that is explicitly specialized in jspubtd.h for externally-visible pointer types, but still falls back on T::rootKind() for other pointer types so that we don't need to explicitly specialize on eg all subclasses of JSString or JSObject. --HG-- extra : rebase_source : 138a859babc1aac60e61fe6f08089d8dbb4995d2 --- js/src/gc/Root.h | 10 +++++++++- js/src/jspubtd.h | 14 ++++++++++++++ js/src/jsscope.h | 3 +++ 3 files changed, 26 insertions(+), 1 deletion(-) diff --git a/js/src/gc/Root.h b/js/src/gc/Root.h index e063d202a70f..ecf4640e2ef4 100644 --- a/js/src/gc/Root.h +++ b/js/src/gc/Root.h @@ -221,11 +221,19 @@ class MutableHandle : public MutableHandleBase<T> typedef MutableHandle<JSObject*> MutableHandleObject; typedef MutableHandle<Value> MutableHandleValue; +/* + * By default, pointers should use the inheritance hierarchy to find their + * ThingRootKind. Some pointer types are explicitly set in jspubtd.h so that + * Rooted<T> may be used without the class definition being available. + */ +template <typename T> +struct RootKind<T *> { static ThingRootKind rootKind() { return T::rootKind(); }; }; + template <typename T> struct RootMethods<T *> { static T *initial() { return NULL; } - static ThingRootKind kind() { return T::rootKind(); } + static ThingRootKind kind() { return RootKind<T *>::rootKind(); } static bool poisoned(T *v) { return IsPoisonedPtr(v); } }; diff --git a/js/src/jspubtd.h b/js/src/jspubtd.h index 45864ca0d695..7ba52f829188 100644 --- a/js/src/jspubtd.h +++ b/js/src/jspubtd.h @@ -240,6 +240,20 @@ enum ThingRootKind THING_ROOT_LIMIT }; +template <typename T> +struct RootKind; + +/* + * Specifically mark the ThingRootKind of externally visible types, so that + * JSAPI users may use JSRooted... types without having the class definition + * available. + */ +template <> struct RootKind<JSObject *> { static ThingRootKind rootKind() { return THING_ROOT_OBJECT; }; }; +template <> struct RootKind<JSString *> { static ThingRootKind rootKind() { return THING_ROOT_STRING; }; }; +template <> struct RootKind<JSScript *> { static ThingRootKind rootKind() { return THING_ROOT_SCRIPT; }; }; +template <> struct RootKind<jsid> { static ThingRootKind rootKind() { return THING_ROOT_ID; }; }; +template <> struct RootKind<Value> { static ThingRootKind rootKind() { return THING_ROOT_VALUE; }; }; + struct ContextFriendFields { JSRuntime *const runtime; diff --git a/js/src/jsscope.h b/js/src/jsscope.h index 657c675a9c15..99820e1a5f3d 100644 --- a/js/src/jsscope.h +++ b/js/src/jsscope.h @@ -1182,6 +1182,9 @@ MarkNonNativePropertyFound(HandleObject obj, MutableHandleShape propp); namespace JS { template<> class AnchorPermitted<js::Shape *> { }; template<> class AnchorPermitted<const js::Shape *> { }; + + template<> struct RootKind<js::Shape *> { static ThingRootKind rootKind() { return THING_ROOT_SHAPE; }; }; + template<> struct RootKind<js::BaseShape *> { static ThingRootKind rootKind() { return THING_ROOT_BASE_SHAPE; }; }; } #endif /* jsscope_h___ */