forked from ava-labs/coreth
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpredicate_bytes.go
64 lines (53 loc) · 2.66 KB
/
predicate_bytes.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
// (c) 2023, Ava Labs, Inc. All rights reserved.
// See the file LICENSE for licensing terms.
package predicate
import (
"fmt"
"github.com/ava-labs/coreth/params"
"github.com/ethereum/go-ethereum/common"
)
// EndByte is used as a delimiter for the bytes packed into a precompile predicate.
// Precompile predicates are encoded in the Access List of transactions in the access tuples
// which means that its length must be a multiple of 32 (common.HashLength).
// For messages with a length that does not comply to that, this delimiter is used to
// append/remove padding.
var EndByte = byte(0xff)
var (
ErrInvalidAllZeroBytes = fmt.Errorf("predicate specified invalid all zero bytes")
ErrInvalidPadding = fmt.Errorf("predicate specified invalid padding")
ErrInvalidEndDelimiter = fmt.Errorf("invalid end delimiter")
ErrorInvalidExtraData = fmt.Errorf("header extra data too short for predicate verification")
)
// PackPredicate packs [predicate] by delimiting the actual message with [PredicateEndByte]
// and zero padding to reach a length that is a multiple of 32.
func PackPredicate(predicateBytes []byte) []byte {
predicateBytes = append(predicateBytes, EndByte)
return common.RightPadBytes(predicateBytes, (len(predicateBytes)+31)/32*32)
}
// UnpackPredicate unpacks a predicate by stripping right padded zeroes, checking for the delimter,
// ensuring there is not excess padding, and returning the original message.
// Returns an error if it finds an incorrect encoding.
func UnpackPredicate(paddedPredicate []byte) ([]byte, error) {
trimmedPredicateBytes := common.TrimRightZeroes(paddedPredicate)
if len(trimmedPredicateBytes) == 0 {
return nil, fmt.Errorf("%w: 0x%x", ErrInvalidAllZeroBytes, paddedPredicate)
}
if expectedPaddedLength := (len(trimmedPredicateBytes) + 31) / 32 * 32; expectedPaddedLength != len(paddedPredicate) {
return nil, fmt.Errorf("%w: got length (%d), expected length (%d)", ErrInvalidPadding, len(paddedPredicate), expectedPaddedLength)
}
if trimmedPredicateBytes[len(trimmedPredicateBytes)-1] != EndByte {
return nil, ErrInvalidEndDelimiter
}
return trimmedPredicateBytes[:len(trimmedPredicateBytes)-1], nil
}
// GetPredicateResultBytes returns the predicate result bytes from the extra data and
// true iff the predicate results bytes have non-zero length.
func GetPredicateResultBytes(extraData []byte) ([]byte, bool) {
// Prior to Durango, the VM enforces the extra data is smaller than or equal to this size.
// After Durango, the VM pre-verifies the extra data past the dynamic fee rollup window is
// valid.
if len(extraData) <= params.DynamicFeeExtraDataSize {
return nil, false
}
return extraData[params.DynamicFeeExtraDataSize:], true
}