2020-06-03 15:52:02 +02:00
|
|
|
# Required for deserialisation of ValidatorSig in Attestation due to
|
|
|
|
# https://github.com/nim-lang/Nim/issues/11225
|
|
|
|
|
2019-11-27 21:43:02 +01:00
|
|
|
import
|
2020-06-25 12:23:10 +02:00
|
|
|
stew/ptrops, stew/ranges/ptr_arith, chronicles,
|
2020-06-23 13:54:24 +00:00
|
|
|
../beacon_chain/extras,
|
2020-06-03 15:52:02 +02:00
|
|
|
../beacon_chain/spec/[crypto, datatypes, digest, validator, beaconstate,
|
2020-06-23 13:54:24 +00:00
|
|
|
state_transition_block, state_transition],
|
2020-06-03 15:52:02 +02:00
|
|
|
../beacon_chain/ssz/[merkleization, ssz_serialization]
|
2019-11-27 21:43:02 +01:00
|
|
|
|
|
|
|
type
|
2019-12-17 14:01:30 +11:00
|
|
|
AttestationInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2019-12-17 14:01:30 +11:00
|
|
|
attestation: Attestation
|
|
|
|
AttesterSlashingInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2019-12-17 14:01:30 +11:00
|
|
|
attesterSlashing: AttesterSlashing
|
2019-11-27 21:43:02 +01:00
|
|
|
BlockInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2020-03-05 01:29:27 +01:00
|
|
|
beaconBlock: SignedBeaconBlock
|
2019-12-17 15:50:47 +11:00
|
|
|
BlockHeaderInput = BlockInput
|
2019-12-17 14:01:30 +11:00
|
|
|
DepositInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2019-12-17 14:01:30 +11:00
|
|
|
deposit: Deposit
|
|
|
|
ProposerSlashingInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2019-12-17 14:01:30 +11:00
|
|
|
proposerSlashing: ProposerSlashing
|
|
|
|
VoluntaryExitInput = object
|
2020-04-29 22:12:07 +02:00
|
|
|
state: BeaconState
|
2020-03-05 01:29:27 +01:00
|
|
|
exit: SignedVoluntaryExit
|
2019-12-16 16:46:58 +11:00
|
|
|
# This and AssertionError are raised to indicate programming bugs
|
2020-01-08 12:03:33 +11:00
|
|
|
# A wrapper to allow exception tracking to identify unexpected exceptions
|
2020-03-24 13:13:07 +02:00
|
|
|
FuzzCrashError = object of CatchableError
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2019-11-28 23:01:12 +01:00
|
|
|
# TODO: change ptr uint to ptr csize_t when available in newer Nim version.
|
2020-06-25 12:23:10 +02:00
|
|
|
proc copyState(state: BeaconState, xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint): bool {.raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
var resultState =
|
|
|
|
try:
|
|
|
|
SSZ.encode(state)
|
|
|
|
except IOError as e:
|
|
|
|
# Shouldn't occur as the writer isn't a file
|
|
|
|
raise newException(FuzzCrashError, "Unexpected failure to serialize.", e)
|
2019-12-17 14:01:30 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
if unlikely(resultState.len.uint > xoutput_size[]):
|
2020-01-08 12:03:33 +11:00
|
|
|
let msg = (
|
2020-06-25 12:23:10 +02:00
|
|
|
"Not enough xoutput buffer provided to nimbus harness. Provided: " &
|
|
|
|
$(xoutput_size[]) &
|
2020-01-08 12:03:33 +11:00
|
|
|
"Required: " &
|
|
|
|
$resultState.len.uint
|
|
|
|
)
|
|
|
|
raise newException(FuzzCrashError, msg)
|
2020-06-25 12:23:10 +02:00
|
|
|
xoutput_size[] = resultState.len.uint
|
|
|
|
# TODO: improvement might be to write directly to buffer with xoutputStream
|
2019-12-17 14:01:30 +11:00
|
|
|
# and SszWriter (but then need to ensure length doesn't overflow)
|
2020-06-25 12:23:10 +02:00
|
|
|
copyMem(xoutput, unsafeAddr resultState[0], xoutput_size[])
|
2019-12-17 14:01:30 +11:00
|
|
|
result = true
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2020-03-05 01:29:27 +01:00
|
|
|
template decodeAndProcess(typ, process: untyped): bool =
|
|
|
|
let flags {.inject.} = if disable_bls: {skipBlsValidation} else: {}
|
2019-12-17 15:50:47 +11:00
|
|
|
|
2020-03-05 01:29:27 +01:00
|
|
|
var
|
|
|
|
cache {.used, inject.} = get_empty_per_epoch_cache()
|
2020-04-29 22:12:07 +02:00
|
|
|
data {.inject.} = newClone(
|
2020-03-05 01:29:27 +01:00
|
|
|
try:
|
|
|
|
SSZ.decode(input, typ)
|
|
|
|
except MalformedSszError as e:
|
|
|
|
raise newException(
|
|
|
|
FuzzCrashError,
|
|
|
|
"Malformed SSZ, likely bug in preprocessing.", e)
|
|
|
|
except SszSizeMismatchError as e:
|
|
|
|
raise newException(
|
|
|
|
FuzzCrashError,
|
|
|
|
"SSZ size mismatch, likely bug in preprocessing.", e)
|
2020-04-29 22:12:07 +02:00
|
|
|
)
|
2020-03-05 01:29:27 +01:00
|
|
|
let processOk =
|
|
|
|
try:
|
|
|
|
process
|
|
|
|
except IOError as e:
|
|
|
|
raise newException(
|
|
|
|
FuzzCrashError, "Unexpected (logging?) IOError in state transition", e,
|
|
|
|
)
|
|
|
|
except ValueError as e:
|
|
|
|
raise newException(
|
|
|
|
FuzzCrashError,
|
|
|
|
"Unexpected (logging?) IOError in state transition", e)
|
|
|
|
except Exception as e:
|
|
|
|
# TODO why an Exception?
|
|
|
|
# Lots of vendor code looks like it might raise a bare exception type
|
|
|
|
raise newException(FuzzCrashError, "Unexpected Exception in state transition", e)
|
|
|
|
|
|
|
|
if processOk:
|
2020-06-25 12:23:10 +02:00
|
|
|
copyState(data.state, xoutput, xoutput_size)
|
2020-03-05 01:29:27 +01:00
|
|
|
else:
|
|
|
|
false
|
2019-12-17 15:50:47 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_attestation(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(AttestationInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_attestation(data.state, data.attestation, flags, cache).isOk
|
2019-12-17 15:50:47 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_attester_slashing(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(AttesterSlashingInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_attester_slashing(data.state, data.attesterSlashing, flags, cache).isOk
|
2019-12-17 15:50:47 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_block(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-04-30 16:27:17 +00:00
|
|
|
# There's not a perfect approach here, but it's not worth switching the rest
|
|
|
|
# and requiring HashedBeaconState (yet). So to keep consistent, puts wrapper
|
|
|
|
# only in one function.
|
|
|
|
proc state_transition(
|
|
|
|
data: auto, blck: auto, flags: auto, rollback: RollbackHashedProc):
|
|
|
|
auto =
|
|
|
|
var hashedState =
|
|
|
|
HashedBeaconState(data: data.state, root: hash_tree_root(data.state))
|
|
|
|
result = state_transition(hashedState, blck, flags, rollback)
|
|
|
|
data.state = hashedState.data
|
|
|
|
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(BlockInput):
|
2020-04-30 16:27:17 +00:00
|
|
|
state_transition(data, data.beaconBlock, flags, noRollback)
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_block_header(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(BlockHeaderInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_block_header(data.state, data.beaconBlock.message, flags, cache).isOk
|
2019-12-17 15:50:47 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_deposit(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(DepositInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_deposit(data.state, data.deposit, flags).isOk
|
2019-12-17 14:01:30 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_proposer_slashing(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(ProposerSlashingInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_proposer_slashing(data.state, data.proposerSlashing, flags, cache).isOk
|
2019-12-17 14:01:30 +11:00
|
|
|
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_voluntary_exit(input: openArray[byte], xoutput: ptr byte,
|
|
|
|
xoutput_size: ptr uint, disable_bls: bool): bool {.exportc, raises: [FuzzCrashError, Defect].} =
|
2020-03-05 01:29:27 +01:00
|
|
|
decodeAndProcess(VoluntaryExitInput):
|
2020-07-03 17:03:14 +00:00
|
|
|
process_voluntary_exit(data.state, data.exit, flags).isOk
|
2019-12-17 14:01:30 +11:00
|
|
|
|
2019-11-27 21:43:02 +01:00
|
|
|
# Note: Could also accept raw input pointer and access list_size + seed here.
|
2020-06-25 12:23:10 +02:00
|
|
|
# However, list_size needs to be known also outside this proc to allocate xoutput.
|
2019-11-27 21:43:02 +01:00
|
|
|
# TODO: rework to copy immediatly in an uint8 openArray, considering we have to
|
|
|
|
# go over the list anyhow?
|
2020-06-25 12:23:10 +02:00
|
|
|
proc nfuzz_shuffle(input_seed: ptr byte, xoutput: var openArray[uint64]): bool
|
2020-01-08 12:03:33 +11:00
|
|
|
{.exportc, raises: [Defect].} =
|
2019-11-27 21:43:02 +01:00
|
|
|
var seed: Eth2Digest
|
2019-11-28 23:01:12 +01:00
|
|
|
# Should be OK as max 2 bytes are passed by the framework.
|
2020-06-25 12:23:10 +02:00
|
|
|
let list_size = xoutput.len.uint64
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2019-11-28 23:01:12 +01:00
|
|
|
copyMem(addr(seed.data), input_seed, sizeof(seed.data))
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2019-11-28 23:01:12 +01:00
|
|
|
var shuffled_seq: seq[ValidatorIndex]
|
2019-12-17 14:01:30 +11:00
|
|
|
shuffled_seq = get_shuffled_seq(seed, list_size)
|
2019-11-27 21:43:02 +01:00
|
|
|
|
2020-01-08 12:03:33 +11:00
|
|
|
doAssert(
|
|
|
|
list_size == shuffled_seq.len.uint64,
|
|
|
|
"Shuffled list should be of requested size."
|
|
|
|
)
|
2019-11-27 21:43:02 +01:00
|
|
|
|
|
|
|
for i in 0..<list_size:
|
|
|
|
# ValidatorIndex is currently wrongly uint32 so we copy this 1 by 1,
|
2020-06-25 12:23:10 +02:00
|
|
|
# assumes passed xoutput is zeroed.
|
|
|
|
copyMem(offset(addr xoutput, i.int), shuffled_seq[i.int].unsafeAddr,
|
2019-11-27 21:43:02 +01:00
|
|
|
sizeof(ValidatorIndex))
|
2019-11-28 23:01:12 +01:00
|
|
|
|
|
|
|
result = true
|