Skip to content

Commit ac79c21

Browse files
authored
Merge branch 'master' into _update-deps/runtimeverification/k
2 parents 73d71de + 296dddd commit ac79c21

16 files changed

+16111
-41
lines changed

kmir/pyproject.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ build-backend = "poetry.core.masonry.api"
44

55
[tool.poetry]
66
name = "kmir"
7-
version = "0.3.98"
7+
version = "0.3.99"
88
description = ""
99
authors = [
1010
"Runtime Verification, Inc. <[email protected]>",

kmir/src/kmir/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
from typing import Final
22

3-
VERSION: Final = '0.3.98'
3+
VERSION: Final = '0.3.99'

kmir/src/kmir/kdist/mir-semantics/rt/data.md

+152-31
Original file line numberDiff line numberDiff line change
@@ -588,7 +588,11 @@ The `Value` sort above operates at a higher level than the bytes representation
588588
Integer(Bytes2Int(BYTES, LE, Signed), 128, true)
589589
requires lengthBytes(BYTES) ==Int 16
590590
// Isize for 64bit platforms
591-
rule #decodeConstant(constantKindAllocated(allocation(BYTES, _, _, _)), rigidTyInt(intTyIsize)) => Integer(Bytes2Int(BYTES, LE, Signed), 64, false) requires lengthBytes(BYTES) ==Int 8
591+
rule #decodeConstant(constantKindAllocated(allocation(BYTES, _, _, _)), rigidTyInt(intTyIsize))
592+
=>
593+
Integer(Bytes2Int(BYTES, LE, Signed), 64, true)
594+
requires lengthBytes(BYTES) ==Int 8
595+
592596
/////////////////////////////////////////////////////////////////////////////////////////////////
593597
// TODO Float decoding: not supported natively in K
594598
@@ -815,33 +819,161 @@ An important prerequisite of this rule is that when passing references to a call
815819
rule #decrementRef(TL) => #adjustRef(TL, -1)
816820
```
817821

822+
### Writing data to places with projections
818823

819-
#### Writing data to places with projections
824+
A `Deref` projection in the projections list changes the target of the write operation, while `Field` updates change the value that is being written (updating just one field of it, recursively). `Index`ing operations may have to read an index from another local, which is another rewrite. Therefore a simple update _function_ cannot cater for all projections, neither can a rewrite (the context of the recursion would need to be held explicitly).
820825

821-
When writing data to a place with projections, the updated value gets constructed recursively by a function over the projections.
826+
The solution is to use rewrite operations in a downward pass through the projections, and build the resulting updated value in an upward pass with information collected in the downward one.
822827

823828
```k
824-
syntax TypedLocal ::= #updateProjected( TypedLocal, ProjectionElems, TypedLocal) [function]
829+
syntax WriteTo ::= toLocal ( Int )
830+
| toStack ( Int , Local )
825831
826-
rule #updateProjected(_, .ProjectionElems, NEW) => NEW
832+
syntax KItem ::= #projectedUpdate ( WriteTo , TypedLocal, ProjectionElems, TypedLocal, Contexts , Bool )
827833
828-
rule #updateProjected(
829-
typedLocal(Aggregate(ARGS), TY, MUT),
830-
projectionElemField(fieldIdx(I), _TY) PROJS,
831-
NEW)
832-
=>
833-
typedLocal(Aggregate(ARGS[I <- #updateProjected({ARGS[I]}:>TypedLocal, PROJS, NEW)]), TY, MUT)
834+
syntax TypedLocal ::= #buildUpdate ( TypedLocal, Contexts ) [function]
835+
836+
// retains information about the value that was deconstructed by a projection
837+
syntax Context ::= CtxField( Ty, List, Int )
838+
// | array context will be added here
839+
840+
syntax Contexts ::= List{Context, ""}
841+
842+
rule #buildUpdate(VAL, .Contexts) => VAL
843+
844+
rule #buildUpdate(VAL, CtxField(TY, ARGS, I) CTXS)
845+
=> #buildUpdate(typedLocal(Aggregate(ARGS[I <- VAL]), TY, mutabilityMut), CTXS)
846+
847+
rule <k> #projectedUpdate(
848+
DEST,
849+
typedLocal(Aggregate(ARGS), TY, MUT),
850+
projectionElemField(fieldIdx(I), _) PROJS,
851+
UPDATE,
852+
CTXTS,
853+
FORCE
854+
) =>
855+
#projectedUpdate(DEST, {ARGS[I]}:>TypedLocal, PROJS, UPDATE, CtxField(TY, ARGS, I) CTXTS, FORCE)
856+
...
857+
</k>
858+
requires 0 <=Int I
859+
andBool I <Int size(ARGS)
860+
andBool isTypedLocal(ARGS[I])
861+
andBool (FORCE orBool MUT ==K mutabilityMut)
862+
863+
864+
rule <k> #projectedUpdate(
865+
_DEST,
866+
typedLocal(Reference(OFFSET, place(LOCAL, PLACEPROJ), MUT), _, _),
867+
projectionElemDeref PROJS,
868+
UPDATE,
869+
_CTXTS,
870+
FORCE
871+
)
872+
=>
873+
#projectedUpdate(
874+
toStack(OFFSET, LOCAL),
875+
#localFromFrame({STACK[OFFSET -Int 1]}:>StackFrame, LOCAL, OFFSET),
876+
appendP(PLACEPROJ, PROJS), // apply reference projections first, then rest
877+
UPDATE,
878+
.Contexts, // previous contexts obsolete
879+
FORCE
880+
)
881+
...
882+
</k>
883+
<stack> STACK </stack>
884+
requires 0 <Int OFFSET
885+
andBool OFFSET <=Int size(STACK)
886+
andBool isStackFrame(STACK[OFFSET -Int 1])
887+
andBool (FORCE orBool MUT ==K mutabilityMut)
888+
[preserves-definedness]
889+
890+
rule <k> #projectedUpdate(
891+
_DEST,
892+
typedLocal(Reference(OFFSET, place(local(I), PLACEPROJ), MUT), _, _),
893+
projectionElemDeref PROJS,
894+
UPDATE,
895+
_CTXTS,
896+
FORCE
897+
)
898+
=>
899+
#projectedUpdate(
900+
toLocal(I),
901+
{LOCALS[I]}:>TypedLocal,
902+
appendP(PLACEPROJ, PROJS), // apply reference projections first, then rest
903+
UPDATE,
904+
.Contexts, // previous contexts obsolete
905+
FORCE
906+
)
907+
...
908+
</k>
909+
<locals> LOCALS </locals>
910+
requires OFFSET ==Int 0
911+
andBool 0 <=Int I
912+
andBool I <Int size(LOCALS)
913+
andBool (FORCE orBool MUT ==K mutabilityMut)
914+
[preserves-definedness]
915+
916+
rule <k> #projectedUpdate(toLocal(I), _ORIGINAL, .ProjectionElems, NEW, CONTEXTS, false)
917+
=>
918+
#buildUpdate(NEW, CONTEXTS) ~> #setLocalValue(place(local(I), .ProjectionElems))
919+
...
920+
</k>
921+
922+
rule <k> #projectedUpdate(toLocal(I), _ORIGINAL, .ProjectionElems, NEW, CONTEXTS, true)
923+
=>
924+
#buildUpdate(NEW, CONTEXTS) ~> #forceSetLocal(local(I))
925+
...
926+
</k>
927+
928+
syntax KItem ::= #forceSetLocal ( Local )
929+
930+
// #forceSetLocal sets the given value unconditionally (to write Moved values)
931+
rule <k> VALUE:TypedLocal ~> #forceSetLocal(local(I))
932+
=>
933+
.K
934+
...
935+
</k>
936+
<locals> LOCALS => LOCALS[I <- VALUE] </locals>
937+
requires 0 <=Int I
938+
andBool I <Int size(LOCALS)
939+
[preserves-definedness] // valid list indexing checked
940+
941+
rule <k> #projectedUpdate(toStack(FRAME, local(I)), _ORIGINAL, .ProjectionElems, NEW, CONTEXTS, _) => .K ... </k>
942+
<stack> STACK
943+
=>
944+
STACK[(FRAME -Int 1) <-
945+
#updateStackLocal({STACK[FRAME -Int 1]}:>StackFrame, I, #buildUpdate(NEW, CONTEXTS))
946+
]
947+
</stack>
948+
requires 0 <Int FRAME
949+
andBool FRAME <=Int size(STACK)
950+
andBool isStackFrame(STACK[FRAME -Int 1])
951+
952+
syntax StackFrame ::= #updateStackLocal ( StackFrame, Int, TypedLocal ) [function]
953+
954+
rule #updateStackLocal(StackFrame(CALLER, DEST, TARGET, UNWIND, LOCALS), I, Moved)
955+
=> StackFrame(CALLER, DEST, TARGET, UNWIND, LOCALS[I <- Moved])
956+
requires 0 <=Int I
957+
andBool I <Int size(LOCALS)
958+
[preserves-definedness]
959+
960+
rule #updateStackLocal(StackFrame(CALLER, DEST, TARGET, UNWIND, LOCALS), I, typedLocal(VAL, _, _))
961+
=> StackFrame(CALLER, DEST, TARGET, UNWIND, LOCALS[I <- typedLocal(VAL, tyOfLocal({LOCALS[I]}:>TypedLocal), mutabilityMut)])
962+
requires 0 <=Int I
963+
andBool I <Int size(LOCALS)
964+
[preserves-definedness]
834965
```
835966

967+
836968
Potential errors caused by invalid projections or type mismatch will materialise as unevaluted function calls.
837969
Mutability of the nested components is not checked (but also not modified) while computing the value.
838970
We could first read the original value using `#readProjection` and compare the types to uncover these errors.
839971

