From 8b91cb0cd4c14c337013dab214ad6c5c27ba892c Mon Sep 17 00:00:00 2001 From: Gerard Marull-Paretas Date: Wed, 16 Mar 2022 16:04:49 +0100 Subject: [PATCH] scripts: utils: pinctrl_nrf_migrate: support more peripherals Add support for more peripherals: - I2C - SPI (master mode assumed) - PDM - QDEC - QSPI - PWM - I2S Signed-off-by: Gerard Marull-Paretas --- scripts/utils/pinctrl_nrf_migrate.py | 194 +++++++++++++++++++++++---- 1 file changed, 168 insertions(+), 26 deletions(-) diff --git a/scripts/utils/pinctrl_nrf_migrate.py b/scripts/utils/pinctrl_nrf_migrate.py index 6c6438d9df..512313d335 100644 --- a/scripts/utils/pinctrl_nrf_migrate.py +++ b/scripts/utils/pinctrl_nrf_migrate.py @@ -20,6 +20,8 @@ partially ported. Devicetree files will be converted correctly. **ADJUSTED/GENERATED FILES MUST BE MANUALLY REVIEWED**. +Known limitations: All SPI nodes will be assumed to be a master device. + Usage:: python3 pinctrl_nrf_migrate.py @@ -95,17 +97,24 @@ class PIN_CONFIG(enum.Enum): PULL_UP = "bias-pull-up" PULL_DOWN = "bias-pull-down" LOW_POWER = "low-power-enable" + NORDIC_INVERT = "nordic,invert" class Device(object): """Device configuration class""" def __init__( - self, pattern: str, callback: Callable, signals: Dict[str, str] + self, + pattern: str, + callback: Callable, + signals: Dict[str, str], + needs_sleep: bool, ) -> None: self.pattern = pattern self.callback = callback self.signals = signals + self.needs_sleep = needs_sleep + self.attrs = {} class SignalMapping(object): @@ -272,7 +281,7 @@ def fmt_pinctrl_groups(groups: List[PinGroup]) -> str: # write psels entries for i, mapping in enumerate(group.pins): - prefix = "psels = " if i == 0 else " " + prefix = "psels = " if i == 0 else "\t" suffix = ";" if i == len(group.pins) - 1 else "," pin = mapping.pin port = 0 if pin < 32 else 1 @@ -291,24 +300,33 @@ def fmt_pinctrl_groups(groups: List[PinGroup]) -> str: return content -def fmt_states(device: str, indent: str) -> str: +def fmt_states(device: str, indent: str, needs_sleep: bool) -> str: """Format state entries for the given device. Args: device: Device name. indent: Indentation. + needs_sleep: If sleep entry is needed. Returns: State entries to be appended to the device. """ - return "\n".join( - ( - f"{indent}pinctrl-0 = <&{device}_default>;", - f"{indent}pinctrl-1 = <&{device}_sleep>;", - f'{indent}pinctrl-names = "default", "sleep";\n', + if needs_sleep: + return "\n".join( + ( + f"{indent}pinctrl-0 = <&{device}_default>;", + f"{indent}pinctrl-1 = <&{device}_sleep>;", + f'{indent}pinctrl-names = "default", "sleep";\n', + ) + ) + else: + return "\n".join( + ( + f"{indent}pinctrl-0 = <&{device}_default>;", + f'{indent}pinctrl-names = "default";\n', + ) ) - ) def insert_pinctrl_include(content: List[str], board: str) -> None: @@ -365,6 +383,7 @@ def adjust_content(content: List[str], board: str) -> List[DeviceConfiguration]: configs: List[DeviceConfiguration] = [] level = 0 in_device = False + states_written = False new_content = [] @@ -384,10 +403,11 @@ def adjust_content(content: List[str], board: str) -> List[DeviceConfiguration]: break # we are now inside a device node - in_device = True level = 1 + in_device = True + states_written = False else: - # entering subnode + # entering subnode (must come after all properties) if re.match(r"[^\/\*]*{.*", line): level += 1 # exiting subnode (or device node) @@ -407,10 +427,14 @@ def adjust_content(content: List[str], board: str) -> List[DeviceConfiguration]: # process each device line, append states at the end if current_device: - if in_device: + if level == 1: line = current_device.callback(config, current_device.signals, line) - else: - line = fmt_states(config.name, indent) + line + if (level == 2 or not in_device) and not states_written: + line = ( + fmt_states(config.name, indent, current_device.needs_sleep) + + line + ) + states_written = True current_device = None if line: @@ -431,7 +455,7 @@ def adjust_content(content: List[str], board: str) -> List[DeviceConfiguration]: def match_and_store_pin( config: DeviceConfiguration, signals: Dict[str, str], line: str -) -> bool: +) -> Optional[str]: """Match and store a pin mapping. Args: @@ -440,15 +464,23 @@ def match_and_store_pin( line: Line containing potential pin mapping. Returns: - True if line contains a pin mapping, False otherwise. + Line if found a pin mapping, None otherwise. """ - m = re.match(r"\s*([a-z]+)-pin\s*=\s*<(\d+)>.*", line) + # handle qspi special case for io-pins (array case) + m = re.match(r"\s*io-pins\s*=\s*([\s<>,0-9]+).*", line) + if m: + pins = re.sub(r"[<>,]", "", m.group(1)).split() + for i, pin in enumerate(pins): + config.set_signal_pin(signals[f"io{i}"], int(pin)) + return + + m = re.match(r"\s*([a-z]+\d?)-pins?\s*=\s*<(\d+)>.*", line) if m: config.set_signal_pin(signals[m.group(1)], int(m.group(2))) - return True + return - return False + return line # @@ -460,14 +492,52 @@ def process_uart(config: DeviceConfiguration, signals, line: str) -> Optional[st """Process UART/UARTE devices.""" # check if line specifies a pin - if match_and_store_pin(config, signals, line): - return None + if not match_and_store_pin(config, signals, line): + return # check if pull-up is specified m = re.match(r"\s*([a-z]+)-pull-up.*", line) if m: config.add_signal_config(signals[m.group(1)], PIN_CONFIG.PULL_UP) - return None + return + + return line + + +def process_spi(config: DeviceConfiguration, signals, line: str) -> Optional[str]: + """Process SPI devices.""" + + # check if line specifies a pin + if not match_and_store_pin(config, signals, line): + return + + # check if pull-up is specified + m = re.match(r"\s*miso-pull-up.*", line) + if m: + config.add_signal_config(signals["miso"], PIN_CONFIG.PULL_UP) + return + + # check if pull-down is specified + m = re.match(r"\s*miso-pull-down.*", line) + if m: + config.add_signal_config(signals["miso"], PIN_CONFIG.PULL_DOWN) + return + + return line + + +def process_pwm(config: DeviceConfiguration, signals, line: str) -> Optional[str]: + """Process PWM devices.""" + + # check if line specifies a pin + if not match_and_store_pin(config, signals, line): + return + + # check if channel inversion is specified + m = re.match(r"\s*([a-z0-9]+)-inverted.*", line) + if m: + config.add_signal_config(signals[m.group(1)], PIN_CONFIG.NORDIC_INVERT) + return return line @@ -482,6 +552,81 @@ DEVICES = [ "rts": "UART_RTS", "cts": "UART_CTS", }, + needs_sleep=True, + ), + Device( + r"i2c\d", + match_and_store_pin, + { + "sda": "TWIM_SDA", + "scl": "TWIM_SCL", + }, + needs_sleep=True, + ), + Device( + r"spi\d", + process_spi, + { + "sck": "SPIM_SCK", + "miso": "SPIM_MISO", + "mosi": "SPIM_MOSI", + }, + needs_sleep=True, + ), + Device( + r"pdm\d", + match_and_store_pin, + { + "clk": "PDM_CLK", + "din": "PDM_DIN", + }, + needs_sleep=False, + ), + Device( + r"qdec", + match_and_store_pin, + { + "a": "QDEC_A", + "b": "QDEC_B", + "led": "QDEC_LED", + }, + needs_sleep=True, + ), + Device( + r"qspi", + match_and_store_pin, + { + "sck": "QSPI_SCK", + "io0": "QSPI_IO0", + "io1": "QSPI_IO1", + "io2": "QSPI_IO2", + "io3": "QSPI_IO3", + "csn": "QSPI_CSN", + }, + needs_sleep=True, + ), + Device( + r"pwm\d", + process_pwm, + { + "ch0": "PWM_OUT0", + "ch1": "PWM_OUT1", + "ch2": "PWM_OUT2", + "ch3": "PWM_OUT3", + }, + needs_sleep=True, + ), + Device( + r"i2s\d", + match_and_store_pin, + { + "sck": "I2S_SCK_M", + "lrck": "I2S_LRCK_M", + "sdout": "I2S_SDOUT", + "sdin": "I2S_SDIN", + "mck": "I2S_MCK", + }, + needs_sleep=False, ), ] """Supported devices and associated configuration""" @@ -533,10 +678,7 @@ if __name__ == "__main__": help="Skip checking if board is nRF-based", ) parser.add_argument( - "--header", - default="", - type=str, - help="Header to be prepended to pinctrl files" + "--header", default="", type=str, help="Header to be prepended to pinctrl files" ) args = parser.parse_args()