1# Copyright (c) 2023 Peter Johanson <peter@peterjohanson.com>
2#
3# SPDX-License-Identifier: Apache-2.0
4
5'''UF2 runner (flash only) for UF2 compatible bootloaders.'''
6
7from pathlib import Path
8from shutil import copy
9
10from runners.core import RunnerCaps, ZephyrBinaryRunner
11
12try:
13    import psutil
14    MISSING_PSUTIL = False
15except ImportError:
16    # This can happen when building the documentation for the
17    # runners package if psutil is not on sys.path. This is fine
18    # to ignore in that case.
19    MISSING_PSUTIL = True
20
21class UF2BinaryRunner(ZephyrBinaryRunner):
22    '''Runner front-end for copying to UF2 USB-MSC mounts.'''
23
24    def __init__(self, cfg, board_id=None):
25        super().__init__(cfg)
26        self.board_id = board_id
27
28    @classmethod
29    def name(cls):
30        return 'uf2'
31
32    @classmethod
33    def capabilities(cls):
34        return RunnerCaps(commands={'flash'})
35
36    @classmethod
37    def do_add_parser(cls, parser):
38        parser.add_argument('--board-id', dest='board_id',
39                            help='Board-ID value to match from INFO_UF2.TXT')
40
41    @classmethod
42    def do_create(cls, cfg, args):
43        return UF2BinaryRunner(cfg, board_id=args.board_id)
44
45    @staticmethod
46    def get_uf2_info_path(part) -> Path:
47        return Path(part.mountpoint) / "INFO_UF2.TXT"
48
49    @staticmethod
50    def is_uf2_partition(part):
51        try:
52            return ((part.fstype in ['vfat', 'FAT', 'msdos']) and
53                    UF2BinaryRunner.get_uf2_info_path(part).is_file())
54        except PermissionError:
55            return False
56
57    @staticmethod
58    def get_uf2_info(part):
59        lines = UF2BinaryRunner.get_uf2_info_path(part).read_text().splitlines()
60
61        lines = lines[1:] # Skip the first summary line
62
63        def split_uf2_info(line: str):
64            k, _, val = line.partition(':')
65            return k.strip(), val.strip()
66
67        return {k: v for k, v in (split_uf2_info(line) for line in lines) if k and v}
68
69    def match_board_id(self, part):
70        info = self.get_uf2_info(part)
71
72        return info.get('Board-ID') == self.board_id
73
74    def get_uf2_partitions(self):
75        parts = [part for part in psutil.disk_partitions() if self.is_uf2_partition(part)]
76
77        if (self.board_id is not None) and parts:
78            parts = [part for part in parts if self.match_board_id(part)]
79            if not parts:
80                self.logger.warning("Discovered UF2 partitions don't match Board-ID '%s'",
81                                    self.board_id)
82
83        return parts
84
85    def copy_uf2_to_partition(self, part):
86        self.ensure_output('uf2')
87
88        copy(self.cfg.uf2_file, part.mountpoint)
89
90    def do_run(self, command, **kwargs):
91        if MISSING_PSUTIL:
92            raise RuntimeError(
93                'could not import psutil; something may be wrong with the '
94                'python environment')
95
96        partitions = self.get_uf2_partitions()
97        if not partitions:
98            raise RuntimeError('No matching UF2 partitions found')
99
100        if len(partitions) > 1:
101            raise RuntimeError('More than one matching UF2 partitions found')
102
103        part = partitions[0]
104        self.logger.info("Copying UF2 file to '%s'", part.mountpoint)
105        self.copy_uf2_to_partition(part)
106