Bug 778409 - Enter the compartment of unwrappedProto rather than obj in Rewrap. r=gabor

This can happen if chrome sets its proto to a content object from a different scope
than the one doing the wrapping. In this case, the prototype chain looks like this:

chromeobj => CCW(examplecom_obj) => CCW(examplecom_scope.Object.prototype)

When wrapping chromeobj for exampleorg_scope, things will look like this:

COW(chromeobj) => CCW(examplecom_obj) => CCW(examplecom_scope.Object.prototype)

Note that we don't remap the proto of CCW(examplecom_scope) to
exampleorg_scope.Object.prototype, because the proto remapping only happens when
the object we're wrapping is chrome. There's no reason it has to be this way, but
even if we changed it we still wouldn't get the nice remapped lookup behavior to
exampleorg_scope.Object.prototype, because the proxy handler for CCW(examplecom_obj)
isn't a ChromeObjectWrapper, and thus doesn't know how to to the prototype bouncing
correctly.

Anyway, I suspect this case isn't worth worrying about as long as we don't crash.
This commit is contained in:
Bobby Holley 2012-07-30 22:18:55 +02:00
Родитель 3182617118
Коммит 7a038349c8
3 изменённых файлов: 13 добавлений и 1 удалений

Просмотреть файл

@ -0,0 +1,11 @@
function run_test() {
var Cu = Components.utils;
var sb1 = Cu.Sandbox('http://example.com');
var sb2 = Cu.Sandbox('http://example.org');
var chromeObj = {foo: 2};
var sb1obj = Cu.evalInSandbox('new Object()', sb1);
chromeObj.__proto__ = sb1obj;
sb2.wrapMe = chromeObj;
do_check_true(true, "Didn't crash");
do_check_eq(sb2.wrapMe.__proto__, sb1obj, 'proto set correctly');
}

Просмотреть файл

@ -11,6 +11,7 @@ tail =
[test_bug641378.js]
[test_bug677864.js]
[test_bug711404.js]
[test_bug778409.js]
[test_bug_442086.js]
[test_file.js]
[test_import.js]

Просмотреть файл

@ -402,7 +402,7 @@ WrapperFactory::Rewrap(JSContext *cx, JSObject *obj, JSObject *wrappedProto, JSO
if (wrappedProto && IsCrossCompartmentWrapper(wrappedProto) &&
(unwrappedProto = Wrapper::wrappedObject(wrappedProto))) {
JSAutoEnterCompartment ac;
if (!ac.enter(cx, obj))
if (!ac.enter(cx, unwrappedProto))
return NULL;
key = JS_IdentifyClassPrototype(cx, unwrappedProto);
}