Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Integrate with atree inlining and data deduplication #2882

Merged
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
dbbb777
Integrate with atree inlining and deduplication
fxamacker Oct 19, 2023
ca1a9ea
Fix lint warnings
fxamacker Oct 19, 2023
bc7b622
Use UnreachableError in panic
fxamacker Oct 19, 2023
bfd43ca
Simplify TypeInfo.Identifier()
fxamacker Oct 19, 2023
713c6c3
Reformat with extra lines for readability
fxamacker Oct 19, 2023
e9ae0d2
meter allocations of array static types
turbolent Oct 19, 2023
cba775c
disable atree validation for memory metering tests, adjust expected v…
turbolent Oct 19, 2023
5e51834
pass proper location range
turbolent Oct 20, 2023
a93c048
simplify dictionary iterator to just support key iteration, which all…
turbolent Oct 20, 2023
da81afe
use type ID constructor function
turbolent Oct 20, 2023
936c6e4
SomeStorable.Encode should not ever be called. use proper encode func…
turbolent Oct 20, 2023
13950f9
address TODOs: Identifier functions can use StaticType.ID, Copy can r…
turbolent Oct 20, 2023
f52f42a
Update to use latest Atree inlining
fxamacker Oct 22, 2023
aa087cf
Refactor SomeStorable encoding
fxamacker Oct 24, 2023
1b673ea
Update to use latest Atree inlining
fxamacker Oct 24, 2023
f01949a
lower size of generated containers
turbolent Oct 24, 2023
e268003
add tests for "moving" struct and composite between arrays, and mutat…
turbolent Oct 24, 2023
2f3642f
Cleanup and add more comments
fxamacker Oct 25, 2023
5c2ee46
Add another test for SomeValue encoding
fxamacker Oct 25, 2023
79f3ab3
Add smoke test flag smokeTestSeed
fxamacker Oct 25, 2023
593cead
Add reproducer for false alarm (unreachable slab)
fxamacker Oct 31, 2023
15343c1
add atRoot parameter to Value.DeepRemove. only validate atree value i…
turbolent Oct 31, 2023
a81c720
always perform atree value validation, only perform atree storage val…
turbolent Oct 31, 2023
211ded1
Merge pull request #2920 from onflow/bastian/deep-remove-atroot-flag
turbolent Nov 1, 2023
b98a84e
Add reproducer for false alarm (unreachable slab)
fxamacker Nov 9, 2023
1ecf35c
Add `atRoot` parameter to Value.Transfer()
fxamacker Nov 9, 2023
d35b7d6
Use Atree non-readonly iterators in Transfer()
fxamacker Nov 21, 2023
7ddc21d
Rename atRoot param for Transfer() and DeepRemove()
fxamacker Nov 21, 2023
d08a2a8
Use hasNoParentContainer for inner value DeepRemove()
fxamacker Nov 21, 2023
a6df504
Update StorageMapKey funcs needed by atree iterators
fxamacker Jan 18, 2024
ea8fd35
Merge branch 'feature/atree-register-inlining' into fxamacker/integra…
turbolent Jan 24, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ require (
github.com/kr/pretty v0.3.1
github.com/leanovate/gopter v0.2.9
github.com/logrusorgru/aurora/v4 v4.0.0
github.com/onflow/atree v0.6.1-0.20230706233410-78f997992600
github.com/onflow/atree v0.6.1-0.20231019010146-eb9a50354126
github.com/rivo/uniseg v0.4.4
github.com/schollz/progressbar/v3 v3.13.1
github.com/stretchr/testify v1.8.4
Expand Down
4 changes: 2 additions & 2 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -52,8 +52,8 @@ github.com/mattn/go-tty v0.0.4/go.mod h1:u5GGXBtZU6RQoKV8gY5W6UhMudbR5vXnUe7j3px
github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db h1:62I3jR2EmQ4l5rM/4FEfDWcRD+abF5XlKShorW5LRoQ=
github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db/go.mod h1:l0dey0ia/Uv7NcFFVbCLtqEBQbrT4OCwCSKTEv6enCw=
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs=
github.com/onflow/atree v0.6.1-0.20230706233410-78f997992600 h1:OOnC8buJso6dSZ6W+RGQDnJW9UTh7HfEGyS/ivXJ7NI=
github.com/onflow/atree v0.6.1-0.20230706233410-78f997992600/go.mod h1:7YNAyCd5JENq+NzH+fR1ABUZVzbSq9dkt0+5fZH3L2A=
github.com/onflow/atree v0.6.1-0.20231019010146-eb9a50354126 h1:pGgWNlvLikhoq6DpD7sTv/G783WPXBFUlRfTuNVqavM=
github.com/onflow/atree v0.6.1-0.20231019010146-eb9a50354126/go.mod h1:7YNAyCd5JENq+NzH+fR1ABUZVzbSq9dkt0+5fZH3L2A=
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
github.com/pkg/term v1.2.0-beta.2 h1:L3y/h2jkuBVFdWiJvNfYfKmzcCnILw7mJWm2JQuMppw=
github.com/pkg/term v1.2.0-beta.2/go.mod h1:E25nymQcrSllhX42Ok8MRm1+hyBdHY0dCeiKZ9jpNGw=
Expand Down
12 changes: 9 additions & 3 deletions runtime/cmd/decode-state-values/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import (
"github.com/schollz/progressbar/v3"

"github.com/onflow/cadence/runtime/common"
runtimeErr "github.com/onflow/cadence/runtime/errors"
"github.com/onflow/cadence/runtime/interpreter"
)

Expand Down Expand Up @@ -91,8 +92,8 @@ func storageKeyToSlabID(address atree.Address, key string) atree.SlabID {
return atree.NewSlabID(address, index)
}

func decodeStorable(decoder *cbor.StreamDecoder, storableSlabStorageID atree.SlabID) (atree.Storable, error) {
return interpreter.DecodeStorable(decoder, storableSlabStorageID, nil)
func decodeStorable(decoder *cbor.StreamDecoder, storableSlabStorageID atree.SlabID, inlinedExtraData []atree.ExtraData) (atree.Storable, error) {
return interpreter.DecodeStorable(decoder, storableSlabStorageID, inlinedExtraData, nil)
fxamacker marked this conversation as resolved.
Show resolved Hide resolved
}

func decodeTypeInfo(decoder *cbor.StreamDecoder) (atree.TypeInfo, error) {
Expand Down Expand Up @@ -136,6 +137,11 @@ type slabStorage struct{}

var _ atree.SlabStorage = &slabStorage{}

func (s *slabStorage) RetrieveIfLoaded(atree.SlabID) atree.Slab {
// RetrieveIfLoaded() is used for loaded resource tracking. So it isn't needed here.
panic(runtimeErr.NewUnreachableError())
ramtinms marked this conversation as resolved.
Show resolved Hide resolved
}

func (s *slabStorage) Retrieve(id atree.SlabID) (atree.Slab, bool, error) {
data, ok := storage[slabIDToStorageKey(id)]
if !ok {
Expand Down Expand Up @@ -350,7 +356,7 @@ func loadStorageKey(

reader := bytes.NewReader(data)
decoder := interpreter.CBORDecMode.NewStreamDecoder(reader)
storable, err := interpreter.DecodeStorable(decoder, atree.SlabIDUndefined, nil)
storable, err := interpreter.DecodeStorable(decoder, atree.SlabIDUndefined, nil, nil)
if err != nil {
log.Printf(
"Failed to decode storable @ 0x%x %s: %s (data: %x)\n",
Expand Down
86 changes: 63 additions & 23 deletions runtime/interpreter/decode.go
Original file line number Diff line number Diff line change
Expand Up @@ -118,23 +118,39 @@ func decodeInt64(d StorableDecoder) (int64, error) {
func DecodeStorable(
decoder *cbor.StreamDecoder,
slabID atree.SlabID,
inlinedExtraData []atree.ExtraData,
memoryGauge common.MemoryGauge,
) (
atree.Storable,
error,
) {
return NewStorableDecoder(decoder, slabID, memoryGauge).decodeStorable()
return NewStorableDecoder(decoder, slabID, inlinedExtraData, memoryGauge).decodeStorable()
}

func newStorableDecoderFunc(memoryGauge common.MemoryGauge) atree.StorableDecoder {
return func(
decoder *cbor.StreamDecoder,
slabID atree.SlabID,
inlinedExtraData []atree.ExtraData,
) (
atree.Storable,
error,
) {
return NewStorableDecoder(decoder, slabID, inlinedExtraData, memoryGauge).decodeStorable()
}
}

func NewStorableDecoder(
decoder *cbor.StreamDecoder,
slabID atree.SlabID,
inlinedExtraData []atree.ExtraData,
memoryGauge common.MemoryGauge,
) StorableDecoder {
return StorableDecoder{
decoder: decoder,
memoryGauge: memoryGauge,
slabID: slabID,
decoder: decoder,
memoryGauge: memoryGauge,
slabID: slabID,
inlinedExtraData: inlinedExtraData,
TypeDecoder: NewTypeDecoder(
decoder,
memoryGauge,
Expand All @@ -144,9 +160,10 @@ func NewStorableDecoder(

type StorableDecoder struct {
TypeDecoder
memoryGauge common.MemoryGauge
decoder *cbor.StreamDecoder
slabID atree.SlabID
memoryGauge common.MemoryGauge
decoder *cbor.StreamDecoder
slabID atree.SlabID
inlinedExtraData []atree.ExtraData
}

func (d StorableDecoder) decodeStorable() (atree.Storable, error) {
Expand Down Expand Up @@ -203,6 +220,29 @@ func (d StorableDecoder) decodeStorable() (atree.Storable, error) {
case atree.CBORTagSlabID:
return atree.DecodeSlabIDStorable(d.decoder)

case atree.CBORTagInlinedArray:
return atree.DecodeInlinedArrayStorable(
d.decoder,
newStorableDecoderFunc(d.memoryGauge),
d.slabID,
d.inlinedExtraData)

case atree.CBORTagInlinedMap:
return atree.DecodeInlinedMapStorable(
d.decoder,
newStorableDecoderFunc(d.memoryGauge),
d.slabID,
d.inlinedExtraData,
)

case atree.CBORTagInlinedCompactMap:
return atree.DecodeInlinedCompactMapStorable(
d.decoder,
newStorableDecoderFunc(d.memoryGauge),
d.slabID,
d.inlinedExtraData,
)

case CBORTagVoidValue:
err := d.decoder.Skip()
if err != nil {
Expand Down Expand Up @@ -1560,35 +1600,35 @@ func (d TypeDecoder) decodeInterfaceStaticType() (InterfaceStaticType, error) {
return NewInterfaceStaticTypeComputeTypeID(d.memoryGauge, location, qualifiedIdentifier), nil
}

func (d TypeDecoder) decodeVariableSizedStaticType() (StaticType, error) {
func (d TypeDecoder) decodeVariableSizedStaticType() (VariableSizedStaticType, error) {
staticType, err := d.DecodeStaticType()
if err != nil {
return nil, errors.NewUnexpectedError(
return VariableSizedStaticType{}, errors.NewUnexpectedError(
"invalid variable-sized static type encoding: %w",
err,
)
}
return NewVariableSizedStaticType(d.memoryGauge, staticType), nil
}

func (d TypeDecoder) decodeConstantSizedStaticType() (StaticType, error) {
func (d TypeDecoder) decodeConstantSizedStaticType() (ConstantSizedStaticType, error) {

const expectedLength = encodedConstantSizedStaticTypeLength

arraySize, err := d.decoder.DecodeArrayHead()
if err != nil {
if e, ok := err.(*cbor.WrongTypeError); ok {
return nil, errors.NewUnexpectedError(
return ConstantSizedStaticType{}, errors.NewUnexpectedError(
"invalid constant-sized static type encoding: expected [%d]any, got %s",
expectedLength,
e.ActualType.String(),
)
}
return nil, err
return ConstantSizedStaticType{}, err
}

if arraySize != expectedLength {
return nil, errors.NewUnexpectedError(
return ConstantSizedStaticType{}, errors.NewUnexpectedError(
"invalid constant-sized static type encoding: expected [%d]any, got [%d]any",
expectedLength,
arraySize,
Expand All @@ -1599,17 +1639,17 @@ func (d TypeDecoder) decodeConstantSizedStaticType() (StaticType, error) {
size, err := decodeUint64(d.decoder, d.memoryGauge)
if err != nil {
if e, ok := err.(*cbor.WrongTypeError); ok {
return nil, errors.NewUnexpectedError(
return ConstantSizedStaticType{}, errors.NewUnexpectedError(
"invalid constant-sized static type size encoding: %s",
e.ActualType.String(),
)
}
return nil, err
return ConstantSizedStaticType{}, err
}

const max = math.MaxInt64
if size > max {
return nil, errors.NewUnexpectedError(
return ConstantSizedStaticType{}, errors.NewUnexpectedError(
"invalid constant-sized static type size: got %d, expected max %d",
size,
max,
Expand All @@ -1619,7 +1659,7 @@ func (d TypeDecoder) decodeConstantSizedStaticType() (StaticType, error) {
// Decode type at array index encodedConstantSizedStaticTypeTypeFieldKey
staticType, err := d.DecodeStaticType()
if err != nil {
return nil, errors.NewUnexpectedError(
return ConstantSizedStaticType{}, errors.NewUnexpectedError(
"invalid constant-sized static type inner type encoding: %w",
err,
)
Expand Down Expand Up @@ -1685,24 +1725,24 @@ func (d TypeDecoder) decodeReferenceStaticType() (StaticType, error) {
), nil
}

func (d TypeDecoder) decodeDictionaryStaticType() (StaticType, error) {
func (d TypeDecoder) decodeDictionaryStaticType() (DictionaryStaticType, error) {
const expectedLength = encodedDictionaryStaticTypeLength

arraySize, err := d.decoder.DecodeArrayHead()

if err != nil {
if e, ok := err.(*cbor.WrongTypeError); ok {
return nil, errors.NewUnexpectedError(
return DictionaryStaticType{}, errors.NewUnexpectedError(
"invalid dictionary static type encoding: expected [%d]any, got %s",
expectedLength,
e.ActualType.String(),
)
}
return nil, err
return DictionaryStaticType{}, err
}

if arraySize != expectedLength {
return nil, errors.NewUnexpectedError(
return DictionaryStaticType{}, errors.NewUnexpectedError(
"invalid dictionary static type encoding: expected [%d]any, got [%d]any",
expectedLength,
arraySize,
Expand All @@ -1712,7 +1752,7 @@ func (d TypeDecoder) decodeDictionaryStaticType() (StaticType, error) {
// Decode key type at array index encodedDictionaryStaticTypeKeyTypeFieldKey
keyType, err := d.DecodeStaticType()
if err != nil {
return nil, errors.NewUnexpectedError(
return DictionaryStaticType{}, errors.NewUnexpectedError(
"invalid dictionary static type key type encoding: %w",
err,
)
Expand All @@ -1721,7 +1761,7 @@ func (d TypeDecoder) decodeDictionaryStaticType() (StaticType, error) {
// Decode value type at array index encodedDictionaryStaticTypeValueTypeFieldKey
valueType, err := d.DecodeStaticType()
if err != nil {
return nil, errors.NewUnexpectedError(
return DictionaryStaticType{}, errors.NewUnexpectedError(
"invalid dictionary static type value type encoding: %w",
err,
)
Expand Down
47 changes: 47 additions & 0 deletions runtime/interpreter/encode.go
Original file line number Diff line number Diff line change
Expand Up @@ -681,6 +681,8 @@ func (v UFix64Value) Encode(e *atree.Encoder) error {
return e.CBOR.EncodeUint64(uint64(v))
}

var _ atree.ContainerStorable = &SomeStorable{}

// Encode encodes SomeStorable as
//
// cbor.Tag{
Expand All @@ -699,6 +701,25 @@ func (s SomeStorable) Encode(e *atree.Encoder) error {
return s.Storable.Encode(e)
}

// EncodeAsElement encodes SomeStorable as
//
// cbor.Tag{
// Number: CBORTagSomeValue,
// Content: Value(v.Value),
// }
func (s SomeStorable) EncodeAsElement(e *atree.Encoder, inlinedExtraData atree.InlinedExtraData) error {
turbolent marked this conversation as resolved.
Show resolved Hide resolved
// NOTE: when updating, also update SomeStorable.ByteSize
err := e.CBOR.EncodeRawBytes([]byte{
// tag number
0xd8, CBORTagSomeValue,
})
if err != nil {
return err
}

return atree.EncodeStorableAsElement(e, s.Storable, inlinedExtraData)
}

// Encode encodes AddressValue as
//
// cbor.Tag{
Expand Down Expand Up @@ -1601,6 +1622,20 @@ var _ atree.TypeInfo = compositeTypeInfo{}

const encodedCompositeTypeInfoLength = 3

func (c compositeTypeInfo) IsComposite() bool {
return true
}

func (c compositeTypeInfo) Identifier() string {
// TODO: maybe improve this to be more efficient, etc.
return "composite(" + c.location.ID() + "/" + c.qualifiedIdentifier + ")"
turbolent marked this conversation as resolved.
Show resolved Hide resolved
}

func (c compositeTypeInfo) Copy() atree.TypeInfo {
// Return c as is because c is a value type.
return c
}

func (c compositeTypeInfo) Encode(e *cbor.StreamEncoder) error {
err := e.EncodeRawBytes([]byte{
// tag number
Expand Down Expand Up @@ -1645,4 +1680,16 @@ func (e EmptyTypeInfo) Encode(encoder *cbor.StreamEncoder) error {
return encoder.EncodeNil()
}

func (e EmptyTypeInfo) IsComposite() bool {
return false
}

func (e EmptyTypeInfo) Identifier() string {
return ""
}

func (e EmptyTypeInfo) Copy() atree.TypeInfo {
return e
}

var emptyTypeInfo atree.TypeInfo = EmptyTypeInfo{}
Loading
Loading