From fc7a52ce073779917f4bed184172d3d813f9e099 Mon Sep 17 00:00:00 2001 From: Rustan Leino Date: Thu, 21 Mar 2024 14:18:30 -0700 Subject: [PATCH] fix: Optimize datatype wrappers only in the absence of trait parents Fixes #5233 As a bonus, allow the optimization when there are no _compiled_ fields (previously, the optimization was disabled when _any_ fields were present). --- .../Backends/DatatypeWrapperEraser.cs | 12 ++++--- .../LitTest/comp/ErasableTypeWrappers.dfy | 33 ++++++++++++++++++- .../comp/ErasableTypeWrappers.dfy.expect | 2 ++ 3 files changed, 41 insertions(+), 6 deletions(-) diff --git a/Source/DafnyCore/Backends/DatatypeWrapperEraser.cs b/Source/DafnyCore/Backends/DatatypeWrapperEraser.cs index cd70d585888..efe344eeab8 100644 --- a/Source/DafnyCore/Backends/DatatypeWrapperEraser.cs +++ b/Source/DafnyCore/Backends/DatatypeWrapperEraser.cs @@ -136,9 +136,10 @@ public static bool GetInnerTypeOfErasableDatatypeWrapper(DafnyOptions options, D /// 2 -- be an inductive datatype (not a "codatatype"), and /// 3 -- have exactly one non-ghost constructor, and /// 4 -- that constructor must have exactly one non-ghost destructor parameter (say, "d" of type "D"), and - /// 5 -- have no fields declared as members, and - /// 6 -- the compiled parts of type "D" must not include the datatype itself, and - /// 7 -- not be declared with {:extern} (since extern code may rely on it being there). + /// 5 -- have no non-ghost fields declared as members, and + /// 6 -- have no parent traits, and + /// 7 -- the compiled parts of type "D" must not include the datatype itself, and + /// 8 -- not be declared with {:extern} (since extern code may rely on it being there). /// /// If the conditions above apply, then the method returns true and sets the out-parameter to the core DatatypeDestructor "d". /// From this return, the compiler (that is, the caller) will arrange to compile type "dt" as type "D". @@ -161,12 +162,13 @@ public static bool IsErasableDatatypeWrapper(DafnyOptions options, DatatypeDecl } /// - /// Check for conditions 2, 3, 4, 5, and 7 (but not 0, 1, and 6) mentioned in the description of IsErasableDatatypeWrapper. + /// Check for conditions 2, 3, 4, 5, 6, and 8 (but not 0, 1, and 7) mentioned in the description of IsErasableDatatypeWrapper. /// private static bool FindUnwrappedCandidate(DafnyOptions options, DatatypeDecl datatypeDecl, out DatatypeDestructor coreDtor) { if (datatypeDecl is IndDatatypeDecl && !datatypeDecl.IsExtern(options, out _, out _) && - !datatypeDecl.Members.Any(member => member is Field)) { + !datatypeDecl.Members.Any(member => member is Field { IsGhost: false }) && + datatypeDecl.ParentTraits.Count == 0) { var nonGhostConstructors = datatypeDecl.Ctors.Where(ctor => !ctor.IsGhost).ToList(); if (nonGhostConstructors.Count == 1) { // there is exactly one non-ghost constructor diff --git a/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy b/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy index 22bb3ed1de0..d2ef145f661 100644 --- a/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy +++ b/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy @@ -1,4 +1,4 @@ -// RUN: %testDafnyForEachCompiler --refresh-exit-code=0 "%s" -- --relax-definite-assignment --spill-translation --unicode-char:false +// RUN: %testDafnyForEachCompiler --refresh-exit-code=0 "%s" -- --relax-definite-assignment --spill-translation --unicode-char:false --type-system-refresh --general-traits=datatype datatype SingletonRecord = SingletonRecord(u: int) datatype GhostOrNot = ghost Ghost(a: int, b: int) | Compiled(x: int) @@ -18,6 +18,7 @@ method Main() { TestMembers(); OptimizationChecks.Test(); PrintRegressionTests.Test(); + ConditionsThatDisableTheOptimization.Test(); } method TestTargetTypesAndConstructors() { @@ -412,3 +413,33 @@ module PrintRegressionTests { print x, "\n"; } } + +// -------------------------------------------------------------------------------- + +module ConditionsThatDisableTheOptimization { + trait Trait { + function F(): int + } + datatype WithTrait extends Trait = WithTrait(u: int) + { + function F(): int { 5 } + } + + datatype WithCompiledField = WithCompiledField(u: int) + { + const n: int := 100 + } + + datatype WithGhostField = WithGhostField(u: int) + { + ghost const n: int := 100 + } + + method Test() { + var t0 := WithTrait(40); + var t1 := WithCompiledField(41); + var t2 := WithGhostField(42); + print t0, " ", t0.F(), " ", (t0 as Trait).F(), "\n"; // WithTrait(40) 5 5 + print t1, " ", t1.n, " ", t2, "\n"; // WithCompiledField(41) 100 42 + } +} diff --git a/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy.expect b/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy.expect index e9ca5b48732..1466b3dce90 100644 --- a/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy.expect +++ b/Source/IntegrationTests/TestFiles/LitTests/LitTest/comp/ErasableTypeWrappers.dfy.expect @@ -18,3 +18,5 @@ HasConst.MakeC(4) (4, 4) OptimizationChecks.Ints.Ints(5, 7) OptimizationChecks.Ints.Ints(5, 7) OptimizationChecks.Ints.AnotherIntsWrapper(OptimizationChecks.Ints.Ints(5, 7)) D D 5 5 +ConditionsThatDisableTheOptimization.WithTrait.WithTrait(40) 5 5 +ConditionsThatDisableTheOptimization.WithCompiledField.WithCompiledField(41) 100 42