Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
N
nebula
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
exchain
nebula
Commits
b9ed1518
Commit
b9ed1518
authored
Dec 06, 2023
by
pcw109550
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
op-node: Span batch protectedBits implementation
parent
a35425c5
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
112 additions
and
8 deletions
+112
-8
span_batch.go
op-node/rollup/derive/span_batch.go
+1
-1
span_batch_txs.go
op-node/rollup/derive/span_batch_txs.go
+111
-7
No files found.
op-node/rollup/derive/span_batch.go
View file @
b9ed1518
...
...
@@ -26,7 +26,7 @@ import (
// spanBatch := SpanBatchType ++ prefix ++ payload
// prefix := rel_timestamp ++ l1_origin_num ++ parent_check ++ l1_origin_check
// payload := block_count ++ origin_bits ++ block_tx_counts ++ txs
// txs := contract_creation_bits ++ y_parity_bits ++ tx_sigs ++ tx_tos ++ tx_datas ++ tx_nonces ++ tx_gases
// txs := contract_creation_bits ++ y_parity_bits ++ tx_sigs ++ tx_tos ++ tx_datas ++ tx_nonces ++ tx_gases
++ protected_bits
var
ErrTooBigSpanBatchSize
=
errors
.
New
(
"span batch size limit reached"
)
...
...
op-node/rollup/derive/span_batch_txs.go
View file @
b9ed1518
...
...
@@ -18,7 +18,7 @@ type spanBatchTxs struct {
// this field must be manually set
totalBlockTxCount
uint64
//
7
fields
//
8
fields
contractCreationBits
*
big
.
Int
yParityBits
*
big
.
Int
txSigs
[]
spanBatchSignature
...
...
@@ -26,8 +26,11 @@ type spanBatchTxs struct {
txGases
[]
uint64
txTos
[]
common
.
Address
txDatas
[]
hexutil
.
Bytes
protectedBits
*
big
.
Int
txTypes
[]
int
// intermediate variables which can be recovered
txTypes
[]
int
totalLegacyTxCount
uint64
}
type
spanBatchSignature
struct
{
...
...
@@ -91,6 +94,62 @@ func (btx *spanBatchTxs) decodeContractCreationBits(r *bytes.Reader) error {
return
nil
}
// protectedBits is bitlist right-padded to a multiple of 8 bits
func
(
btx
*
spanBatchTxs
)
encodeProtectedBits
(
w
io
.
Writer
)
error
{
protectedBitBufferLen
:=
btx
.
totalLegacyTxCount
/
8
if
btx
.
totalLegacyTxCount
%
8
!=
0
{
protectedBitBufferLen
++
}
protectedBitBuffer
:=
make
([]
byte
,
protectedBitBufferLen
)
for
i
:=
0
;
i
<
int
(
btx
.
totalLegacyTxCount
);
i
+=
8
{
end
:=
i
+
8
if
end
>
int
(
btx
.
totalLegacyTxCount
)
{
end
=
int
(
btx
.
totalLegacyTxCount
)
}
var
bits
uint
=
0
for
j
:=
i
;
j
<
end
;
j
++
{
bits
|=
btx
.
protectedBits
.
Bit
(
j
)
<<
(
j
-
i
)
}
protectedBitBuffer
[
i
/
8
]
=
byte
(
bits
)
}
if
_
,
err
:=
w
.
Write
(
protectedBitBuffer
);
err
!=
nil
{
return
fmt
.
Errorf
(
"cannot write protected bits: %w"
,
err
)
}
return
nil
}
// protectedBits is bitlist right-padded to a multiple of 8 bits
func
(
btx
*
spanBatchTxs
)
decodeProtectedBits
(
r
*
bytes
.
Reader
)
error
{
protectedBitBufferLen
:=
btx
.
totalLegacyTxCount
/
8
if
btx
.
totalLegacyTxCount
%
8
!=
0
{
protectedBitBufferLen
++
}
// avoid out of memory before allocation
if
protectedBitBufferLen
>
MaxSpanBatchSize
{
return
ErrTooBigSpanBatchSize
}
protectedBitBuffer
:=
make
([]
byte
,
protectedBitBufferLen
)
_
,
err
:=
io
.
ReadFull
(
r
,
protectedBitBuffer
)
if
err
!=
nil
{
return
fmt
.
Errorf
(
"failed to read protected bits: %w"
,
err
)
}
protectedBits
:=
new
(
big
.
Int
)
for
i
:=
0
;
i
<
int
(
btx
.
totalLegacyTxCount
);
i
+=
8
{
end
:=
i
+
8
if
end
>
int
(
btx
.
totalLegacyTxCount
)
{
end
=
int
(
btx
.
totalLegacyTxCount
)
}
bits
:=
protectedBitBuffer
[
i
/
8
]
for
j
:=
i
;
j
<
end
;
j
++
{
bit
:=
uint
((
bits
>>
(
j
-
i
))
&
1
)
protectedBits
.
SetBit
(
protectedBits
,
j
,
bit
)
}
}
btx
.
protectedBits
=
protectedBits
return
nil
}
func
(
btx
*
spanBatchTxs
)
contractCreationCount
()
(
uint64
,
error
)
{
if
btx
.
contractCreationBits
==
nil
{
return
0
,
errors
.
New
(
"dev error: contract creation bits not set"
)
...
...
@@ -290,6 +349,9 @@ func (btx *spanBatchTxs) decodeTxDatas(r *bytes.Reader) error {
}
txDatas
=
append
(
txDatas
,
txData
)
txTypes
=
append
(
txTypes
,
txType
)
if
txType
==
types
.
LegacyTxType
{
btx
.
totalLegacyTxCount
++
}
}
btx
.
txDatas
=
txDatas
btx
.
txTypes
=
txTypes
...
...
@@ -300,13 +362,23 @@ func (btx *spanBatchTxs) recoverV(chainID *big.Int) error {
if
len
(
btx
.
txTypes
)
!=
len
(
btx
.
txSigs
)
{
return
errors
.
New
(
"tx type length and tx sigs length mismatch"
)
}
if
btx
.
protectedBits
==
nil
{
return
errors
.
New
(
"dev error: protected bits not set"
)
}
protectedBitsIdx
:=
0
for
idx
,
txType
:=
range
btx
.
txTypes
{
bit
:=
uint64
(
btx
.
yParityBits
.
Bit
(
idx
))
var
v
uint64
switch
txType
{
case
types
.
LegacyTxType
:
// EIP155
v
=
chainID
.
Uint64
()
*
2
+
35
+
bit
protectedBit
:=
btx
.
protectedBits
.
Bit
(
protectedBitsIdx
)
protectedBitsIdx
++
if
protectedBit
==
0
{
v
=
27
+
bit
}
else
{
// EIP-155
v
=
chainID
.
Uint64
()
*
2
+
35
+
bit
}
case
types
.
AccessListTxType
:
v
=
bit
case
types
.
DynamicFeeTxType
:
...
...
@@ -341,6 +413,9 @@ func (btx *spanBatchTxs) encode(w io.Writer) error {
if
err
:=
btx
.
encodeTxGases
(
w
);
err
!=
nil
{
return
err
}
if
err
:=
btx
.
encodeProtectedBits
(
w
);
err
!=
nil
{
return
err
}
return
nil
}
...
...
@@ -366,6 +441,9 @@ func (btx *spanBatchTxs) decode(r *bytes.Reader) error {
if
err
:=
btx
.
decodeTxGases
(
r
);
err
!=
nil
{
return
err
}
if
err
:=
btx
.
decodeProtectedBits
(
r
);
err
!=
nil
{
return
err
}
return
nil
}
...
...
@@ -408,9 +486,14 @@ func convertVToYParity(v uint64, txType int) (uint, error) {
var
yParityBit
uint
switch
txType
{
case
types
.
LegacyTxType
:
// EIP155: v = 2 * chainID + 35 + yParity
// v - 35 = yParity (mod 2)
yParityBit
=
uint
((
v
-
35
)
&
1
)
if
isProtectedV
(
v
,
txType
)
{
// EIP-155: v = 2 * chainID + 35 + yParity
// v - 35 = yParity (mod 2)
yParityBit
=
uint
((
v
-
35
)
&
1
)
}
else
{
// unprotected legacy txs must have v = 27 or 28
yParityBit
=
uint
(
v
-
27
)
}
case
types
.
AccessListTxType
:
yParityBit
=
uint
(
v
)
case
types
.
DynamicFeeTxType
:
...
...
@@ -421,6 +504,15 @@ func convertVToYParity(v uint64, txType int) (uint, error) {
return
yParityBit
,
nil
}
func
isProtectedV
(
v
uint64
,
txType
int
)
bool
{
if
txType
==
types
.
LegacyTxType
{
// if EIP-155 applied, v = 2 * chainID + 35 + yParity
return
v
!=
27
&&
v
!=
28
}
// every non legacy tx are protected
return
true
}
func
newSpanBatchTxs
(
txs
[][]
byte
,
chainID
*
big
.
Int
)
(
*
spanBatchTxs
,
error
)
{
totalBlockTxCount
:=
uint64
(
len
(
txs
))
var
txSigs
[]
spanBatchSignature
...
...
@@ -431,11 +523,21 @@ func newSpanBatchTxs(txs [][]byte, chainID *big.Int) (*spanBatchTxs, error) {
var
txTypes
[]
int
contractCreationBits
:=
new
(
big
.
Int
)
yParityBits
:=
new
(
big
.
Int
)
protectedBits
:=
new
(
big
.
Int
)
totalLegacyTxCount
:=
uint64
(
0
)
for
idx
:=
0
;
idx
<
int
(
totalBlockTxCount
);
idx
++
{
var
tx
types
.
Transaction
if
err
:=
tx
.
UnmarshalBinary
(
txs
[
idx
]);
err
!=
nil
{
return
nil
,
errors
.
New
(
"failed to decode tx"
)
}
if
tx
.
Type
()
==
types
.
LegacyTxType
{
protectedBit
:=
uint
(
0
)
if
tx
.
Protected
()
{
protectedBit
=
uint
(
1
)
}
protectedBits
.
SetBit
(
protectedBits
,
int
(
totalLegacyTxCount
),
protectedBit
)
totalLegacyTxCount
++
}
if
tx
.
Protected
()
&&
tx
.
ChainId
()
.
Cmp
(
chainID
)
!=
0
{
return
nil
,
fmt
.
Errorf
(
"protected tx has chain ID %d, but expected chain ID %d"
,
tx
.
ChainId
(),
chainID
)
}
...
...
@@ -481,5 +583,7 @@ func newSpanBatchTxs(txs [][]byte, chainID *big.Int) (*spanBatchTxs, error) {
txTos
:
txTos
,
txDatas
:
txDatas
,
txTypes
:
txTypes
,
protectedBits
:
protectedBits
,
totalLegacyTxCount
:
totalLegacyTxCount
,
},
nil
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment