Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

treewide: Use hyphens instead of underscores for property separation #83352

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions .ruff-excludes.toml
Original file line number Diff line number Diff line change
Expand Up @@ -435,6 +435,9 @@
"I001", # https://docs.astral.sh/ruff/rules/unsorted-imports
"UP035", # https://docs.astral.sh/ruff/rules/deprecated-import
]
"./scripts/migrate_bindings_style.py" = [
"I001", # https://docs.astral.sh/ruff/rules/unsorted-imports
]
Comment on lines +438 to +440
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably can work around this using ruff - @pdgendt is fairly familiar with that.

"./scripts/dts/python-devicetree/src/devicetree/dtlib.py" = [
"E701", # https://docs.astral.sh/ruff/rules/multiple-statements-on-one-line-colon
"I001", # https://docs.astral.sh/ruff/rules/unsorted-imports
Expand Down
3 changes: 3 additions & 0 deletions doc/releases/migration-guide-4.1.rst
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,9 @@ Devicetree
The :kconfig:option:`CONFIG_INPUT_CAP1203_PERIOD` has been replaced with
the devicetree property ``poll-interval-ms``.
In interrupt mode, the devicetree property ``repeat`` is supported.
* Property names in devicetree and bindings use hyphens(``-``) as separators, and replacing
all previously used underscores(``_``). For local code, you can migrate property names in
bindings to use hyphens by running the ``scripts/migrate_bindings_style.py`` script.

Raspberry Pi
============
Expand Down
11 changes: 11 additions & 0 deletions scripts/ci/bindings_propertys_name_wl.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# This file is a YAML whitelist of property names that are allowed to
# bypass the underscore check in bindings. These properties are exempt
# from the rule that requires using '-' instead of '_'.
#
# The file content can be as shown below:
# - propname1
# - propname2
# - ...

- mmc-hs200-1_8v
- mmc-hs400-1_8v
114 changes: 85 additions & 29 deletions scripts/ci/check_compliance.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
import shutil
import textwrap
import unidiff
import yaml

from yamllint import config, linter

Expand All @@ -36,6 +37,28 @@
import list_boards
import list_hardware

sys.path.insert(0, os.path.join(str(Path(__file__).resolve().parents[2]),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm surprised that so much needs to be added in this file

'scripts/dts/python-devicetree/src'))
from devicetree import edtlib


# Let the user run this script as ./scripts/ci/check_compliance.py without
# making them set ZEPHYR_BASE.
ZEPHYR_BASE = os.environ.get('ZEPHYR_BASE')
if not ZEPHYR_BASE:
ZEPHYR_BASE = str(Path(__file__).resolve().parents[2])
# Propagate this decision to child processes.
os.environ['ZEPHYR_BASE'] = ZEPHYR_BASE

# Initialize the propertys name whitelist
BINDINGS_PROPERTYS_WL = None
with open(Path(__file__).parent / 'bindings_propertys_name_wl.yaml', 'r') as f:
wl = yaml.safe_load(f.read())
if wl is not None:
BINDINGS_PROPERTYS_WL = set(wl)
else:
BINDINGS_PROPERTYS_WL = {'',}

logger = None

def git(*args, cwd=None, ignore_non_zero=False):
Expand Down Expand Up @@ -335,32 +358,75 @@ class DevicetreeBindingsCheck(ComplianceTest):
path_hint = "<zephyr-base>"

def run(self, full=True):
dts_bindings = self.parse_dt_bindings()

for dts_binding in dts_bindings:
self.required_false_check(dts_binding)
bindings_diff, bindings = self.get_yaml_bindings()

def parse_dt_bindings(self):
# If no bindings are changed, skip this check.
try:
subprocess.check_call(['git', 'diff', '--quiet', COMMIT_RANGE]
+ bindings_diff)
nodiff = True
except subprocess.CalledProcessError:
nodiff = False
if nodiff:
self.skip('no changes to bindings were made')

for binding in bindings:
self.check(binding, self.check_yaml_property_name)
self.check(binding, self.required_false_check)

@staticmethod
def check(binding, callback):
while binding is not None:
callback(binding)
binding = binding.child_binding

def get_yaml_bindings(self):
"""
Returns a list of dts/bindings/**/*.yaml files
Returns a list of 'dts/bindings/**/*.yaml'
"""
from glob import glob
BINDINGS_PATH = 'dts/bindings/'
bindings_diff_dir, bindings = set(), []

dt_bindings = []
for file_name in get_files(filter="d"):
if 'dts/bindings/' in file_name and file_name.endswith('.yaml'):
dt_bindings.append(file_name)
for file_name in get_files(filter='d'):
if BINDINGS_PATH in file_name:
p = file_name.partition(BINDINGS_PATH)
bindings_diff_dir.add(os.path.join(p[0], p[1]))

return dt_bindings
for path in bindings_diff_dir:
yamls = glob(f'{os.fspath(path)}/**/*.yaml', recursive=True)
bindings.extend(yamls)

def required_false_check(self, dts_binding):
with open(dts_binding) as file:
for line_number, line in enumerate(file, 1):
if 'required: false' in line:
self.fmtd_failure(
'warning', 'Devicetree Bindings', dts_binding,
line_number, col=None,
desc="'required: false' is redundant, please remove")
bindings = edtlib.bindings_from_paths(bindings, ignore_errors=True)
return list(bindings_diff_dir), bindings

def check_yaml_property_name(self, binding):
"""
Checks if the property names in the binding file contain underscores.
"""
for prop_name in binding.prop2specs:
if '_' in prop_name and prop_name not in BINDINGS_PROPERTYS_WL:
better_prop = prop_name.replace('_', '-')
print(f"Required: In '{binding.path}', "
f"the property '{prop_name}' "
f"should be renamed to '{better_prop}'.")
self.failure(
f"{binding.path}: property '{prop_name}' contains underscores.\n"
f"\tUse '{better_prop}' instead unless this property name is from Linux.\n"
"Or another authoritative upstream source of bindings for "
f"compatible '{binding.compatible}'.\n"
"\tHint: update 'bindings_propertys_name_wl.yaml' if you need to "
"override this check for this property."
)

def required_false_check(self, binding):
raw_props = binding.raw.get('properties', {})
for prop_name, raw_prop in raw_props.items():
if raw_prop.get('required') is False:
self.failure(
f'{binding.path}: property "{prop_name}": '
"'required: false' is redundant, please remove"
)

class KconfigCheck(ComplianceTest):
"""
Expand Down Expand Up @@ -1802,16 +1868,6 @@ def _main(args):
# The "real" main(), which is wrapped to catch exceptions and report them
# to GitHub. Returns the number of test failures.

global ZEPHYR_BASE
ZEPHYR_BASE = os.environ.get('ZEPHYR_BASE')
if not ZEPHYR_BASE:
# Let the user run this script as ./scripts/ci/check_compliance.py without
# making them set ZEPHYR_BASE.
ZEPHYR_BASE = str(Path(__file__).resolve().parents[2])

# Propagate this decision to child processes.
os.environ['ZEPHYR_BASE'] = ZEPHYR_BASE

# The absolute path of the top-level git directory. Initialize it here so
# that issues running Git can be reported to GitHub.
global GIT_TOP
Expand Down
91 changes: 91 additions & 0 deletions scripts/migrate_bindings_style.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
#!/usr/bin/env python3

# Copyright (c) 2025 James Roy <[email protected]>
# SPDX-License-Identifier: Apache-2.0

# This script is used to replace the properties containing
# underscores in the binding.


import argparse
import sys
import os
from pathlib import Path

import yaml


# bindings base
BINDINGS_PATH = ["dts/bindings/"]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably want this to be a list of Path objects

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Strings list are convenient for command lines to pass parameters, see line 84

BINDINGS_PROPERTYS_WL = {
"mmc-hs400-1_8v",
"mmc-hs200-1_8v",
}


def parse_cmd_args():
parse = argparse.ArgumentParser(allow_abbrev=False)
parse.add_argument("--path", nargs="+", help="User binding path directory")
return parse.parse_args()


def get_yaml_binding_paths() -> list:
"""Returns a list of 'dts/bindings/**/*.yaml'"""
from glob import glob

bindings = []
for path in BINDINGS_PATH:
yamls = glob(f"{os.fspath(path)}/**/*.yaml", recursive=True)
bindings.extend(yamls)
return bindings


def replace_bindings_underscores(binding_paths):
"""
Replace all property names containing underscores in the bindings.
"""
for binding_path in binding_paths:
with open(binding_path, encoding="utf-8") as f:
yaml_binding = yaml.safe_load(f)
properties = yaml_binding.get("properties", {})
for prop_name in properties:
if '_' in prop_name and prop_name not in BINDINGS_PROPERTYS_WL:
_replace_underscores(binding_path, prop_name)


def _replace_underscores(binding_path, prop_name):
"""Replace implementation"""
with open(binding_path, "r+", encoding="utf-8") as f:
lines = f.readlines()
for i, rowline in enumerate(lines):
line = rowline.split(":")[0].strip()
if prop_name == line and "#" not in rowline:
new_key = prop_name.replace("_", "-")
if new_key != line:
lines[i] = rowline.replace(line, new_key)
f.seek(0)
f.writelines(lines)


def check_valid_cmd_path():
"""
Checks whether the given command line argument is a valid path.
"""
for path in BINDINGS_PATH:
path = Path(path)
if not path.exists() or not path.is_dir():
raise FileNotFoundError(f"No such directory: '{path}'")


if __name__ == '__main__':
args = parse_cmd_args()

if args.path is not None:
BINDINGS_PATH.extend(args.path)

check_valid_cmd_path()

bindings = get_yaml_binding_paths()
replace_bindings_underscores(bindings)

sys.exit(0)
Loading