mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-12-22 22:38:08 +00:00
feat(python): add screenshotting capability for T1 in Debuglink
This commit is contained in:
parent
2ce1e6ba7d
commit
58fb6c77a7
1
python/.changelog.d/2093.added
Normal file
1
python/.changelog.d/2093.added
Normal file
@ -0,0 +1 @@
|
|||||||
|
Support T1 screenshot saving in Debuglink
|
@ -20,6 +20,7 @@ from collections import namedtuple
|
|||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
from enum import IntEnum
|
from enum import IntEnum
|
||||||
from itertools import zip_longest
|
from itertools import zip_longest
|
||||||
|
from pathlib import Path
|
||||||
from typing import (
|
from typing import (
|
||||||
TYPE_CHECKING,
|
TYPE_CHECKING,
|
||||||
Any,
|
Any,
|
||||||
@ -37,6 +38,7 @@ from typing import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
from mnemonic import Mnemonic
|
from mnemonic import Mnemonic
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
from . import mapping, messages, protobuf
|
from . import mapping, messages, protobuf
|
||||||
from .client import TrezorClient
|
from .client import TrezorClient
|
||||||
@ -69,6 +71,14 @@ class DebugLink:
|
|||||||
self.allow_interactions = auto_interact
|
self.allow_interactions = auto_interact
|
||||||
self.mapping = mapping.DEFAULT_MAPPING
|
self.mapping = mapping.DEFAULT_MAPPING
|
||||||
|
|
||||||
|
# To be set by TrezorClientDebugLink (is not known during creation time)
|
||||||
|
self.model: Optional[str] = None
|
||||||
|
|
||||||
|
# For T1 screenshotting functionality in DebugUI
|
||||||
|
self.t1_take_screenshots = False
|
||||||
|
self.t1_screenshot_directory: Optional[Path] = None
|
||||||
|
self.t1_screenshot_counter = 0
|
||||||
|
|
||||||
def open(self) -> None:
|
def open(self) -> None:
|
||||||
self.transport.begin_session()
|
self.transport.begin_session()
|
||||||
|
|
||||||
@ -204,10 +214,20 @@ class DebugLink:
|
|||||||
return self._call(messages.DebugLinkReseedRandom(value=value))
|
return self._call(messages.DebugLinkReseedRandom(value=value))
|
||||||
|
|
||||||
def start_recording(self, directory: str) -> None:
|
def start_recording(self, directory: str) -> None:
|
||||||
|
# Different recording logic between TT and T1
|
||||||
|
if self.model == "T":
|
||||||
self._call(messages.DebugLinkRecordScreen(target_directory=directory))
|
self._call(messages.DebugLinkRecordScreen(target_directory=directory))
|
||||||
|
else:
|
||||||
|
self.t1_screenshot_directory = Path(directory)
|
||||||
|
self.t1_screenshot_counter = 0
|
||||||
|
self.t1_take_screenshots = True
|
||||||
|
|
||||||
def stop_recording(self) -> None:
|
def stop_recording(self) -> None:
|
||||||
|
# Different recording logic between TT and T1
|
||||||
|
if self.model == "T":
|
||||||
self._call(messages.DebugLinkRecordScreen(target_directory=None))
|
self._call(messages.DebugLinkRecordScreen(target_directory=None))
|
||||||
|
else:
|
||||||
|
self.t1_take_screenshots = False
|
||||||
|
|
||||||
@expect(messages.DebugLinkMemory, field="memory", ret_type=bytes)
|
@expect(messages.DebugLinkMemory, field="memory", ret_type=bytes)
|
||||||
def memory_read(self, address: int, length: int) -> protobuf.MessageType:
|
def memory_read(self, address: int, length: int) -> protobuf.MessageType:
|
||||||
@ -226,6 +246,36 @@ class DebugLink:
|
|||||||
def erase_sd_card(self, format: bool = True) -> messages.Success:
|
def erase_sd_card(self, format: bool = True) -> messages.Success:
|
||||||
return self._call(messages.DebugLinkEraseSdCard(format=format))
|
return self._call(messages.DebugLinkEraseSdCard(format=format))
|
||||||
|
|
||||||
|
def take_t1_screenshot_if_relevant(self) -> None:
|
||||||
|
"""Conditionally take screenshots on T1.
|
||||||
|
|
||||||
|
TT handles them differently, see debuglink.start_recording.
|
||||||
|
"""
|
||||||
|
if self.model == "1" and self.t1_take_screenshots:
|
||||||
|
self.save_screenshot_for_t1()
|
||||||
|
|
||||||
|
def save_screenshot_for_t1(self) -> None:
|
||||||
|
layout = self.state().layout
|
||||||
|
assert layout is not None
|
||||||
|
assert len(layout) == 128 * 64 // 8
|
||||||
|
|
||||||
|
pixels: List[int] = []
|
||||||
|
for byteline in range(64 // 8):
|
||||||
|
offset = byteline * 128
|
||||||
|
row = layout[offset : offset + 128]
|
||||||
|
for bit in range(8):
|
||||||
|
pixels.extend(bool(px & (1 << bit)) for px in row)
|
||||||
|
|
||||||
|
im = Image.new("1", (128, 64))
|
||||||
|
im.putdata(pixels[::-1])
|
||||||
|
|
||||||
|
assert self.t1_screenshot_directory is not None
|
||||||
|
img_location = (
|
||||||
|
self.t1_screenshot_directory / f"{self.t1_screenshot_counter:04d}.png"
|
||||||
|
)
|
||||||
|
im.save(img_location)
|
||||||
|
self.t1_screenshot_counter += 1
|
||||||
|
|
||||||
|
|
||||||
class NullDebugLink(DebugLink):
|
class NullDebugLink(DebugLink):
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
@ -265,6 +315,8 @@ class DebugUI:
|
|||||||
] = None
|
] = None
|
||||||
|
|
||||||
def button_request(self, br: messages.ButtonRequest) -> None:
|
def button_request(self, br: messages.ButtonRequest) -> None:
|
||||||
|
self.debuglink.take_t1_screenshot_if_relevant()
|
||||||
|
|
||||||
if self.input_flow is None:
|
if self.input_flow is None:
|
||||||
if br.code == messages.ButtonRequestType.PinEntry:
|
if br.code == messages.ButtonRequestType.PinEntry:
|
||||||
self.debuglink.input(self.get_pin())
|
self.debuglink.input(self.get_pin())
|
||||||
@ -283,6 +335,8 @@ class DebugUI:
|
|||||||
self.input_flow = self.INPUT_FLOW_DONE
|
self.input_flow = self.INPUT_FLOW_DONE
|
||||||
|
|
||||||
def get_pin(self, code: Optional["PinMatrixRequestType"] = None) -> str:
|
def get_pin(self, code: Optional["PinMatrixRequestType"] = None) -> str:
|
||||||
|
self.debuglink.take_t1_screenshot_if_relevant()
|
||||||
|
|
||||||
if self.pins is None:
|
if self.pins is None:
|
||||||
raise RuntimeError("PIN requested but no sequence was configured")
|
raise RuntimeError("PIN requested but no sequence was configured")
|
||||||
|
|
||||||
@ -292,6 +346,7 @@ class DebugUI:
|
|||||||
raise AssertionError("PIN sequence ended prematurely")
|
raise AssertionError("PIN sequence ended prematurely")
|
||||||
|
|
||||||
def get_passphrase(self, available_on_device: bool) -> str:
|
def get_passphrase(self, available_on_device: bool) -> str:
|
||||||
|
self.debuglink.take_t1_screenshot_if_relevant()
|
||||||
return self.passphrase
|
return self.passphrase
|
||||||
|
|
||||||
|
|
||||||
@ -415,6 +470,9 @@ class TrezorClientDebugLink(TrezorClient):
|
|||||||
|
|
||||||
super().__init__(transport, ui=self.ui)
|
super().__init__(transport, ui=self.ui)
|
||||||
|
|
||||||
|
# So that we can choose right screenshotting logic (T1 vs TT)
|
||||||
|
self.debug.model = self.features.model
|
||||||
|
|
||||||
def reset_debug_features(self) -> None:
|
def reset_debug_features(self) -> None:
|
||||||
"""Prepare the debugging client for a new testcase.
|
"""Prepare the debugging client for a new testcase.
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user