aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python/usrp_mpm/sys_utils/udev.py
blob: ef79ed7479331a7298ae92ad61e4b1d54e44fb77 (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
#
# Copyright 2017 Ettus Research, a National Instruments Company
#
# SPDX-License-Identifier: GPL-3.0-or-later
#
"""
Utilities for udev lookups
"""

import os
import glob
import pyudev
from pathlib import Path

DT_BASE = "/proc/device-tree"

def get_eeprom_paths_by_symbol(symbol_name_glob):
    """
    Searches for EEPROM file locaction of symbol names under
    /proc/device-tree/__symbols__.
    args: symbol_name_glob: symbol name(s) to search (allows glob expressions)
    returns: dictionary with name found under __symbols__ as key
             and corresponding nvmem file path as value.
             The dictionary keys are sorted alphabetically.
    raises: FileNotFoundExcepiton: in case a symbol file could not be read.
    """
    symbol_base = os.path.join(DT_BASE, "__symbols__")
    context = pyudev.Context()
    devices = context.list_devices(subsystem="nvmem")
    of_nodes = [os.path.join(dev.sys_path, "of_node") for dev in devices]

    def read_symbol_file(symbol_file):
        with open(symbol_file) as f:
            symbol_path = f.read()
            # remove leading slash and trailing terminating char before
            # building the full path for symbol
            symbol_path = os.path.join(DT_BASE, symbol_path[1:-1])
            return symbol_path

    def find_device_path(path):
        for dev in of_nodes:
            if not os.path.exists(dev):
                # not all nvmem devices have an of_node
                continue
            elif os.path.samefile(dev, path):
                return os.path.join(os.path.dirname(dev), "nvmem")
        return None

    eeproms = glob.glob(os.path.join(symbol_base, symbol_name_glob))
    paths = {os.path.basename(eeprom): read_symbol_file(eeprom)
             for eeprom in eeproms}
    return {name: find_device_path(path)
            for name, path in sorted(paths.items())}


def get_device_from_dt_symbol(symbol, subsystem=None, context=None):
    """
    Return the device associated with the device tree symbol, which usually
    is a label on a specific node of interest
    """
    symfile = Path(DT_BASE) / '__symbols__' / symbol
    fullname = symfile.read_text()
    if context is None:
        context = pyudev.Context()
    devices = list(context.list_devices(OF_FULLNAME=fullname, subsystem=subsystem))
    if not devices:
        return None
    return devices[0]


def get_eeprom_paths(address):
    """
    Return list of EEPROM device paths for a given I2C address.
    If no device paths are found, an empty list is returned.
    """
    context = pyudev.Context()
    parent = pyudev.Device.from_name(context, "platform", address)
    paths = [d.device_node if d.device_node is not None else d.sys_path
             for d in context.list_devices(parent=parent, subsystem="nvmem")]
    if len(paths) == 0:
        return []
    # We need to sort this so 9-0050 comes before 10-0050 (etc.)
    maxlen = max((len(os.path.split(p)[1]) for p in paths))
    paths = sorted(
        paths,
        key=lambda x: "{:>0{maxlen}}".format(os.path.split(x)[1], maxlen=maxlen)
    )
    return [os.path.join(x, 'nvmem') for x in paths]

def get_spidev_nodes(spi_master):
    """
    Return list of spidev device paths for a given SPI master. If no valid paths
    can be found, an empty list is returned.
    """
    context = pyudev.Context()
    parent = pyudev.Device.from_name(context, "platform", spi_master)
    return [
        device.device_node
        for device in context.list_devices(parent=parent, subsystem="spidev")
    ]

def get_device_from_symbol(symbol, subsystems):
    """
    Find the first device and return its name where the parent device the DT
    symbol name matches and the hierarchy of subsystems (e.g. ['spi', 'spidev'])
    match
    """
    assert isinstance(subsystems,list)
    context = pyudev.Context()
    device = get_device_from_dt_symbol(symbol, subsystem=subsystems.pop(0), context=context)
    if device is None:
        return None
    for subsystem in subsystems:
        devices = list(context.list_devices(parent=device, subsystem=subsystem))
        if not devices:
            return None
        device = devices[0]
    return device.properties.get('DEVNAME')

def dt_symbol_get_spidev(symbol):
    """
    Return spidev associated with the given device tree symbol
    """
    return get_device_from_symbol(symbol, ['spi', 'spidev'])