1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-20 12:21:01 +00:00

feat(core): Recognize Taproot inputs.

This commit is contained in:
Andrew Kozlik 2021-10-27 16:50:59 +02:00 committed by Andrew Kozlik
parent 57fddcfd9d
commit 68ad1b07d2
3 changed files with 20 additions and 10 deletions

View File

@ -20,6 +20,10 @@ SIGHASH_ALL = const(0x01)
# The number of bip32 levels used in a wallet (chain and address) # The number of bip32 levels used in a wallet (chain and address)
BIP32_WALLET_DEPTH = const(2) BIP32_WALLET_DEPTH = const(2)
# Bitcoin opcodes
OP_0 = const(0x00)
OP_1 = const(0x51)
# supported witness versions for bech32 addresses # supported witness versions for bech32 addresses
_BECH32_WITVERS = (0, 1) _BECH32_WITVERS = (0, 1)
@ -48,6 +52,7 @@ CHANGE_OUTPUT_SCRIPT_TYPES = tuple(CHANGE_OUTPUT_TO_INPUT_SCRIPT_TYPES.keys())
SEGWIT_INPUT_SCRIPT_TYPES = ( SEGWIT_INPUT_SCRIPT_TYPES = (
InputScriptType.SPENDP2SHWITNESS, InputScriptType.SPENDP2SHWITNESS,
InputScriptType.SPENDWITNESS, InputScriptType.SPENDWITNESS,
InputScriptType.SPENDTAPROOT,
) )
SEGWIT_OUTPUT_SCRIPT_TYPES = ( SEGWIT_OUTPUT_SCRIPT_TYPES = (
@ -101,10 +106,16 @@ def input_is_segwit(txi: TxInput) -> bool:
) )
def input_is_nonsegwit(txi: TxInput) -> bool: def input_is_taproot(txi: TxInput) -> bool:
return txi.script_type in NONSEGWIT_INPUT_SCRIPT_TYPES or ( if txi.script_type == InputScriptType.SPENDTAPROOT:
txi.script_type == InputScriptType.EXTERNAL and txi.witness is None return True
)
if txi.script_type == InputScriptType.EXTERNAL:
assert txi.script_pubkey is not None
if txi.script_pubkey[0] == OP_1:
return True
return False
def input_is_external(txi: TxInput) -> bool: def input_is_external(txi: TxInput) -> bool:

View File

@ -223,7 +223,7 @@ class Bitcoin:
if i in self.segwit: if i in self.segwit:
if i in self.external: if i in self.external:
txi = await helpers.request_tx_input(self.tx_req, i, self.coin) txi = await helpers.request_tx_input(self.tx_req, i, self.coin)
self.serialized_tx.extend(txi.witness or b"") self.serialized_tx.extend(txi.witness or b"\0")
else: else:
await self.sign_segwit_input(i) await self.sign_segwit_input(i)
else: else:
@ -455,7 +455,7 @@ class Bitcoin:
# STAGE_REQUEST_SEGWIT_INPUT in legacy # STAGE_REQUEST_SEGWIT_INPUT in legacy
txi = await helpers.request_tx_input(self.tx_req, i, self.coin) txi = await helpers.request_tx_input(self.tx_req, i, self.coin)
if not input_is_segwit(txi): if txi.script_type not in common.SEGWIT_INPUT_SCRIPT_TYPES:
raise wire.ProcessError("Transaction has changed during signing") raise wire.ProcessError("Transaction has changed during signing")
self.tx_info.check_input(txi) self.tx_info.check_input(txi)
@ -491,8 +491,7 @@ class Bitcoin:
async def sign_segwit_input(self, i: int) -> None: async def sign_segwit_input(self, i: int) -> None:
# STAGE_REQUEST_SEGWIT_WITNESS in legacy # STAGE_REQUEST_SEGWIT_WITNESS in legacy
txi = await helpers.request_tx_input(self.tx_req, i, self.coin) txi = await helpers.request_tx_input(self.tx_req, i, self.coin)
if txi.script_type not in common.SEGWIT_INPUT_SCRIPT_TYPES:
if not input_is_segwit(txi):
raise wire.ProcessError("Transaction has changed during signing") raise wire.ProcessError("Transaction has changed during signing")
public_key, signature = self.sign_bip143_input(txi) public_key, signature = self.sign_bip143_input(txi)

View File

@ -6,7 +6,7 @@ from trezor.messages import PrevTx, SignTx, TxInput
from apps.common.writers import write_bitcoin_varint from apps.common.writers import write_bitcoin_varint
from .. import multisig, writers from .. import multisig, writers
from ..common import input_is_nonsegwit from ..common import NONSEGWIT_INPUT_SCRIPT_TYPES
from . import helpers from . import helpers
from .bitcoin import Bitcoin from .bitcoin import Bitcoin
@ -21,7 +21,7 @@ class Bitcoinlike(Bitcoin):
async def sign_nonsegwit_bip143_input(self, i_sign: int) -> None: async def sign_nonsegwit_bip143_input(self, i_sign: int) -> None:
txi = await helpers.request_tx_input(self.tx_req, i_sign, self.coin) txi = await helpers.request_tx_input(self.tx_req, i_sign, self.coin)
if not input_is_nonsegwit(txi): if txi.script_type not in NONSEGWIT_INPUT_SCRIPT_TYPES:
raise wire.ProcessError("Transaction has changed during signing") raise wire.ProcessError("Transaction has changed during signing")
public_key, signature = self.sign_bip143_input(txi) public_key, signature = self.sign_bip143_input(txi)