aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python/usrp_mpm/dtoverlay.py
blob: 7f1bf653fcc677d54bb74e8c1979835c4354e555 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#
# Copyright 2017 Ettus Research (National Instruments)
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
#
"""
Manipulation of device tree overlays (Linux kernel)
"""

import os
from .mpmlog import get_logger

SYSFS_OVERLAY_BASE_DIR = '/sys/kernel/config/device-tree/overlays'
OVERLAY_DEFAULT_PATH = '/lib/firmware'

def get_overlay_attrs(overlay_name):
    """
    List those attributes that are connected to an overlay entry in a sysfs
    node.
    """
    overlay_path = os.path.join(SYSFS_OVERLAY_BASE_DIR, overlay_name)
    attrs = {}
    for attr_name in os.listdir(overlay_path):
        try:
            attr_val = open(
                os.path.join(overlay_path, attr_name), 'r'
            ).read().strip()
        except OSError:
            pass
        if len(attr_val):
            attrs[attr_name] = attr_val
    return attrs

def is_applied(overlay_name):
    """
    Returns True if the overlay is already applied, False if not.
    """
    try:
        return open(
            os.path.join(SYSFS_OVERLAY_BASE_DIR, overlay_name, 'status')
        ).read().strip() == 'applied'
    except IOError:
        return False

def list_overlays(applied_only=False):
    """
    List all registered kernel modules. Returns a dict of dicts:
    {
        '<overlay_name>': {
            # attributes
        },
    }

    All the attributes come from sysfs. See get_overlay_attrs().

    applied_only -- Only return those overlays that are already applied.
    """
    return {
        overlay_name: get_overlay_attrs(overlay_name)
        for overlay_name in os.listdir(SYSFS_OVERLAY_BASE_DIR)
        if not applied_only \
            or get_overlay_attrs(overlay_name).get('status') == 'applied'
    }

def list_available_overlays(path):
    """
    List available overlay files (dtbo)
    """
    path = path or OVERLAY_DEFAULT_PATH
    return [x.strip()[:-5] for x in os.listdir(path) if x.endswith('.dtbo')]

def apply_overlay(overlay_name):
    """
    Applies the given overlay. Does not check if the overlay is loaded.
    """
    get_logger("DTO").trace("Applying overlay `{}'...".format(overlay_name))
    overlay_path = os.path.join(SYSFS_OVERLAY_BASE_DIR, overlay_name)
    if not os.path.exists(overlay_path):
        os.mkdir(overlay_path)
    open(
        os.path.join(SYSFS_OVERLAY_BASE_DIR, overlay_name, 'path'), 'w'
    ).write("{}.dtbo".format(overlay_name))

def apply_overlay_safe(overlay_name):
    """
    Only apply an overlay if it's not yet applied.

    Finally, checks that the overlay was applied and throws an exception if not.
    """
    if is_applied(overlay_name):
        get_logger("DTO").debug(
            "Overlay `{}' was already applied, not applying again.".format(
                overlay_name
            )
        )
    else:
        apply_overlay(overlay_name)
    if not is_applied(overlay_name):
        raise RuntimeError("Failed to apply overlay `{}'".format(overlay_name))

def rm_overlay(overlay_name):
    """
    Removes the given overlay. Does not check if the overlay is loaded.
    """
    get_logger("DTO").trace("Removing overlay `{}'...".format(overlay_name))
    os.rmdir(os.path.join(SYSFS_OVERLAY_BASE_DIR, overlay_name))

def rm_overlay_safe(overlay_name):
    """
    Only remove an overlay if it's already applied.
    """
    if overlay_name in list(list_overlays(applied_only=True).keys()):
        rm_overlay(overlay_name)
    else:
        get_logger("DTO").debug(
            "Overlay `{}' was not loaded, not removing.".format(overlay_name)
        )