You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
trezor-firmware/core/src/apps/monero/xmr/key_image.py

143 lines
4.5 KiB

from typing import TYPE_CHECKING
from apps.monero.xmr import crypto_helpers, monero
if TYPE_CHECKING:
from trezor.messages import MoneroTransferDetails
from apps.monero.xmr import crypto
from apps.monero.xmr.credentials import AccountCreds
Subaddresses = dict[bytes, tuple[int, int]]
Sig = list[list[crypto.Scalar]]
def compute_hash(rr: MoneroTransferDetails) -> bytes:
from apps.monero.xmr.serialize.int_serialize import dump_uvarint_b
kck = crypto_helpers.get_keccak()
update = kck.update # local_cache_attribute
update(rr.out_key)
update(rr.tx_pub_key)
if rr.additional_tx_pub_keys:
for x in rr.additional_tx_pub_keys:
update(x)
update(dump_uvarint_b(rr.internal_output_index))
return kck.digest()
def export_key_image(
creds: AccountCreds, subaddresses: Subaddresses, td: MoneroTransferDetails
) -> tuple[crypto.Point, Sig]:
decodepoint = crypto_helpers.decodepoint # local_cache_attribute
out_key = decodepoint(td.out_key)
tx_pub_key = decodepoint(td.tx_pub_key)
additional_tx_pub_keys = td.additional_tx_pub_keys # local_cache_attribute
additional_tx_pub_key = None
if len(additional_tx_pub_keys) == 1: # compression
additional_tx_pub_key = decodepoint(additional_tx_pub_keys[0])
elif additional_tx_pub_keys:
if td.internal_output_index >= len(additional_tx_pub_keys):
raise ValueError("Wrong number of additional derivations")
additional_tx_pub_key = decodepoint(
additional_tx_pub_keys[td.internal_output_index]
)
# _export_key_image
# Generates key image for the TXO + signature for the key image
r = monero.generate_tx_spend_and_key_image_and_derivation(
creds,
subaddresses,
out_key,
tx_pub_key,
additional_tx_pub_key,
td.internal_output_index,
td.sub_addr_major,
td.sub_addr_minor,
)
xi, ki, _ = r[:3]
phash = crypto_helpers.encodepoint(ki)
sig = generate_ring_signature(phash, ki, [out_key], xi, 0, True)
return ki, sig
def generate_ring_signature(
prefix_hash: bytes,
image: crypto.Point,
pubs: list[crypto.Point],
sec: crypto.Scalar,
sec_idx: int,
test: bool = False,
) -> Sig:
"""
Generates ring signature with key image.
void crypto_ops::generate_ring_signature()
"""
from trezor.utils import memcpy
from apps.monero.xmr import crypto
Scalar = crypto.Scalar # local_cache_attribute
encodepoint_into = crypto.encodepoint_into # local_cache_attribute
if test:
t = crypto.scalarmult_base_into(None, sec)
if not crypto.point_eq(t, pubs[sec_idx]):
raise ValueError("Invalid sec key")
k_i = monero.generate_key_image(crypto_helpers.encodepoint(pubs[sec_idx]), sec)
if not crypto.point_eq(k_i, image):
raise ValueError("Key image invalid")
for k in pubs:
crypto.ge25519_check(k)
buff_off = len(prefix_hash)
buff = bytearray(buff_off + 2 * 32 * len(pubs))
memcpy(buff, 0, prefix_hash, 0, buff_off)
mvbuff = memoryview(buff)
sum = Scalar(0)
k = Scalar(0)
sig = []
for _ in range(len(pubs)):
sig.append([Scalar(0), Scalar(0)]) # c, r
for i in range(len(pubs)):
if i == sec_idx:
k = crypto.random_scalar()
tmp3 = crypto.scalarmult_base_into(None, k)
encodepoint_into(mvbuff[buff_off : buff_off + 32], tmp3)
buff_off += 32
tmp3 = crypto.hash_to_point_into(None, crypto_helpers.encodepoint(pubs[i]))
tmp2 = crypto.scalarmult_into(None, tmp3, k)
encodepoint_into(mvbuff[buff_off : buff_off + 32], tmp2)
buff_off += 32
else:
sig[i] = [crypto.random_scalar(), crypto.random_scalar()]
tmp3 = pubs[i]
tmp2 = crypto.ge25519_double_scalarmult_vartime_into(
None, tmp3, sig[i][0], sig[i][1]
)
encodepoint_into(mvbuff[buff_off : buff_off + 32], tmp2)
buff_off += 32
tmp3 = crypto.hash_to_point_into(None, crypto_helpers.encodepoint(tmp3))
tmp2 = crypto.add_keys3_into(None, sig[i][1], tmp3, sig[i][0], image)
encodepoint_into(mvbuff[buff_off : buff_off + 32], tmp2)
buff_off += 32
crypto.sc_add_into(sum, sum, sig[i][0])
h = crypto.hash_to_scalar_into(None, buff)
sig[sec_idx][0] = crypto.sc_sub_into(None, h, sum)
sig[sec_idx][1] = crypto.sc_mulsub_into(None, sig[sec_idx][0], sec, k)
return sig