Bug 1082662 - Don't crash trying to mark a dictionary accessor shape whose getter/setter field has previously been mutated from a callable to |undefined|. r=terrence

--HG--
extra : rebase_source : d471a2bfeaf59f83a6c0213d27ef4be8205d54ae
This commit is contained in:
Jeff Walden 2014-10-14 14:43:53 -07:00
parent 54e4584429
commit 42043d3de2
2 changed files with 58 additions and 0 deletions

View File

@ -361,6 +361,9 @@ ShapeGetterSetterRef::mark(JSTracer *trc)
JSObject *obj = *objp;
JSObject *prior = obj;
if (!prior)
return;
trc->setTracingLocation(&*prior);
gc::Mark(trc, &obj, "AccessorShape getter or setter");
if (obj == *objp)

View File

@ -0,0 +1,55 @@
/*
* Any copyright is dedicated to the Public Domain.
* http://creativecommons.org/licenses/publicdomain/
*/
var gTestfile = "clear-dictionary-accessor-getset.js";
var BUGNUMBER = 1082662;
var summary =
"Properly handle GC of a dictionary accessor property whose [[Get]] or " +
"[[Set]] has been changed to |undefined|";
print(BUGNUMBER + ": " + summary);
/**************
* BEGIN TEST *
**************/
function test(field)
{
var prop = "[[" + field[0].toUpperCase() + field.substring(1) + "]]";
print("Testing for GC crashes after setting " + prop + " to undefined...");
function inner()
{
// Create an object with an accessor property.
var obj = { x: 42, get y() {}, set y(v) {} };
// 1) convert it to dictionary mode, in the process 2) creating a new
// version of that accessor property whose [[Get]] and [[Set]] are objects
// that trigger post barriers.
delete obj.x;
var desc = {};
desc[field] = undefined;
// Overwrite [[field]] with undefined. Note #1 above is necessary so this
// is an actual *overwrite*, and not (possibly) a shape-tree fork that
// doesn't overwrite.
Object.defineProperty(obj, "y", desc);
}
inner();
gc(); // In unfixed code, this crashes trying to mark a null [[field]].
}
test("get");
test("set");
/******************************************************************************/
if (typeof reportCompare === "function")
reportCompare(true, true);
print("Tests complete");