txscript: Match Bitcoin Core OP_IFDUP behavior.
Unlike OP_IF and OP_NOTIF which interpret the top stack item as a number, OP_IFDUP interprets it as a boolean. This has important consequences because numbers are imited to int32s while booleans can be an arbitrary number of bytes. The offending script was found and reported by Jonas Nick through the use of fuzzing.
This commit is contained in:
parent
6801c0000a
commit
f284b9b394
2 changed files with 4 additions and 3 deletions
|
@ -80,6 +80,7 @@
|
|||
|
||||
["0 IFDUP", "DEPTH 1 EQUALVERIFY 0 EQUAL", "P2SH,STRICTENC"],
|
||||
["1 IFDUP", "DEPTH 2 EQUALVERIFY 1 EQUALVERIFY 1 EQUAL", "P2SH,STRICTENC"],
|
||||
["0x05 0x0100000000 IFDUP", "DEPTH 2 EQUALVERIFY 0x05 0x0100000000 EQUAL", "P2SH,STRICTENC", "IFDUP dups non ints"],
|
||||
["0 DROP", "DEPTH 0 EQUAL", "P2SH,STRICTENC"],
|
||||
["0", "DUP 1 ADD 1 EQUALVERIFY 0 EQUAL", "P2SH,STRICTENC"],
|
||||
["0 1", "NIP", "P2SH,STRICTENC"],
|
||||
|
|
|
@ -1084,14 +1084,14 @@ func opcode2Swap(op *parsedOpcode, vm *Engine) error {
|
|||
// Stack transformation (x1==0): [... x1] -> [...]
|
||||
// Stack transformation (x1!=0): [... x1] -> [... x1]
|
||||
func opcodeIfDup(op *parsedOpcode, vm *Engine) error {
|
||||
val, err := vm.dstack.PeekInt(0)
|
||||
so, err := vm.dstack.PeekByteArray(0)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Push copy of data iff it isn't zero
|
||||
if val != 0 {
|
||||
vm.dstack.PushInt(val)
|
||||
if asBool(so) {
|
||||
vm.dstack.PushByteArray(so)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
|
Loading…
Reference in a new issue