summaryrefslogtreecommitdiff
path: root/Source/Dafny/RefinementTransformer.cs
diff options
context:
space:
mode:
authorGravatar Rustan Leino <leino@microsoft.com>2012-01-09 23:16:37 -0800
committerGravatar Rustan Leino <leino@microsoft.com>2012-01-09 23:16:37 -0800
commit814a47c0b8d492f7dcfe9b815c74350b4893f4a6 (patch)
treeda65f0d02489ff6934fcfd905b25307540df13ce /Source/Dafny/RefinementTransformer.cs
parent4dc9d5a6a2b2e1ae3513016b34341dbf0959da3c (diff)
Dafny: added support for simple superposition refinements
Diffstat (limited to 'Source/Dafny/RefinementTransformer.cs')
-rw-r--r--Source/Dafny/RefinementTransformer.cs74
1 files changed, 73 insertions, 1 deletions
diff --git a/Source/Dafny/RefinementTransformer.cs b/Source/Dafny/RefinementTransformer.cs
index fb956883..d7fde895 100644
--- a/Source/Dafny/RefinementTransformer.cs
+++ b/Source/Dafny/RefinementTransformer.cs
@@ -104,6 +104,7 @@ namespace Microsoft.Dafny {
}
// Merge the declarations of prev into the declarations of m
+
foreach (var d in prev.TopLevelDecls) {
int index;
if (!declaredNames.TryGetValue(d.Name, out index)) {
@@ -513,7 +514,78 @@ namespace Microsoft.Dafny {
// -------------------------------------------------- Merging ---------------------------------------------------------------
ClassDecl MergeClass(ClassDecl nw, ClassDecl prev) {
- // TODO
+ // Create a simple name-to-member dictionary. Ignore any duplicates at this time.
+ var declaredNames = new Dictionary<string, int>();
+ for (int i = 0; i < nw.Members.Count; i++) {
+ var member = nw.Members[i];
+ if (!declaredNames.ContainsKey(member.Name)) {
+ declaredNames.Add(member.Name, i);
+ }
+ }
+
+ // Merge the declarations of prev into the declarations of m
+ foreach (var member in prev.Members) {
+ int index;
+ if (!declaredNames.TryGetValue(member.Name, out index)) {
+ nw.Members.Add(CloneMember(member));
+ } else {
+ var nwMember = nw.Members[index];
+ if (nwMember is Field) {
+ reporter.Error(nwMember, "a field declaration ({0}) in a refining class ({1}) is not allowed replace an existing declaration in the refinement base", nwMember.Name, nw.Name);
+
+ } else if (nwMember is Function) {
+ var f = (Function)nwMember;
+ if (!(member is Function)) {
+ reporter.Error(nwMember, "a function declaration ({0}) can only refine a function", nwMember.Name);
+ } else {
+ var clone = (Function)CloneMember(member);
+ if (f.Decreases.Expressions.Count != 0) {
+ reporter.Error(nwMember, "decreases clause on refining function not supported");
+ }
+ if (f.Reads.Count != 0) {
+ reporter.Error(nwMember, "a refining function is not allowed to extend the reads clause");
+ }
+ if (f.Req.Count != 0) {
+ reporter.Error(nwMember, "a refining function is not allowed to add preconditions");
+ }
+ // TODO: check agreement for f.TypeArgs, f.ResultType, f.Formals, and flags. For now, they are ignored and assumed to be the same.
+ foreach (var e in f.Ens) {
+ clone.Ens.Add(e);
+ }
+ if (f.Body != null) {
+ reporter.Error(nwMember, "body of refining function is not yet supported"); // TODO
+ }
+ nw.Members[index] = clone;
+ }
+
+ } else {
+ var m = (Method)nwMember;
+ if (!(member is Method)) {
+ reporter.Error(nwMember, "a method declaration ({0}) can only refine a method", nwMember.Name);
+ } else {
+ var clone = (Method)CloneMember(member);
+ if (m.Decreases.Expressions.Count != 0) {
+ reporter.Error(nwMember, "decreases clause on refining method not supported");
+ }
+ if (m.Mod.Expressions.Count != 0) {
+ reporter.Error(nwMember, "a refining method is not allowed to extend the modifies clause");
+ }
+ if (m.Req.Count != 0) {
+ reporter.Error(nwMember, "a refining method is not allowed to add preconditions");
+ }
+ // TODO: check agreement for m.TypeArgs, m.Ins, m.Outs, and flags. For now, they are ignored and assumed to be the same.
+ foreach (var e in m.Ens) {
+ clone.Ens.Add(e);
+ }
+ if (m.Body != null) {
+ reporter.Error(nwMember, "body of refining method is not yet supported"); // TODO
+ }
+ nw.Members[index] = clone;
+ }
+ }
+ }
+ }
+
return nw;
}
}