972+
840973
```k
841974
rule <k> VAL ~> #setLocalValue(place(local(I), PROJ))
842975
=>
843-
// #readProjection(LOCAL, PROJ) ~> #checkTypeMatch(VAL) ~> // optional, type-check and projection check
844-
#updateProjected({LOCALS[I]}:>TypedLocal, PROJ, VAL) ~> #setLocalValue(place(local(I), .ProjectionElems))
976+
#projectedUpdate(toLocal(I), {LOCALS[I]}:>TypedLocal, PROJ, VAL, .Contexts, false)
845977
...
846978
</k>
847979
<locals> LOCALS </locals>
@@ -850,10 +982,12 @@ We could first read the original value using `#readProjection` and compare the t
850982
andBool PROJ =/=K .ProjectionElems
851983
andBool isTypedLocal(LOCALS[I])
852984
[preserves-definedness]
985+
853986
```
854987

855-
Reading `Moved` operands requires a write operation to the read place, too, however the mutability should be ignored.
856-
Therefore a wrapper `#forceSetLocal` is used to side-step the mutability error in `#setLocalValue`.
988+
#### Moving operands under projections
989+
990+
Reading `Moved` operands requires a write operation to the read place, too, however the mutability should be ignored while computing the update.
857991

858992
```k
859993
rule <k> #readOperand(operandMove(place(local(I) #as LOCAL, PROJECTIONS)))
@@ -870,27 +1004,14 @@ Therefore a wrapper `#forceSetLocal` is used to side-step the mutability error i
8701004
[preserves-definedness] // valid list indexing checked
8711005
8721006
syntax KItem ::= #markMoved ( TypedLocal, Local, ProjectionElems )
873-
| #forceSetLocal ( Local )
8741007
875-
rule <k> VAL:TypedLocal ~> #markMoved(OLDLOCAL, LOCAL, PROJECTIONS) ~> CONT
1008+
rule <k> VAL:TypedLocal ~> #markMoved(OLDLOCAL, local(I), PROJECTIONS) ~> CONT
8761009
=>
877-
#updateProjected(OLDLOCAL, PROJECTIONS, Moved)
878-
~> #forceSetLocal(LOCAL)
1010+
#projectedUpdate(toLocal(I), OLDLOCAL, PROJECTIONS, Moved, .Contexts, true)
8791011
~> VAL
8801012
~> CONT
8811013
</k>
882-
[preserves-definedness] // projections already used when reading, updateProjected should succeed
883-
884-
// #forceSetLocal sets the given value unconditionally
885-
rule <k> VALUE:TypedLocal ~> #forceSetLocal(local(I))
886-
=>
887-
.K
888-
...
889-
</k>
890-
<locals> LOCALS => LOCALS[I <- VALUE] </locals>
891-
requires 0 <=Int I
892-
andBool I <Int size(LOCALS)
893-
[preserves-definedness] // valid list indexing checked
1014+
[preserves-definedness] // projections already used when reading
8941015
```
8951016

8961017
### Primitive operations on numeric data

kmir/src/tests/integration/data/exec-smir/references/doubleRef.smir.json

+2,695-1
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
fn main() {
2+
let mut x = 42i8;
3+
4+
f(&mut x);
5+
6+
assert!(x == 32);
7+
8+
let xref = &mut x;
9+
10+
*xref = 22;
11+
12+
assert!(x == 22);
13+
}
14+
15+
fn f(y: &mut i8) {
16+
*y = 32
17+
}

0 commit comments

Comments
 (0)