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:
parent
57fddcfd9d
commit
68ad1b07d2
@ -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:
|
||||||
|
@ -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)
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user