Commit 6edc3e31 authored by conradw's avatar conradw Committed by Commit bot

[strong] Implement per-object restrictions behaviour of property freezing

Implements the strong mode proposal's restrictions on changing a strong object's
writable, non-configurable property to non-writable.

Setting the strong bit is still wip, so this change will only affect those
objects that have the bit correctly set. The tests reflect this, and will be
expanded as more objects can be marked as strong.

BUG=v8:3956
LOG=N

Review URL: https://codereview.chromium.org/1142393003

Cr-Commit-Position: refs/heads/master@{#28698}
parent 9058ac3b
...@@ -227,8 +227,12 @@ class CallSite { ...@@ -227,8 +227,12 @@ class CallSite {
T(StrongArity, \ T(StrongArity, \
"In strong mode, calling a function with too few arguments is deprecated") \ "In strong mode, calling a function with too few arguments is deprecated") \
T(StrongImplicitCast, "In strong mode, implicit conversions are deprecated") \ T(StrongImplicitCast, "In strong mode, implicit conversions are deprecated") \
T(StrongRedefineDisallowed, \
"Cannot redefine non-configurable property '%' of strong object % to be " \
"non-writable") \
T(StrongSetProto, \ T(StrongSetProto, \
"On strong object %, redefining the internal prototype is deprecated") \ "On strong object %, redefining writable, non-configurable property '%' " \
"to be non-writable is deprecated") \
T(SymbolKeyFor, "% is not a symbol") \ T(SymbolKeyFor, "% is not a symbol") \
T(SymbolToPrimitive, \ T(SymbolToPrimitive, \
"Cannot convert a Symbol wrapper object to a primitive value") \ "Cannot convert a Symbol wrapper object to a primitive value") \
......
...@@ -682,14 +682,19 @@ function DefineObjectProperty(obj, p, desc, should_throw) { ...@@ -682,14 +682,19 @@ function DefineObjectProperty(obj, p, desc, should_throw) {
} }
// Step 10a // Step 10a
if (IsDataDescriptor(current) && IsDataDescriptor(desc)) { if (IsDataDescriptor(current) && IsDataDescriptor(desc)) {
if (!current.isWritable() && desc.isWritable()) { var currentIsWritable = current.isWritable();
if (should_throw) { if (currentIsWritable != desc.isWritable()) {
throw MakeTypeError(kRedefineDisallowed, p); if (!currentIsWritable || IS_STRONG(obj)) {
} else { if (should_throw) {
return false; throw currentIsWritable
? MakeTypeError(kStrongRedefineDisallowed, obj, p)
: MakeTypeError(kRedefineDisallowed, p);
} else {
return false;
}
} }
} }
if (!current.isWritable() && desc.hasValue() && if (!currentIsWritable && desc.hasValue() &&
!$sameValue(desc.getValue(), current.getValue())) { !$sameValue(desc.getValue(), current.getValue())) {
if (should_throw) { if (should_throw) {
throw MakeTypeError(kRedefineDisallowed, p); throw MakeTypeError(kRedefineDisallowed, p);
...@@ -1223,7 +1228,10 @@ function ObjectSealJS(obj) { ...@@ -1223,7 +1228,10 @@ function ObjectSealJS(obj) {
function ObjectFreezeJS(obj) { function ObjectFreezeJS(obj) {
if (!IS_SPEC_OBJECT(obj)) return obj; if (!IS_SPEC_OBJECT(obj)) return obj;
var isProxy = %_IsJSProxy(obj); var isProxy = %_IsJSProxy(obj);
if (isProxy || %HasSloppyArgumentsElements(obj) || %IsObserved(obj)) { // TODO(conradw): Investigate modifying the fast path to accommodate strong
// objects.
if (isProxy || %HasSloppyArgumentsElements(obj) || %IsObserved(obj) ||
IS_STRONG(obj)) {
if (isProxy) { if (isProxy) {
ProxyFix(obj); ProxyFix(obj);
} }
......
// Copyright 2015 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --strong-mode --allow-natives-syntax
// TODO(conradw): Track implementation of strong bit for other objects, add
// tests.
function getSloppyObjects() {
return [(function(){}), ({})];
}
function getStrictObjects() {
"use strict";
return [(function(){}), ({})];
}
function getStrongObjects() {
"use strong";
// Strong functions can't have properties added to them.
return [{}];
}
(function testStrongObjectFreezePropValid() {
"use strict";
let strongObjects = getStrongObjects();
for (let o of strongObjects) {
Object.defineProperty(o, "foo", { configurable: true, writable: true });
assertDoesNotThrow(
function() {
"use strong";
Object.defineProperty(o, "foo", {configurable: true, writable: false });
});
}
})();
(function testStrongObjectFreezePropInvalid() {
"use strict";
let sloppyObjects = getSloppyObjects();
let strictObjects = getStrictObjects();
let strongObjects = getStrongObjects();
let weakObjects = sloppyObjects.concat(strictObjects);
for (let o of weakObjects) {
Object.defineProperty(o, "foo", { writable: true });
assertDoesNotThrow(
function() {
"use strong";
Object.defineProperty(o, "foo", { writable: false });
});
}
for (let o of strongObjects) {
function defProp(o) {
Object.defineProperty(o, "foo", { writable: false });
}
function defProps(o) {
Object.defineProperties(o, { "foo": { writable: false } });
}
function freezeProp(o) {
Object.freeze(o);
}
Object.defineProperty(o, "foo", { writable: true });
for (let func of [defProp, defProps, freezeProp]) {
assertThrows(function(){func(o)}, TypeError);
assertThrows(function(){func(o)}, TypeError);
assertThrows(function(){func(o)}, TypeError);
%OptimizeFunctionOnNextCall(func);
assertThrows(function(){func(o)}, TypeError);
%DeoptimizeFunction(func);
assertThrows(function(){func(o)}, TypeError);
}
}
})();
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment