Skip to content

Commit

Permalink
Merge branch 'topic/958' into 'master'
Browse files Browse the repository at this point in the history
Support Subtype Predicates self-reference

Closes #958

See merge request eng/libadalang/libadalang!1326
  • Loading branch information
thvnx committed Oct 2, 2023
2 parents 539e510 + d280b5c commit 6f6d86d
Show file tree
Hide file tree
Showing 13 changed files with 996 additions and 32 deletions.
154 changes: 127 additions & 27 deletions ada/ast.py
Original file line number Diff line number Diff line change
Expand Up @@ -712,7 +712,10 @@ def xref_equation():
"""
pass

xref_stop_resolution = Property(False)
xref_stop_resolution = Property(
False,
dynamic_vars=[env, origin]
)
stop_resolution_equation = Property(
LogicTrue(),
dynamic_vars=[env, origin]
Expand Down Expand Up @@ -847,27 +850,28 @@ def resolve_names_from_closest_entry_point_impl():
res=Entity.parent
._.resolve_names_from_closest_entry_point_impl:

Cond(
# Resolution failed for the parent, so return None as well
res == No(T.LexicalEnv),
env.bind(
res,

# Resolution succeeded for the parent and this is a stop
# resolution, so re-use the parent environment to resolve
# Self's names.
Entity.xref_stop_resolution,
env.bind(
res,
origin.bind(
Self.origin_node,
If(Entity.resolve_own_names, res, No(LexicalEnv))
origin.bind(
Self.origin_node,
Cond(
# Resolution failed for the parent, so return None
# as well.
res == No(T.LexicalEnv),
res,

# Resolution succeeded for the parent and this is a
# stop resolution, so re-use the parent environment
# to resolve Self's names.
Entity.xref_stop_resolution,
If(Entity.resolve_own_names, res, No(LexicalEnv)),

# Resolution succeeded but there is nothing to do
# on that particular node: return the parent
# environment, so that deeper children can use it.
res
)
),

# Resolution succeeded but there is nothing to do on that
# particular node: return the parent environment, so that
# deeper children can use it.
res
)
)
)
)
Expand Down Expand Up @@ -1641,7 +1645,13 @@ def origin_node():
Return a null node iff we are in the definition of an aspect clause
where sequential lookup needs to be deactivated. Return Self otherwise.
"""
return If(Self.in_contract, No(T.AdaNode), Self)
return Cond(
Self.in_contract,
No(T.AdaNode),
Self.is_a(ExprFunction),
Self.cast(ExprFunction).expr,
Self
)

@langkit_property()
def env_hook():
Expand Down Expand Up @@ -6065,6 +6075,33 @@ def find_derived_types(root=T.AdaNode.entity):
)
))

@lazy_field(return_type=T.env_assoc)
def synthetic_type_predicate_object_decl():
"""
Create an env_assoc embedding the synthetic object declaration
required for predicates name resolution.

This property should only be called once by ``env_spec``
(``SubtypeDecl``, ``TypeDecl``).
"""
return new_env_assoc(
key=Self.name_symbol,
# This synthetic object declaration is used for resolving the
# references to its own derived/subtype identifier that can be used
# in predicates as object references. This virtual object only
# lives in the scope of the derived/subtype declaration, and has
# the same name and type than the declaration it derives from.
value=T.SyntheticTypePredicateObjectDecl.new(
name=Self.name,
# A `SubtypeDecl` has a type expression that we can reuse On
# the contrary, we have to embed the TypeDecl into a synthetic
# `TypeExpr` for `TypeDecl`s.
type_expr=Self.as_bare_entity.type_expression.node._or(
SyntheticTypeExpr.new(target_type=Self)
)
)
)

is_task_type = Property(False, doc="Whether type is a task type")

is_real_type = Property(
Expand Down Expand Up @@ -7709,7 +7746,19 @@ def predefined_operators():
dest_env=Self.node_env,
cond=Self.type_def.is_a(T.DerivedTypeDef, T.InterfaceTypeDef),
category="inherited_primitives"
)
),

# If this `TypeDecl` can have a predicate, add a synthetic object
# declaration into its environement in order to support name resolution
# of self-references that can appear in predicates (see
# `SyntheticTypePredicateObjectDecl`).
add_to_env(If(
# Try to filter which type decls can have predicate to save some
# space in envs.
Self.type_def.is_a(T.DerivedTypeDef, T.TypeAccessDef),
Entity.synthetic_type_predicate_object_decl,
No(T.env_assoc)
))
)

