@@ -588,7 +588,11 @@ The `Value` sort above operates at a higher level than the bytes representation
588
588
Integer(Bytes2Int(BYTES, LE, Signed), 128, true)
589
589
requires lengthBytes(BYTES) ==Int 16
590
590
// 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
+
592
596
/////////////////////////////////////////////////////////////////////////////////////////////////
593
597
// TODO Float decoding: not supported natively in K
594
598
@@ -815,33 +819,161 @@ An important prerequisite of this rule is that when passing references to a call
815
819
rule #decrementRef(TL) => #adjustRef(TL, -1)
816
820
```
817
821
822
+ ### Writing data to places with projections
818
823
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).
820
825
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 .
822
827
823
828
``` k
824
- syntax TypedLocal ::= #updateProjected( TypedLocal, ProjectionElems, TypedLocal) [function]
829
+ syntax WriteTo ::= toLocal ( Int )
830
+ | toStack ( Int , Local )
825
831
826
- rule #updateProjected(_, . ProjectionElems, NEW) => NEW
832
+ syntax KItem ::= #projectedUpdate ( WriteTo , TypedLocal, ProjectionElems, TypedLocal, Contexts , Bool )
827
833
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]
834
965
```
835
966
967
+
836
968
Potential errors caused by invalid projections or type mismatch will materialise as unevaluted function calls.
837
969
Mutability of the nested components is not checked (but also not modified) while computing the value.
838
970
We could first read the original value using ` #readProjection ` and compare the types to uncover these errors.
839
971
972
+
840
973
``` k
841
974
rule <k> VAL ~> #setLocalValue(place(local(I), PROJ))
842
975
=>
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)
845
977
...
846
978
</k>
847
979
<locals> LOCALS </locals>
@@ -850,10 +982,12 @@ We could first read the original value using `#readProjection` and compare the t
850
982
andBool PROJ =/=K .ProjectionElems
851
983
andBool isTypedLocal(LOCALS[I])
852
984
[preserves-definedness]
985
+
853
986
```
854
987
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.
857
991
858
992
``` k
859
993
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
870
1004
[preserves-definedness] // valid list indexing checked
871
1005
872
1006
syntax KItem ::= #markMoved ( TypedLocal, Local, ProjectionElems )
873
- | #forceSetLocal ( Local )
874
1007
875
- rule <k> VAL:TypedLocal ~> #markMoved(OLDLOCAL, LOCAL , PROJECTIONS) ~> CONT
1008
+ rule <k> VAL:TypedLocal ~> #markMoved(OLDLOCAL, local(I) , PROJECTIONS) ~> CONT
876
1009
=>
877
- #updateProjected(OLDLOCAL, PROJECTIONS, Moved)
878
- ~> #forceSetLocal(LOCAL)
1010
+ #projectedUpdate(toLocal(I), OLDLOCAL, PROJECTIONS, Moved, .Contexts, true)
879
1011
~> VAL
880
1012
~> CONT
881
1013
</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
894
1015
```
895
1016
896
1017
### Primitive operations on numeric data
0 commit comments