Commit b14305c1 authored by conradw's avatar conradw Committed by Commit bot

[strong] Implement per-object restrictions behaviour of delete operator

Implements the strong mode proposal's restrictions on the behaviour of the
delete operator for strong objects.

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.

Attempt 2, last version did not work with API.

BUG=v8:3956
LOG=N

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

Cr-Commit-Position: refs/heads/master@{#28724}
parent 3be8651f
......@@ -925,7 +925,7 @@ class FastElementsAccessor
return length_object;
}
static Handle<Object> DeleteCommon(Handle<JSObject> obj, uint32_t key,
static MaybeHandle<Object> DeleteCommon(Handle<JSObject> obj, uint32_t key,
LanguageMode language_mode) {
DCHECK(obj->HasFastSmiOrObjectElements() ||
obj->HasFastDoubleElements() ||
......@@ -949,6 +949,16 @@ class FastElementsAccessor
? Smi::cast(Handle<JSArray>::cast(obj)->length())->value()
: backing_store->length());
if (key < length) {
if (obj->map()->is_strong() && !backing_store->is_the_hole(key)) {
if (is_strict(language_mode)) {
Handle<Object> name = isolate->factory()->NewNumberFromUint(key);
THROW_NEW_ERROR(
isolate,
NewTypeError(MessageTemplate::kStrongDeleteProperty, obj, name),
Object);
}
return isolate->factory()->false_value();
}
if (!is_sloppy_arguments_elements_map) {
ElementsKind kind = KindTraits::Kind;
if (IsFastPackedElementsKind(kind)) {
......@@ -1420,12 +1430,21 @@ class DictionaryElementsAccessor
Handle<SeededNumberDictionary>::cast(backing_store);
int entry = dictionary->FindEntry(key);
if (entry != SeededNumberDictionary::kNotFound) {
Handle<Object> result =
SeededNumberDictionary::DeleteProperty(dictionary, entry);
if (*result == *isolate->factory()->false_value()) {
Handle<Object> result;
bool strong = obj->map()->is_strong();
if (!strong) {
result = SeededNumberDictionary::DeleteProperty(dictionary, entry);
}
if (strong || *result == *isolate->factory()->false_value()) {
// Fail if the property is not configurable, or on a strong object.
if (is_strict(language_mode)) {
// Deleting a non-configurable property in strict mode.
Handle<Object> name = isolate->factory()->NewNumberFromUint(key);
if (strong) {
THROW_NEW_ERROR(
isolate,
NewTypeError(MessageTemplate::kStrongDeleteProperty, obj, name),
Object);
}
THROW_NEW_ERROR(
isolate,
NewTypeError(MessageTemplate::kStrictDeleteProperty, name, obj),
......
......@@ -226,6 +226,8 @@ class CallSite {
T(StrictReadOnlyProperty, "Cannot assign to read only property '%' of %") \
T(StrongArity, \
"In strong mode, calling a function with too few arguments is deprecated") \
T(StrongDeleteProperty, \
"On strong object %, deletion of property % is deprecated") \
T(StrongImplicitCast, "In strong mode, implicit conversions are deprecated") \
T(StrongRedefineDisallowed, \
"Cannot redefine non-configurable property '%' of strong object % to be " \
......
......@@ -5467,9 +5467,16 @@ MaybeHandle<Object> JSObject::DeleteProperty(Handle<JSObject> object,
}
// Fall through.
case LookupIterator::ACCESSOR: {
if (!it.IsConfigurable()) {
// Fail if the property is not configurable.
if (!it.IsConfigurable() || object->map()->is_strong()) {
// Fail if the property is not configurable, or on a strong object.
if (is_strict(language_mode)) {
if (object->map()->is_strong()) {
THROW_NEW_ERROR(
it.isolate(),
NewTypeError(MessageTemplate::kStrongDeleteProperty, object,
name),
Object);
}
THROW_NEW_ERROR(it.isolate(),
NewTypeError(MessageTemplate::kStrictDeleteProperty,
name, object),
......
......@@ -21328,6 +21328,28 @@ TEST(StrongModeArityCallFromApi2) {
}
TEST(StrongObjectDelete) {
i::FLAG_strong_mode = true;
LocalContext env;
v8::Isolate* isolate = env->GetIsolate();
v8::HandleScope scope(isolate);
Local<Object> obj;
{
v8::TryCatch try_catch;
obj = Local<Object>::Cast(CompileRun(
"'use strong';"
"({});"));
CHECK(!try_catch.HasCaught());
}
obj->ForceSet(v8_str("foo"), v8_num(1), v8::None);
obj->ForceSet(v8_str("2"), v8_num(1), v8::None);
CHECK(obj->HasOwnProperty(v8_str("foo")));
CHECK(obj->HasOwnProperty(v8_str("2")));
CHECK(!obj->Delete(v8_str("foo")));
CHECK(!obj->Delete(2));
}
TEST(ExtrasExportsObject) {
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope handle_scope(isolate);
......
// 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 getWeakObjects() {
return getSloppyObjects().concat(getStrictObjects());
}
function getStrongObjects() {
"use strong";
// Strong functions can't have properties added to them, and will be tested as a
// special case.
return [({})];
}
function strongFunction() {
"use strong";
}
function deleteFromObjectSloppy(o) {
return delete o.foo;
}
function deleteFromObjectKeyedSloppy(o) {
return delete o["foo"];
}
function deleteFromObjectKeyedVarSloppy(o) {
var a = "foo";
return delete o[a];
}
function deleteFromObjectKeyedComputedSloppy(o) {
var a = "o";
return delete o["fo" + a];
}
function deleteFromObjectWith(o) {
with (o) {
return delete foo;
}
}
function deleteFromObjectElementSloppy(o) {
return delete o[0];
}
function deleteFromObjectElementVarSloppy(o) {
var a = 0;
return delete o[a];
}
function deleteFromObjectElementSparseSloppy(o) {
return delete o[100000];
}
function deleteFromObjectElementVarSloppy(o) {
var a = 0;
return delete o[a];
}
function deleteFromObjectElementSparseVarSloppy(o) {
var a = 100000;
return delete o[a];
}
function deleteFromObjectStrict(o) {
"use strict";
return delete o.foo;
}
function deleteFromObjectKeyedStrict(o) {
"use strict";
return delete o["foo"];
}
function deleteFromObjectKeyedVarStrict(o) {
"use strict";
var a = "foo";
return delete o[a];
}
function deleteFromObjectKeyedComputedStrict(o) {
"use strict";
var a = "o";
return delete o["fo" + a];
}
function deleteFromObjectElementStrict(o) {
"use strict";
return delete o[0];
}
function deleteFromObjectElementSparseStrict(o) {
"use strict";
return delete o[100000];
}
function deleteFromObjectElementVarStrict(o) {
"use strict";
var a = 0;
return delete o[a];
}
function deleteFromObjectElementSparseVarStrict(o) {
"use strict";
var a = 100000;
return delete o[a];
}
function testStrongObjectDelete() {
"use strict";
let deletePropertyFuncsSloppy = [
deleteFromObjectSloppy,
deleteFromObjectKeyedSloppy,
deleteFromObjectKeyedVarSloppy,
deleteFromObjectKeyedComputedSloppy,
deleteFromObjectWith
];
let deletePropertyFuncsStrict = [
deleteFromObjectStrict,
deleteFromObjectKeyedStrict,
deleteFromObjectKeyedVarStrict,
deleteFromObjectKeyedComputedStrict
];
let deleteElementFuncsSloppy = [
deleteFromObjectElementSloppy,
deleteFromObjectElementVarSloppy
];
let deleteElementSparseFuncsSloppy = [
deleteFromObjectElementSparseSloppy,
deleteFromObjectElementSparseVarSloppy
];
let deleteElementFuncsStrict = [
deleteFromObjectElementStrict,
deleteFromObjectElementVarStrict
];
let deleteElementSparseFuncsStrict = [
deleteFromObjectElementSparseStrict,
deleteFromObjectElementSparseVarStrict
];
let deleteFuncs = deletePropertyFuncsSloppy.concat(
deletePropertyFuncsStrict, deleteElementFuncsSloppy,
deleteElementSparseFuncsSloppy, deleteElementFuncsStrict,
deleteElementSparseFuncsStrict);
for (let deleteFunc of deleteFuncs) {
assertTrue(deleteFunc(strongFunction));
}
let testCasesSloppy = [
[deletePropertyFuncsSloppy, "foo"],
[deleteElementFuncsSloppy, "0"],
[deleteElementSparseFuncsSloppy, "100000"]
];
let testCasesStrict = [
[deletePropertyFuncsStrict, "foo"],
[deleteElementFuncsStrict, "0"],
[deleteElementSparseFuncsStrict, "100000"]
];
let propDescs = [
{configurable: true, value: "bar"},
{configurable: true, value: 1},
{configurable: true, enumerable: true, writable: true, value: "bar"},
{configurable: true, enumerable: true, writable: true, value: 1},
{configurable: true, get: (function(){return 0}), set: (function(x){})}
];
for (let propDesc of propDescs) {
for (let testCase of testCasesSloppy) {
let name = testCase[1];
for (let deleteFunc of testCase[0]) {
for (let o of getWeakObjects()) {
Object.defineProperty(o, name, propDesc);
assertTrue(delete o["bar"]);
assertTrue(delete o[5000]);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
%OptimizeFunctionOnNextCall(deleteFunc);
Object.defineProperty(o, name, propDesc);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
%DeoptimizeFunction(deleteFunc);
Object.defineProperty(o, name, propDesc);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
}
for (let o of getStrongObjects()) {
Object.defineProperty(o, name, propDesc);
assertTrue(delete o["bar"]);
assertTrue(delete o[5000]);
assertFalse(deleteFunc(o));
assertTrue(o.hasOwnProperty(name));
%OptimizeFunctionOnNextCall(deleteFunc);
assertFalse(deleteFunc(o));
assertTrue(o.hasOwnProperty(name));
%DeoptimizeFunction(deleteFunc);
assertFalse(deleteFunc(o));
assertTrue(o.hasOwnProperty(name));
}
}
}
for (let testCase of testCasesStrict) {
let name = testCase[1];
for (let deleteFunc of testCase[0]) {
for (let o of getWeakObjects()) {
Object.defineProperty(o, name, propDesc);
assertTrue(delete o["bar"]);
assertTrue(delete o[5000]);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
%OptimizeFunctionOnNextCall(deleteFunc);
Object.defineProperty(o, name, propDesc);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
%DeoptimizeFunction(deleteFunc);
Object.defineProperty(o, name, propDesc);
assertTrue(deleteFunc(o));
assertFalse(o.hasOwnProperty(name));
}
for (let o of getStrongObjects()) {
Object.defineProperty(o, name, propDesc);
assertTrue(delete o["bar"]);
assertTrue(delete o[5000]);
assertThrows(function(){deleteFunc(o)}, TypeError);
assertTrue(o.hasOwnProperty(name));
%OptimizeFunctionOnNextCall(deleteFunc);
assertThrows(function(){deleteFunc(o)}, TypeError);
assertTrue(o.hasOwnProperty(name));
%DeoptimizeFunction(deleteFunc);
assertThrows(function(){deleteFunc(o)}, TypeError);
assertTrue(o.hasOwnProperty(name));
}
}
}
}
}
testStrongObjectDelete();
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