Skip to content

Commit 3e080f7

Browse files
committed
Issue in setting accessor in PathTypeHandlerWithAttr
When the property is deleted (say when we are in SimpleTypeHandler) and then later if we create an accessor on the same property we have converted the type handler to PathTypeHandlerWithAttr. Since a property got deleted and PathTypeHandler should not handle that deleted property we need to convert this to Dictionary type. In order to fix that When we add the accessor on any property we need to check if any property deleted upon that we need to convert to Dictionary type instead of type.
1 parent 6b98ef8 commit 3e080f7

File tree

5 files changed

+54
-1
lines changed

5 files changed

+54
-1
lines changed

lib/Runtime/Types/PathTypeHandler.cpp

+2
Original file line numberDiff line numberDiff line change
@@ -1196,6 +1196,8 @@ namespace Js
11961196
}
11971197
}
11981198

1199+
Assert((attr & ObjectSlotAttr_Deleted) != ObjectSlotAttr_Deleted);
1200+
11991201
SetAttributesHelper(instance, propertyId, propertyIndex, attributes, (ObjectSlotAttributes)(attr | ObjectSlotAttr_Accessor));
12001202
// SetAttributesHelper can convert to dictionary in corner cases, e.g., if we haven't got a full path from the root. Remove this check when that's fixed.
12011203
if (!instance->GetTypeHandler()->IsPathTypeHandler())

lib/Runtime/Types/SimpleTypeHandler.cpp

+18-1
Original file line numberDiff line numberDiff line change
@@ -172,9 +172,26 @@ namespace Js
172172
return newTypeHandler;
173173
}
174174

175+
template<size_t size>
176+
bool SimpleTypeHandler<size>::HasAnyPropertyDeleted()
177+
{
178+
for (PropertyIndex i = 0; i < propertyCount; i++)
179+
{
180+
if (descriptors[i].Attributes & PropertyDeleted)
181+
{
182+
return true;
183+
}
184+
}
185+
186+
return false;
187+
}
188+
189+
175190
template<size_t size>
176191
PathTypeHandlerBase* SimpleTypeHandler<size>::ConvertToPathType(DynamicObject* instance)
177192
{
193+
Assert(!HasAnyPropertyDeleted());
194+
178195
ScriptContext *scriptContext = instance->GetScriptContext();
179196
PathTypeHandlerBase* newTypeHandler =
180197
PathTypeHandlerNoAttr::New(
@@ -956,7 +973,7 @@ namespace Js
956973
template<size_t size>
957974
BOOL SimpleTypeHandler<size>::SetAccessors(DynamicObject* instance, PropertyId propertyId, Var getter, Var setter, PropertyOperationFlags flags)
958975
{
959-
if (DoConvertToPathType(instance->GetDynamicType()))
976+
if (DoConvertToPathType(instance->GetDynamicType()) && !HasAnyPropertyDeleted())
960977
{
961978
return ConvertToPathType(instance)->SetAccessors(instance, propertyId, getter, setter, flags);
962979
}

lib/Runtime/Types/SimpleTypeHandler.h

+2
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,8 @@ namespace Js
9292
ES5ArrayTypeHandler* ConvertToES5ArrayType(DynamicObject* instance);
9393
SimpleTypeHandler<size>* ConvertToNonSharedSimpleType(DynamicObject * instance);
9494

95+
bool HasAnyPropertyDeleted();
96+
9597
BOOL GetDescriptor(PropertyId propertyId, PropertyIndex * index);
9698
BOOL SetAttribute(DynamicObject* instance, PropertyIndex index, PropertyAttributes attribute);
9799
BOOL ClearAttribute(DynamicObject* instance, PropertyIndex index, PropertyAttributes attribute);

test/Bugs/bug_6271.js

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
//-------------------------------------------------------------------------------------------------------
2+
// Copyright (C) Microsoft. All rights reserved.
3+
// Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
4+
//-------------------------------------------------------------------------------------------------------
5+
6+
function testReconfigureAsAccessorProperty(f) {
7+
var length = 2;
8+
Object.defineProperty(f, 'length', {
9+
get: function () {
10+
return length;
11+
},
12+
set: function (v) {
13+
length = v;
14+
}
15+
});
16+
}
17+
(function testSetOnInstance() {
18+
function f() {}
19+
delete f.length;
20+
testReconfigureAsAccessorProperty(f);
21+
Object.defineProperty(Function.prototype, 'length', {
22+
writable: true
23+
});
24+
f.length = 123;
25+
f.length == 123 ? print("Pass") : print('fail');
26+
})();
27+

test/Bugs/rlexe.xml

+5
Original file line numberDiff line numberDiff line change
@@ -623,6 +623,11 @@
623623
<files>bug_6277.js</files>
624624
</default>
625625
</test>
626+
<test>
627+
<default>
628+
<files>bug_6271.js</files>
629+
</default>
630+
</test>
626631
<test>
627632
<default>
628633
<files>bug_OS23102586.js</files>

0 commit comments

Comments
 (0)