record_def = Property(
Expand Down Expand Up @@ -8982,6 +9031,23 @@ class SubtypeDecl(BaseSubtypeDecl):
subtype = Field(type=T.SubtypeIndication)
aspects = Field(type=T.AspectSpec)

env_spec = EnvSpec(
add_to_env_kv(Entity.name_symbol, Self),

# Subtype predicates expressions can refers to its own subtype
# declaration identifier as an object such as in::
#
# subtype Odd is Natural with
# Dynamic_Predicate => Odd mod 2 = 1;
#
# where ``Odd`` should refer to an anonymous object of the same name
# and of its own subtype. This object only virtually exists in the
# subtype environment, so we add a children environement here, just to
# hold this object.
add_env(transitive_parent=True),
add_to_env(Entity.synthetic_type_predicate_object_decl)
)

@langkit_property(return_type=T.BaseTypeDecl.entity, dynamic_vars=[origin])
def get_type():
return Entity.subtype.designated_type.match(
Expand All @@ -9001,6 +9067,32 @@ def xref_equation():
xref_entry_point = Property(True)


@synthetic
class SyntheticTypePredicateObjectDecl(BasicDecl):
"""
SyntheticTypePredicateObjectDecl is a declaration that holds a virtual
object used in type predicates to refer to an object of that type. Such as
in::

subtype Odd is Natural with
Dynamic_Predicate => Odd mod 2 = 1;

where we have to create an object named ``Odd``, and of type ``Odd`` so
that the name in the aspect expression refers to it and can be properly
resolved to the type identifier.

This node has no existance in the Ada RM, it's only used for internal name
resolution purposes.
"""
name = UserField(type=T.DefiningName, public=False)
type_expr = UserField(type=T.TypeExpr, public=False)

aspects = NullField()

type_expression = Property(Self.type_expr.as_entity)
defining_names = Property(Self.name.as_entity.singleton)


@synthetic
class DiscreteBaseSubtypeDecl(BaseSubtypeDecl):
"""
Expand Down Expand Up @@ -9973,7 +10065,7 @@ def xref_initial_env():
"Pre", "Post", "Pre'Class", "Post'Class",
"Precondition", "Postcondition",
"Precondition'Class", "Postcondition'Class",
"Test_Case", "Contract_Cases"
"Test_Case", "Contract_Cases", "Predicate"
),
Entity.associated_entities.at(0).children_env,
Entity.children_env
Expand Down Expand Up @@ -10159,7 +10251,7 @@ def associated_entity_names():
'Volatile', 'Volatile_Components', 'Unchecked_Union',
'Atomic', 'Atomic_Components', 'No_Return', "Discard_Names",
"Independent", "Independent_Components", "Asynchronous",
"Interrupt_Handler", "Attach_Handler",
"Interrupt_Handler", "Attach_Handler", "Predicate"
),
Entity.args.at(0)._.assoc_expr.cast(T.Name)._.singleton,

Expand Down Expand Up @@ -14688,7 +14780,13 @@ def is_write_reference():

# Handle out/inout param case
lambda p=T.ParamAssoc: If(
p.parent.parent.cast(CallExpr)._.is_type_conversion,
env.bind(
Self.node_env,
origin.bind(
Self.origin_node,
p.parent.parent.cast(CallExpr)._.is_type_conversion,
)
),
p.parent.parent.cast(CallExpr).is_write_reference,
p.get_params.any(
lambda m:
Expand Down Expand Up @@ -15211,14 +15309,16 @@ def entity_equation(s=T.BasicDecl.entity, root=T.Name):
LogicFalse()
)

@langkit_property(return_type=Bool)
@langkit_property(return_type=Bool, dynamic_vars=[env, origin])
def is_type_conversion():
"""
Return whether this CallExpr actually represents a type conversion.
"""
return And(
Not(Entity.name.is_a(QualExpr)),
Not(Entity.name.name_designated_type.is_null)
# Directly call designated_type_impl instead of
# name_designated_type to propagate Self's origin.
Not(Entity.name.designated_type_impl.is_null)
)

xref_stop_resolution = Property(
Expand Down
3 changes: 3 additions & 0 deletions extensions/src/libadalang-expr_eval.adb
Original file line number Diff line number Diff line change
Expand Up @@ -317,6 +317,9 @@ package body Libadalang.Expr_Eval is
when Ada_Anonymous_Expr_Decl =>
return Expr_Eval (D.As_Anonymous_Expr_Decl.F_Expr);

when Ada_Synthetic_Type_Predicate_Object_Decl =>
return Eval_Decl (D.Parent.As_Basic_Decl);

when others =>
raise Property_Error
with "Cannot eval decl " & D.Kind'Image;
Expand Down
3 changes: 3 additions & 0 deletions testsuite/tests/lexical_envs/envs_1/test.out
Original file line number Diff line number Diff line change
Expand Up @@ -113,3 +113,6 @@ $root = LexEnv(Static_Primary, Parent=null):
$root = LexEnv(Static_Primary, Parent=null):
standard: [<PackageDecl ["Standard"] __standard:1:1-129:14>]

@3 = LexEnv(Static_Primary, Parent=@2, Node=<SubtypeDecl ["Lol"] input:2:4-2:41>):
lol: [<SyntheticTypePredicateObjectDecl ["Lol"] input:2:4-2:41>]

3 changes: 3 additions & 0 deletions testsuite/tests/lexical_envs/envs_2/test.out
Original file line number Diff line number Diff line change
Expand Up @@ -133,3 +133,6 @@ $root = LexEnv(Static_Primary, Parent=null):
@4 = LexEnv(Static_Primary, Parent=@3, Node=<DeclBlock input:4:7-8:11>):
lol: [<SubtypeDecl ["Lol"] input:5:10-5:47>]

@5 = LexEnv(Static_Primary, Parent=@4, Node=<SubtypeDecl ["Lol"] input:5:10-5:47>):
lol: [<SyntheticTypePredicateObjectDecl ["Lol"] input:5:10-5:47>]

Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ Expr: <AttributeRef test.adb:6:28-6:35>
expected type: <ConcreteTypeDecl ["Integer"] __standard:4:3-4:54>
Expr: <Id "U" test.adb:6:28-6:29>
references: <DefiningName "U" test.adb:5:12-5:13>
type: None
type: <ConcreteTypeDecl ["U"] test.adb:5:7-6:40>
expected type: None
Expr: <Id "First" test.adb:6:30-6:35>
references: None
Expand Down
21 changes: 21 additions & 0 deletions testsuite/tests/name_resolution/type_predicates/conflict.adb
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
-- This test ensure that `Name` in the CallExpr `Name (Name'First)` is correctly
-- resolved to the `To_String`'s Name parameter while `Name (1 .. 4)` in the
-- `Predicate` pragma is correctly bind to the subtype `Name`.

procedure Conflict is
subtype Name is String (1 .. 10);

function To_String (Name : String) return String is
(Name (Name'First) & "!");
pragma Test_Statement;

pragma Predicate (Name, Name (1 .. 4) = "FAIL");
pragma Test_Statement;

function P (N : String) return Boolean is (True);

pragma Predicate (Name, P (Name (1 .. 4)));
pragma Test_Statement;
begin
null;
end Conflict;
33 changes: 33 additions & 0 deletions testsuite/tests/name_resolution/type_predicates/derived.adb
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
procedure Derived is
function F (S : String) return Boolean is (True);

type String_T is new String with
Dynamic_Predicate => F (String (String_T (1 .. 90)));
pragma Test_Block;

type Count is new Integer with
Static_Predicate => Count /= 10;
pragma Test_Block;

type String_T2 is new String with
Predicate => F (String (String_T2 (1 .. 90)));
pragma Test_Block;

type Count2 is new Integer with
Predicate => Count2 /= 10;
pragma Test_Block;

type String_T3 is new String;
pragma Predicate
(Entity => String_T3,
Check => F (String (String_T3 (1 .. 90))));
pragma Test_Statement;

type Count3 is new Integer;
pragma Predicate
(Entity => Count3,
Check => Count3 /= 10);
pragma Test_Statement;
begin
null;
end Derived;
20 changes: 20 additions & 0 deletions testsuite/tests/name_resolution/type_predicates/other.adb
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
procedure Other is
type Color is (White, Red, Yellow, Green, Blue, Brown, Black)
with Static_Predicate => Color in Red;
pragma Test_Block;

type Column is range 1 .. 72
with Predicate => Column mod 2 = 1;
pragma Test_Block;

type Table is array (1 .. 10) of Integer;
pragma Predicate (Table, Table'First = 1);
pragma Test_Statement;

type R is record
I : Integer;
end record with Predicate => R.I = 1;
pragma Test_Block;
begin
null;
end Other;
33 changes: 33 additions & 0 deletions testsuite/tests/name_resolution/type_predicates/test.adb
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
procedure Test is
function F (S : String) return Boolean is (True);

subtype String_T is String (1 .. 99) with
Dynamic_Predicate => F (String_T (1 .. 90));
pragma Test_Block;

subtype Count is Integer with
Static_Predicate => Count /= 10;
pragma Test_Block;

subtype String_T2 is String (1 .. 99) with
Predicate => F (String_T2 (1 .. 90));
pragma Test_Block;

subtype Count2 is Integer with
Predicate => Count2 /= 10;
pragma Test_Block;

subtype String_T3 is String (1 .. 99);
pragma Predicate
(Entity => String_T3,
Check => F (String_T3 (1 .. 90)));
pragma Test_Statement;

subtype Count3 is Integer;
pragma Predicate
(Entity => Count3,
Check => Count3 /= 10);
pragma Test_Statement;
begin
null;
end Test;
Loading

0 comments on commit 6f6d86d

Please sign in to comment.