Skip to content
Snippets Groups Projects
Commit 0a3bac5f authored by Lukas Stadler's avatar Lukas Stadler
Browse files

Merge pull request #554 in G/fastr from ~LUKAS.STADLER_ORACLE.COM/fastr:setAttrFixCheck to master

* commit 'a57c10f2':
  add canStore and canSet checks to SetFixedAttributeNode
  A small fix in SetAttributeNode
parents 4e21219c a57c10f2
No related branches found
No related tags found
No related merge requests found
...@@ -68,4 +68,24 @@ public abstract class AttributeAccessNode extends RBaseNode { ...@@ -68,4 +68,24 @@ public abstract class AttributeAccessNode extends RBaseNode {
return shape != null && shape.check(attrs); return shape != null && shape.check(attrs);
} }
protected static Shape defineProperty(Shape oldShape, Object name, Object value) {
return oldShape.defineProperty(name, value, 0);
}
/**
* There is a subtle difference between {@link Location#canSet} and {@link Location#canStore}.
* We need {@link Location#canSet} for the guard of {@code setExistingAttrCached} because there
* we call {@link Location#set}. We use the more relaxed {@link Location#canStore} for the guard
* of {@code setNewAttrCached} because there we perform a shape transition, i.e., we are not
* actually setting the value of the new location - we only transition to this location as part
* of the shape change.
*/
protected static boolean canSet(Location location, Object value) {
return location.canSet(value);
}
/** See {@link #canSet} for the difference between the two methods. */
protected static boolean canStore(Location location, Object value) {
return location.canStore(value);
}
} }
...@@ -78,9 +78,9 @@ public abstract class SetAttributeNode extends AttributeAccessNode { ...@@ -78,9 +78,9 @@ public abstract class SetAttributeNode extends AttributeAccessNode {
@Cached("lookupShape(attrs)") Shape shape, @Cached("lookupShape(attrs)") Shape shape,
@Cached("lookupLocation(shape, name, value)") Location location) { @Cached("lookupLocation(shape, name, value)") Location location) {
try { try {
location.set(attrs, value); location.set(attrs, value, shape);
} catch (IncompatibleLocationException | FinalLocationException ex) { } catch (IncompatibleLocationException | FinalLocationException ex) {
RInternalError.reportError(ex); throw RInternalError.shouldNotReachHere(ex);
} }
} }
...@@ -105,7 +105,7 @@ public abstract class SetAttributeNode extends AttributeAccessNode { ...@@ -105,7 +105,7 @@ public abstract class SetAttributeNode extends AttributeAccessNode {
try { try {
newLocation.set(attrs, value, oldShape, newShape); newLocation.set(attrs, value, oldShape, newShape);
} catch (IncompatibleLocationException ex) { } catch (IncompatibleLocationException ex) {
RInternalError.reportError(ex); throw RInternalError.shouldNotReachHere(ex);
} }
} }
...@@ -189,26 +189,4 @@ public abstract class SetAttributeNode extends AttributeAccessNode { ...@@ -189,26 +189,4 @@ public abstract class SetAttributeNode extends AttributeAccessNode {
return location; return location;
} }
protected static Shape defineProperty(Shape oldShape, Object name, Object value) {
return oldShape.defineProperty(name, value, 0);
}
/**
* There is a subtle difference between {@link Location#canSet} and {@link Location#canStore}.
* We need {@link Location#canSet} for the guard of {@link #setExistingAttrCached} because there
* we call {@link Location#set}. We use the more relaxed {@link Location#canStore} for the guard
* of {@link #setNewAttrCached} because there we perform a shape transition, i.e., we are not
* actually setting the value of the new location - we only transition to this location as part
* of the shape change.
*/
protected static boolean canSet(Location location, Object value) {
return location.canSet(value);
}
/** See {@link #canSet} for the difference between the two methods. */
protected static boolean canStore(Location location, Object value) {
return location.canStore(value);
}
} }
...@@ -86,7 +86,7 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode { ...@@ -86,7 +86,7 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode {
public abstract void execute(Object attr, Object value); public abstract void execute(Object attr, Object value);
@Specialization(limit = "3", // @Specialization(limit = "3", //
guards = {"shapeCheck(shape, attrs)", "location != null"}, // guards = {"shapeCheck(shape, attrs)", "location != null", "canSet(location, value)"}, //
assumptions = {"shape.getValidAssumption()"}) assumptions = {"shape.getValidAssumption()"})
protected void setAttrCached(DynamicObject attrs, Object value, protected void setAttrCached(DynamicObject attrs, Object value,
@Cached("lookupShape(attrs)") Shape shape, @Cached("lookupShape(attrs)") Shape shape,
...@@ -94,12 +94,12 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode { ...@@ -94,12 +94,12 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode {
try { try {
location.set(attrs, value, shape); location.set(attrs, value, shape);
} catch (IncompatibleLocationException | FinalLocationException ex) { } catch (IncompatibleLocationException | FinalLocationException ex) {
RInternalError.reportError(ex); throw RInternalError.shouldNotReachHere(ex);
} }
} }
@Specialization(limit = "3", // @Specialization(limit = "3", //
guards = {"shapeCheck(oldShape, attrs)", "oldLocation == null"}, // guards = {"shapeCheck(oldShape, attrs)", "oldLocation == null", "canStore(newLocation, value)"}, //
assumptions = {"oldShape.getValidAssumption()", "newShape.getValidAssumption()"}) assumptions = {"oldShape.getValidAssumption()", "newShape.getValidAssumption()"})
protected static void setNewAttrCached(DynamicObject attrs, Object value, protected static void setNewAttrCached(DynamicObject attrs, Object value,
@Cached("lookupShape(attrs)") Shape oldShape, @Cached("lookupShape(attrs)") Shape oldShape,
...@@ -109,7 +109,7 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode { ...@@ -109,7 +109,7 @@ public abstract class SetFixedAttributeNode extends FixedAttributeAccessNode {
try { try {
newLocation.set(attrs, value, oldShape, newShape); newLocation.set(attrs, value, oldShape, newShape);
} catch (IncompatibleLocationException ex) { } catch (IncompatibleLocationException ex) {
RInternalError.reportError(ex); throw RInternalError.shouldNotReachHere(ex);
} }
} }
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment