Merge pull request #1162 from gedankenexperimenter/iwyu-simulator

Expand and improve IWYU coverage
pull/1181/head
Jesse Vincent 3 years ago committed by GitHub
commit 8c15bb79ec
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -36,7 +36,7 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- run: make check-code-style - run: KALEIDOSCOPE_CODE_FORMATTER=clang-format-12 make check-code-style
check-shellcheck: check-shellcheck:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:

@ -1,3 +1,4 @@
examples
src/Kaleidoscope.h src/Kaleidoscope.h
src/Kaleidoscope-LEDControl.h src/Kaleidoscope-LEDControl.h
src/kaleidoscope/HIDTables.h src/kaleidoscope/HIDTables.h
@ -10,3 +11,4 @@ src/kaleidoscope/driver/storage/GD32Flash.h
plugins/Kaleidoscope-FirmwareDump/** plugins/Kaleidoscope-FirmwareDump/**
plugins/Kaleidoscope-HostOS/src/kaleidoscope/plugin/HostOS.h plugins/Kaleidoscope-HostOS/src/kaleidoscope/plugin/HostOS.h
plugins/Kaleidoscope-Hardware-EZ-ErgoDox/src/kaleidoscope/device/ez/ErgoDox/i2cmaster.h plugins/Kaleidoscope-Hardware-EZ-ErgoDox/src/kaleidoscope/device/ez/ErgoDox/i2cmaster.h
testing/googletest

@ -113,21 +113,28 @@ check-code-style:
bin/format-code.py \ bin/format-code.py \
--exclude-dir 'testing/googletest' \ --exclude-dir 'testing/googletest' \
--exclude-file 'generated-testcase.cpp' \ --exclude-file 'generated-testcase.cpp' \
--force \
--check \ --check \
--verbose \ --verbose \
src plugins examples testing src plugins examples testing
.PHONY: check-includes .PHONY: check-includes
check-includes: check-includes:
bin/fix-header-includes
.PHONY: check-all-includes
check-all-includes:
bin/iwyu.py -v src plugins bin/iwyu.py -v src plugins
bin/format-code.py -f -v --check src plugins bin/iwyu.py -v \
-I=$(KALEIDOSCOPE_DIR) \
-I=$(KALEIDOSCOPE_DIR)/testing/googletest/googlemock/include \
-I=$(KALEIDOSCOPE_DIR)/testing/googletest/googletest/include \
testing
bin/format-code.py -f -v --check src plugins testing
.PHONY: cpplint-noisy .PHONY: cpplint-noisy
cpplint-noisy: cpplint-noisy:
-bin/cpplint.py --config=.cpplint-noisy --recursive src plugins examples -bin/cpplint.py --config=.cpplint-noisy --recursive src plugins examples
.PHONY: cpplint .PHONY: cpplint
cpplint: cpplint:
bin/cpplint.py --config=.cpplint --quiet --recursive src plugins examples bin/cpplint.py --config=.cpplint --quiet --recursive src plugins examples

@ -0,0 +1,110 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
# Copyright (c) 2022 Michael Richters <gedankenexperimenter@gmail.com>
# This is free and unencumbered software released into the public domain.
# Anyone is free to copy, modify, publish, use, compile, sell, or
# distribute this software, either in source code form or as a compiled
# binary, for any purpose, commercial or non-commercial, and by any
# means.
# In jurisdictions that recognize copyright laws, the author or authors
# of this software dedicate any and all copyright interest in the
# software to the public domain. We make this dedication for the benefit
# of the public at large and to the detriment of our heirs and
# successors. We intend this dedication to be an overt act of
# relinquishment in perpetuity of all present and future rights to this
# software under copyright law.
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
# IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
# OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
# ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
# OTHER DEALINGS IN THE SOFTWARE.
# For more information, please refer to <http://unlicense.org/>
# ------------------------------------------------------------------------------
"""Utilities shared by Kaleidoscope code maintenance tools."""
import logging
import os
import subprocess
import sys
from contextlib import contextmanager
# ==============================================================================
@contextmanager
def cwd(temp_wd):
"""Execute code in a different working directory
Parameters:
`temp_wd` (`str`): The name of a directory to (temporarily) change to
Change the current working directory, then automatically restore the previous working
directory when done. Invoke `cwd()` like this:
```py
with cwd(temp_wd):
...
```
"""
old_wd = os.getcwd()
os.chdir(temp_wd)
try:
yield
finally:
os.chdir(old_wd)
# ==============================================================================
def split_on_newlines(string):
"""Split the string using newlines as the separator."""
return string.splitlines()
# ------------------------------------------------------------------------------
def split_on_nulls(string):
"""Split the input string using NULL characters as the separator."""
targets = [_ for _ in string.split('\0') if _ != '']
return targets or []
# ==============================================================================
def setup_logging(loglevel):
"""Set up basic logging."""
logformat = "%(message)s"
logging.basicConfig(
level=loglevel,
stream=sys.stdout,
format=logformat,
datefmt="",
)
return logging.getLogger()
# ==============================================================================
def check_git_diff():
"""Check for unstaged changes with `git diff`
Returns: a list of the names of files with unstaged changes
This check isn't perfect, because it can give false positives (if there are unrelated
unstaged changes).
"""
git_diff_cmd = ['git', 'diff', '-z', '--exit-code', '--name-only']
changed_files = []
proc = subprocess.run(git_diff_cmd, capture_output=True)
if proc.returncode != 0:
changed_files = split_on_nulls(proc.stdout.decode('utf-8'))
return changed_files
# ==============================================================================
if __name__ == "__main__":
sys.exit(1)

@ -1,6 +1,43 @@
#!/usr/bin/env bash #!/usr/bin/env bash
# In case it hasn't been set, assume we're being run from the root of the local
# Kaleidoscope repository.
: "${KALEIDOSCOPE_DIR:=$(pwd)}" : "${KALEIDOSCOPE_DIR:=$(pwd)}"
cd "${KALEIDOSCOPE_DIR}" || exit 1
git ls-files -m | grep -E '\.(h|cpp)$' | xargs "${KALEIDOSCOPE_DIR}"/bin/iwyu.py # This variable is a git ref that should point to the current master branch of
# the primary Kaleidoscope repository. In most cases, this would be
# `origin/master`.
: "${KALEIDOSCOPE_MERGE_BASE:=origin/master}"
# Don't do anything if the working tree has unstaged changes, to avoid
# unintentional combining of contentful changes with formatting changes.
if ! git diff -z --exit-code --quiet; then
echo "Working tree has unstaged changes; aborting."
exit 1
fi
# Run git-diff so we only run IWYU on files that differ from `master`. This
# isn't necessarily what we want, but if the current branch has been rebased, it
# shouldn't touch any extra files.
git diff -z --name-only "${KALEIDOSCOPE_MERGE_BASE}" -- src plugins \
| "${KALEIDOSCOPE_DIR}/bin/iwyu.py" -z -v
# After running it on Kaleidoscope source files, run it on the test simulator,
# which requires some additional include dirs.
git diff -z --name-only "${KALEIDOSCOPE_MERGE_BASE}" -- testing \
| "${KALEIDOSCOPE_DIR}/bin/iwyu.py" \
-z -v \
-I="${KALEIDOSCOPE_DIR}" \
-I="${KALEIDOSCOPE_DIR}/testing/googletest/googlemock/include" \
-I="${KALEIDOSCOPE_DIR}/testing/googletest/googletest/include"
# Always run clang-format after IWYU, because they have different indentation
# rules for comments added by IWYU.
git diff -z --name-only "${KALEIDOSCOPE_MERGE_BASE}" -- src plugins testing \
| "${KALEIDOSCOPE_DIR}/bin/format-code.py" \
-z -v \
--exclude-dir='testing/googletest' \
--exclude-file='generated-testcase.cpp' \
--force \
--check \
--verbose

@ -28,23 +28,27 @@
# For more information, please refer to <http://unlicense.org/> # For more information, please refer to <http://unlicense.org/>
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
"""This script runs clang-format on Kaleidoscope's codebase.""" """This script runs clang-format on a Kaleidoscope repository."""
import argparse import argparse
import glob
import logging import logging
import os import os
import re import re
import shutil
import subprocess import subprocess
import sys import sys
sys.dont_write_bytecode = True
from common import check_git_diff, setup_logging, split_on_newlines, split_on_nulls
# ============================================================================== # ==============================================================================
def parse_args(args): def parse_args(args):
"""Parse command line parameters """Parse command line parameters
Args: Args:
args (List[str]): command line parameters as list of strings args (list[str]): command line parameters as list of strings
(for example ``["--help"]``). (for example ``["--help"]``).
Returns: Returns:
@ -55,217 +59,217 @@ def parse_args(args):
Recursively search specified directories and format source files with Recursively search specified directories and format source files with
clang-format. By default, it operates on Arduino C++ source files with clang-format. By default, it operates on Arduino C++ source files with
extensions: *.{cpp,h,hpp,inc,ino}.""") extensions: *.{cpp,h,hpp,inc,ino}.""")
parser.add_argument(
'-q',
'--quiet',
dest='loglevel',
action='store_const',
const=logging.ERROR,
default=logging.WARNING,
help="""
Suppress output except warnings and errors.""",
)
parser.add_argument( parser.add_argument(
'-v', '-v',
'--verbose', '--verbose',
dest='loglevel',
help="Verbose output",
action='store_const', action='store_const',
dest='loglevel',
const=logging.INFO, const=logging.INFO,
help="""
Output verbose debugging information.""",
) )
parser.add_argument( parser.add_argument(
'-q', '-d',
'--quiet', '--debug',
dest='loglevel',
help="Suppress all non-error output",
action='store_const', action='store_const',
const=logging.ERROR, dest='loglevel',
const=logging.DEBUG,
help="""
Save output from `include-what-you-use` for processed files beside the originals, with
a '.iwyu' suffix, for debugging purposes.""",
) )
parser.add_argument( parser.add_argument(
'-X', '-X',
'--exclude-dir', '--exclude-dir',
metavar="<path>",
dest='exclude_dirs',
help="Exclude dir from search (path relative to the pwd)",
action='append', action='append',
dest='exclude_dirs',
default=[], default=[],
metavar="<path>",
help="""
Exclude dir from search (path relative to the pwd)""",
) )
parser.add_argument( parser.add_argument(
'-x', '-x',
'--exclude-file', '--exclude-file',
metavar="<file>",
dest='exclude_files',
help="Exclude <file> (base name only, not a full path) from formatting",
action='append', action='append',
dest='exclude_files',
default=[], default=[],
metavar="<file>",
help="""
Exclude <file> (base name only, not a full path) from formatting""",
) )
parser.add_argument( parser.add_argument(
'-e', '-r',
'--regex', '--regex',
metavar="<regex>", dest='regex',
dest='src_re_str',
help="Regular expression for matching source file names",
default=r'\.(cpp|h|hpp|inc|ino)$', default=r'\.(cpp|h|hpp|inc|ino)$',
metavar="<regex>",
help="""
Regular expression for matching source file names""",
)
parser.add_argument(
'-z',
'-0',
action='store_const',
dest='input_splitter',
const=split_on_nulls,
default=split_on_newlines,
help="""
When reading target filenames from standard input, break on NULL characters instead
of newlines.""",
) )
parser.add_argument( parser.add_argument(
'-f', '-f',
'--force', '--force',
action='store_true', action='store_true',
help="Format code even if there are unstaged changes", help="""
Format code even if there are unstaged changes""",
) )
parser.add_argument( parser.add_argument(
'--check', '--check',
action='store_true', action='store_true',
help="Check for changes after formatting", help="""
Check for changes after formatting by running `git diff --exit-code`. If there are any
changes after formatting, a non-zero exit code is returned.""",
) )
parser.add_argument( parser.add_argument(
'targets', 'targets',
metavar="<search_dir>", metavar="<search_dir>",
nargs='+', nargs='*',
help="""A list of files and/or directories to search for source files to format""", help="""
A list of files and/or directories to search for source files to format.""",
) )
return parser.parse_args(args) return parser.parse_args(args)
# ============================================================================== # ==============================================================================
def setup_logging(loglevel): def main():
"""Setup basic logging """Parse command-line arguments and format source files.
Args:
loglevel (int): minimum loglevel for emitting messages
""" """
logformat = "%(message)s" # Parse command-line argumets:
logging.basicConfig( opts = parse_args(sys.argv[1:])
level=loglevel, # Set up logging system:
stream=sys.stdout, setup_logging(opts.loglevel)
format=logformat,
datefmt="", # ----------------------------------------------------------------------
) # Unless we've been given the `--force` option, check for unstaged changes to avoid
return logging.getLogger() # clobbering any work in progress:
exit_code = 0
if not opts.force:
# ============================================================================== changed_files = check_git_diff()
def format_code(path, opts, clang_format_cmd): if len(changed_files) > 0:
"""Run clang-format on a directory.""" logging.error("Working tree has unstaged changes; aborting")
logging.info("Formatting code in %s...", path) return 1
src_regex = re.compile(opts.src_re_str) # Locate `clang-format` executable:
clang_format_exe = os.getenv('KALEIDOSCOPE_CODE_FORMATTER')
src_files = [] if clang_format_exe is None:
clang_format_exe = shutil.which('clang-format')
for root, dirs, files in os.walk(path): logging.debug("Found `clang-format` executable: %s", clang_format_exe)
for exclude_path in opts.exclude_dirs: clang_format_cmd = [clang_format_exe, '-i']
exclude_path = exclude_path.rstrip(os.path.sep) if opts.loglevel <= logging.INFO:
if os.path.dirname(exclude_path) == root: clang_format_cmd.append('--verbose')
exclude_dir = os.path.basename(exclude_path)
if exclude_dir in dirs: # ----------------------------------------------------------------------
dirs.remove(exclude_dir) # Read targets from command line:
targets = opts.targets
for name in files: logging.debug("CLI target parameters: %s", targets)
if name in opts.exclude_files:
continue # If stdin is a pipe, read target filenames from it:
if src_regex.search(name): if not sys.stdin.isatty():
src_files.append(os.path.join(root, name)) targets += opts.input_splitter(sys.stdin.read())
logging.debug("All targets: %s", targets)
proc = subprocess.run(clang_format_cmd + src_files)
# Prepare exclusion lists. The file excludes are basenames only, and the dirs get
# converted to absolute path names.
exclude_files = set(opts.exclude_files)
exclude_dirs = set(os.path.abspath(_) for _ in opts.exclude_dirs)
# Convert target paths to absolute, and remove any that are excluded:
target_paths = set(os.path.abspath(_) for _ in targets if _ not in exclude_dirs)
logging.debug("Target paths: %s", target_paths)
# Build separate sets of target files and dirs. Later, we'll search target dirs and add
# matching target files to the files set.
target_files = set()
target_dirs = set()
for t in target_paths:
if os.path.isfile(t):
target_files.add(os.path.abspath(t))
elif os.path.isdir(t):
target_dirs.add(os.path.abspath(t))
logging.debug("Target files after separating: %s", target_files)
logging.debug("Target dirs after separating: %s", target_dirs)
# Remove excluded filenames:
target_files -= set(_ for _ in target_files if os.path.basename(_) in exclude_files)
# Remove files and dirs in excluded dirs:
target_files -= set(_ for _ in target_files for x in exclude_dirs if _.startswith(x))
target_dirs -= set(_ for _ in target_dirs for x in exclude_dirs if _.startswith(x))
# Compile regex for matching files to be formatted:
target_matcher = re.compile(opts.regex)
# Remove target files that don't match the regex:
logging.debug("Target files before matching regex: %s", target_files)
target_files = set(_ for _ in target_files if target_matcher.search(_))
logging.debug("Target files after matching regex: %s", target_files)
# Search target dirs for non-excluded files, and add them to `target_files`:
logging.debug("Searching target dirs: %s", target_dirs)
for path in target_dirs:
for root, dirs, files in os.walk(path):
# Prune excluded dirs
for x in exclude_dirs:
if x in (os.path.join(root, _) for _ in dirs):
dirs.remove(os.path.basename(x))
# Add non-excluded files
for f in files:
if target_matcher.search(f) and f not in exclude_files:
target_files.add(os.path.join(root, f))
if len(target_files) == 0:
logging.error("No target files found; exiting.")
return 1
# Run clang-format on target files:
proc = subprocess.run(clang_format_cmd + sorted(target_files))
if proc.returncode != 0: if proc.returncode != 0:
logging.error("Error: clang-format returned non-zero status: %s", proc.returncode) logging.error("Error: clang-format returned non-zero status: %s", proc.returncode)
return return proc.returncode
else:
logging.info("Finished formatting target files.")
# ==============================================================================
def build_file_list(path, src_regex):
"""Docstring"""
# If the specified path is a filename, return it (as a list), regardless of
# whether or not it matches the regex.
if os.path.isfile(path):
return [path]
# If the specified path is not valid, just return an empty list.
if not os.path.isdir(path):
return []
# The specified path is a directory, so we search recursively for files
# contained therein that match the specified regular expression.
source_files = []
for root, dirs, files in os.walk(path):
# First, ignore all dotfiles (and directories).
dotfiles = set(glob.glob('.*'))
dirs = set(dirs) - dotfiles
files = set(dirs) - dotfiles
# Check for a list of file glob patterns that should be excluded.
if IWYU_IGNORE_FILE in files:
with open(os.path.join(root, IWYU_IGNORE_FILE)) as f:
for pattern in f.read().splitlines():
matches = set(glob.glob(os.path.join(root, pattern)))
dirs = set(dirs) - matches
files = set(files) - matches
# Add all matching files to the list of source files to be formatted.
for f in filter(src_regex.search, files):
source_files.append(os.path.join(root, f))
return source_files
# If we've been asked to check for changes made by the formatter:
# ============================================================================== if opts.check:
def warn_if_output(byte_str, msg):
"""Convert a string of bytes to a UTF-8 string, break it on newlines. If
there is any output, print a warning message, followed by each line,
indented by four spaces."""
lines = byte_str.decode('utf-8').splitlines()
if '' in lines:
lines.remove('')
if len(lines) > 0:
logging.warning('%s', msg)
for line in lines:
logging.warning(' %s', line)
return
# ==============================================================================
def main(cli_args):
"""Parse command-line arguments and format source files."""
args = parse_args(cli_args)
if args.loglevel is None:
args.loglevel = logging.WARNING
setup_logging(args.loglevel)
clang_format = os.getenv('CLANG_FORMAT_CMD')
if clang_format is None:
clang_format = 'clang-format'
clang_format_cmd = [clang_format, '-i']
git_diff_cmd = ['git', 'diff', '--exit-code']
proc = subprocess.run(git_diff_cmd + ['--name-only'], capture_output=True)
if proc.returncode != 0:
warn_if_output(proc.stdout, 'Warning: you have unstaged changes to these files:')
if not args.force:
logging.warning(
'Formatting aborted. Stage your changes or use --force to override.')
sys.exit(proc.returncode)
if args.loglevel >= logging.WARNING:
git_diff_cmd.append('--quiet')
elif args.loglevel <= logging.INFO:
git_diff_cmd.append('--name-only')
for path in args.targets:
format_code(path, args, clang_format_cmd)
if args.check:
logging.warning('Checking for changes made by the formatter...') logging.warning('Checking for changes made by the formatter...')
changed_files = check_git_diff()
if len(changed_files) == 0:
logging.warning("No files changed. Congratulations!")
else:
logging.warning("Found files with changes after formatting:")
exit_code = 1
for f in changed_files:
logging.warning(" %s", f)
proc = subprocess.run(git_diff_cmd + ['--cached'], capture_output=True) return exit_code
if proc.returncode != 0:
logging.warning(
'Warning: Your working tree has staged changes. ' +
'Committed changes might not pass this check.')
warn_if_output(proc.stdout, 'The following files have unstaged changes:')
proc = subprocess.run(git_diff_cmd, capture_output=True)
if proc.returncode != 0:
warn_if_output(proc.stdout, 'The following files have changes:')
logging.error(
'Check failed: Please commit formatting changes before submitting.')
sys.exit(proc.returncode)
return
# ============================================================================== # ==============================================================================
if __name__ == "__main__": if __name__ == "__main__":
main(sys.argv[1:]) try:
sys.exit(main())
except KeyboardInterrupt:
logging.info("Aborting")
sys.exit(1)

@ -28,7 +28,9 @@
# For more information, please refer to <http://unlicense.org/> # For more information, please refer to <http://unlicense.org/>
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
"""This is a script for maintenance of the headers included in Kaleidoscope source """Run `include-what-you-use` on Kaleidoscope sources
This is a script for maintenance of the headers included in Kaleidoscope source
files. It is not currently possible to run this automatically on all files. It is not currently possible to run this automatically on all
Kaleidoscope source files, because of the peculiarities therein. It uses Kaleidoscope source files, because of the peculiarities therein. It uses
llvm/clang to determine which headers should be included in a given file, but llvm/clang to determine which headers should be included in a given file, but
@ -55,12 +57,25 @@ import shutil
import subprocess import subprocess
import sys import sys
sys.dont_write_bytecode = True
from common import cwd, setup_logging, split_on_newlines, split_on_nulls
# ============================================================================== # ==============================================================================
def parse_args(args): def parse_args(args):
"""Parse command line parameters
Args:
args (list[str]): command line parameters as list of strings
(for example ``["--help"]``).
Returns:
:obj:`argparse.Namespace`: command line parameters namespace
"""
parser = argparse.ArgumentParser( parser = argparse.ArgumentParser(
description= description="""
"""Run `include-what-you-use` on source files given as command-line arguments and/or read Run `include-what-you-use` on source files given as command-line arguments and/or read
from standard input. When reading target filenames from standard input, they should be from standard input. When reading target filenames from standard input, they should be
either absolute or relative to the current directory, and each line of input (minus the either absolute or relative to the current directory, and each line of input (minus the
line-ending character(s) is treated as a filename.""") line-ending character(s) is treated as a filename.""")
@ -68,56 +83,87 @@ def parse_args(args):
'-q', '-q',
'--quiet', '--quiet',
dest='loglevel', dest='loglevel',
help="Suppress output except warnings and errors.",
action='store_const', action='store_const',
const=logging.ERROR, const=logging.ERROR,
default=logging.WARNING, default=logging.WARNING,
help="""
Suppress output except warnings and errors.""",
) )
parser.add_argument( parser.add_argument(
'-v', '-v',
'--verbose', '--verbose',
dest='loglevel',
help="Output verbose debugging information.",
action='store_const', action='store_const',
dest='loglevel',
const=logging.INFO, const=logging.INFO,
help="""
Output verbose debugging information.""",
) )
parser.add_argument( parser.add_argument(
'-d', '-d',
'--debug', '--debug',
dest='loglevel',
help="""Save output from `include-what-you-use` for processed files beside the
originals, with a '.iwyu' suffix, for debugging purposes.""",
action='store_const', action='store_const',
dest='loglevel',
const=logging.DEBUG, const=logging.DEBUG,
help="""
Save output from `include-what-you-use` for processed files beside the originals, with
a '.iwyu' suffix, for debugging purposes.""",
) )
parser.add_argument( parser.add_argument(
'-r', '-r',
'--regex', '--regex',
action='store',
dest='regex', dest='regex',
help="""A regular expression for matching filenames Only the basename of the file is default=r'\.(h|cpp)$',
help="""
A regular expression for matching filenames Only the basename of the file is
matched, and the regex is only used when searching a directory for files to process, matched, and the regex is only used when searching a directory for files to process,
not on target filenames specified in arguments or read from standard input.""", not on target filenames specified in arguments or read from standard input.""",
action='store',
default=r'\.(h|cpp)$',
) )
parser.add_argument( parser.add_argument(
'-i', '-i',
'--ignores_file', '--ignores_file',
dest='ignores_file',
metavar='<ignores_file>',
help=
"""The name of a file (relative to KALEIDOSCOPE_DIR) that contains a list of glob
patterns that will be ignored when a target directory is searched for filenames that
match <regex>.""",
action='store', action='store',
dest='ignores_file',
default='.iwyu_ignore', default='.iwyu_ignore',
metavar='<ignores_file>',
help="""
The name of a file (relative to KALEIDOSCOPE_DIR) that contains a list of glob patterns
that will be ignored when a target directory is searched for filenames that match
<regex>.""",
)
parser.add_argument(
'-I',
'--include',
action='append',
dest='include_dirs',
metavar='<dir>',
help="""
Add <dir> to the list of directories that will be searched for header files.""",
)
parser.add_argument(
'-z',
'-0',
action='store_const',
dest='input_splitter',
const=split_on_nulls,
default=split_on_newlines,
help="""
When reading target filenames from standard input, break on NULL characters instead
of newlines.""",
)
parser.add_argument(
'--update_comments',
action='store_true',
help="""
Call `include-what-you-use` with `--update_comments` to always rewrite its 'for'
comments regarding which symbols are used.""",
) )
parser.add_argument( parser.add_argument(
'targets', 'targets',
nargs='*',
metavar="<target>", metavar="<target>",
nargs='+', help="""
help= A list of target files and/or directories to search for source files to format. Any
"""A list of target files and/or directories to search for source files to format. Any
target file will be processed, regardless of the filename. Any target directory will target file will be processed, regardless of the filename. Any target directory will
be recursively searched for files matching the regular expression given by --regex. be recursively searched for files matching the regular expression given by --regex.
Filenames and directories beginning with a '.' will always be excluded from the search, Filenames and directories beginning with a '.' will always be excluded from the search,
@ -126,22 +172,6 @@ def parse_args(args):
return parser.parse_args(args) return parser.parse_args(args)
# ==============================================================================
def setup_logging(loglevel):
"""Set up basic logging
Args:
:int:loglevel: minimum loglevel for emitting messages
"""
logformat = "%(message)s"
logging.basicConfig(
level=loglevel,
stream=sys.stdout,
format=logformat,
datefmt="",
)
# ============================================================================== # ==============================================================================
def main(): def main():
"""Main entry point function.""" """Main entry point function."""
@ -153,12 +183,13 @@ def main():
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
# Find include-what-you-use: # Find include-what-you-use:
iwyu = shutil.which('include-what-you-use') iwyu = shutil.which('include-what-you-use')
logging.info("Found `include-what-you-use` executable: %s", iwyu) logging.debug("Found `include-what-you-use` executable: %s", iwyu)
iwyu_opts = [ iwyu_opts = [
'--no_fwd_decls', # No forward declarations '--no_fwd_decls', # No forward declarations
'--max_line_length=100', '--max_line_length=100',
'--update_comments',
] ]
if opts.update_comments:
iwyu_opts.append('--update_comments')
# Prepend '-Xiwyu' to each `include-what-you-use` option: # Prepend '-Xiwyu' to each `include-what-you-use` option:
iwyu_opts = [_ for opt in iwyu_opts for _ in ('-Xiwyu', opt)] iwyu_opts = [_ for opt in iwyu_opts for _ in ('-Xiwyu', opt)]
@ -256,6 +287,9 @@ def main():
] ]
# Include plugin source dirs for plugins that depend on other plugins: # Include plugin source dirs for plugins that depend on other plugins:
includes += glob.glob(os.path.join(kaleidoscope_dir, 'plugins', '*', 'src')) includes += glob.glob(os.path.join(kaleidoscope_dir, 'plugins', '*', 'src'))
# Include dirs specified on the command line, if any:
if opts.include_dirs:
includes += [os.path.abspath(_) for _ in opts.include_dirs]
clang_opts += ['-I' + _ for _ in includes] clang_opts += ['-I' + _ for _ in includes]
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
@ -265,19 +299,23 @@ def main():
fix_includes_cmd = [ fix_includes_cmd = [
fix_includes, fix_includes,
'--update_comments',
'--nosafe_headers', '--nosafe_headers',
'--reorder', '--reorder',
'--separate_project_includes=' + kaleidoscope_src_dir, # Does this help? '--separate_project_includes=' + kaleidoscope_src_dir, # Does this help?
] ]
if opts.update_comments:
fix_includes_cmd.append('--update_comments')
logging.debug("Using `fix_includes` command: %s", ' \\\n\t'.join(fix_includes_cmd)) logging.debug("Using `fix_includes` command: %s", ' \\\n\t'.join(fix_includes_cmd))
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
targets = opts.targets targets = opts.targets
# If stdin is a pipe, read pathname targets, one per line. This allows us to # If stdin is a pipe, read pathname targets, one per line. This allows us to connect the
# connect the output of `find` to our input conveniently: # output of `find` to our input conveniently:
if not sys.stdin.isatty(): if not sys.stdin.isatty():
targets += sys.stdin.read().splitlines() logging.debug("Reading targets from STDIN...")
targets += opts.input_splitter(sys.stdin.read())
for t in targets:
logging.debug(" Target path: %s", t)
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
iwyu_ignores_file = os.path.join(kaleidoscope_dir, opts.ignores_file) iwyu_ignores_file = os.path.join(kaleidoscope_dir, opts.ignores_file)
@ -285,93 +323,142 @@ def main():
# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
regex = re.compile(opts.regex) regex = re.compile(opts.regex)
# Process source files first, then header files, because a source file might have been
# relying on a header included by its associated header, but which that header does not
# need on its own. In this case, if we process the header first, IWYU won't be able to
# parse the source file, and we'll get an error, but if we do them in the other order,
# the source file will get all the includes it needs before the header removes any of them.
source_files = []
header_files = []
for target_file in (_ for t in targets for _ in build_target_list(t, regex, ignores)):
if target_file.endswith('.cpp') or target_file.endswith('.ino'):
source_files.append(target_file)
else:
header_files.append(target_file)
# If there's an error processing any file, return an error code.
exit_code = 0 exit_code = 0
for src in (_ for t in targets for _ in build_target_list(t, regex)): for target_file in source_files + header_files:
if src in ignores: # Run IWYU and fix_headers:
logging.info("Skipping ignored file: %s", os.path.relpath(src)) if not run_iwyu(os.path.relpath(target_file), iwyu_cmd, fix_includes_cmd):
continue
if not run_iwyu(os.path.relpath(src), iwyu_cmd, fix_includes_cmd):
exit_code = 1 exit_code = 1
return exit_code return exit_code
# ============================================================================== # ==============================================================================
def build_target_list(path, src_regex): def build_target_list(target, regex, ignores):
"""Docstring""" """Build the list of target files, starting from a file or directory.
logging.debug("Searching target: %s", path)
Parameters:
target (str): The name of the file or directory to search
regex (Pattern): A compiled regular expression to match target file names
ignores (list): A list of (absolute) file names to be excluded from the target list
Returns:
targets (list): A list of (absolute) file names to be processed
Given a filename or directory (`path`), returns a list of target filenames to run IWYU
on. Target filenames will all be absolute, but `path` can be relative to the current
directory. All target filenames will match `regex`, and any items in `ignores` will be
excluded. If an element of the `ignores` list is a directory, all files and directories
beneath it will be excluded from the search.
"""
logging.debug("Searching target: %s", target)
# If the specified path is a filename, return it (as a list), regardless of # Convert all paths to absolute.
# whether or not it matches the regex. root = os.path.abspath(target)
if os.path.isfile(path):
return [path] # If `path` is a file, and it matches `regex`, add it to the target list.
if os.path.isfile(target) and regex.search(target) and root not in ignores:
return [root]
# If the specified path is not valid, just return an empty list. # If the specified path is not valid, just return an empty list.
if not os.path.isdir(path): if not os.path.isdir(target):
logging.error("Error: File not found: %s", path) logging.error("Error: File not found: %s", target)
return [] return []
# Start with an empty list.
targets = []
# The specified path is a directory, so we search recursively for files # The specified path is a directory, so we search recursively for files
# contained therein that match the specified regular expression. # contained therein that match the specified regular expression.
targets = [] for path, dirs, files in os.walk(root):
for root, dirs, files in os.walk(os.path.abspath(path)): logging.debug("Searching dir: %s", os.path.relpath(path))
logging.debug("Searching dir: %s", root)
# First, ignore all dotfiles (and directories). # First, ignore all dotfiles (and directories).
dotfiles = set(glob.glob('.*')) for x in (os.path.basename(_)
dirs = set(dirs) - dotfiles for _ in ignores + glob.glob(os.path.join(path, '.*'))
files = set(files) - dotfiles if os.path.dirname(_) == path):
if x in dirs:
logging.info("Skipping ignored dir: %s", os.path.join(path, x))
dirs.remove(x)
if x in files:
logging.info("Skipping ignored file: %s", os.path.join(path, x))
files.remove(x)
logging.debug("Files found: %s", ', '.join(files)) logging.debug("Files found: %s", ', '.join(files))
# Add all matching files to the list of source files to be formatted. # Add all matching files to the list of source files to be formatted.
for f in filter(src_regex.search, files): for f in (_ for _ in files if regex.search(_)):
logging.debug("Source file found: %s", f) t = os.path.join(path, f)
targets.append(os.path.join(root, f)) logging.debug("Source file found: %s", t)
targets.append(t)
return [os.path.abspath(_) for _ in targets] return targets
# ============================================================================== # ==============================================================================
def build_ignores_list(ignores_file_path): def build_ignores_list(ignores_file_path):
"""Build a list of files and dirs to exclude from processing by IWYU
Parameters:
ignores_file_path (str): The name of the file to read ignores globs from
Returns:
ignores_list (list): A list of (absolute) file names to exclude from processing
Reads `ignores_file` and expands each line as a glob, returning a list of all the target
file names to be excluded from processing. Each path in the file is treated as a relative
pathname (unless it is already absolute), relative to the directory in which `ignores_file`
is located.
"""
logging.debug("Searching for ignores file: %s", ignores_file_path) logging.debug("Searching for ignores file: %s", ignores_file_path)
# If the ignores file doesn't exist, return an empty list: # If the ignores file doesn't exist, return an empty list:
if not os.path.isfile(ignores_file_path): if not os.path.isfile(ignores_file_path):
logging.debug("Ignores file not found") logging.debug("Ignores file not found")
return [] return []
ignores_list = [] ignores_list = []
with open(ignores_file_path) as f: with open(ignores_file_path) as f:
for line in f.read().splitlines(): for line in f.read().splitlines():
logging.debug("Ignoring files like: %s", line) # Lines starting with `#` are treated as comments.
if line.startswith('#'): if line.startswith('#'):
continue continue
logging.debug("Ignoring files like: %s", line)
ignores_list += glob.glob(line, recursive=True) ignores_list += glob.glob(line, recursive=True)
# Get the dir of the ignores file so we can construct absolute pathnames.
ignores_file_dir = os.path.dirname(ignores_file_path) ignores_file_dir = os.path.dirname(ignores_file_path)
with cwd(ignores_file_dir): with cwd(ignores_file_dir):
ignores_list[:] = [os.path.abspath(_) for _ in ignores_list] ignores_list[:] = [os.path.abspath(_) for _ in ignores_list]
logging.debug("Ignores list:\n\t%s", "\n\t".join(ignores_list)) logging.debug("Ignores list:\n\t%s", "\n\t".join(ignores_list))
return ignores_list return ignores_list
# ------------------------------------------------------------------------------
from contextlib import contextmanager
@contextmanager
def cwd(path):
"""A simple function change directory, an automatically restore the previous working
directory when done, using `with cwd(temp_dir):`"""
old_wd = os.getcwd()
os.chdir(path)
try:
yield
finally:
os.chdir(old_wd)
# ============================================================================== # ==============================================================================
def run_iwyu(source_file, iwyu_cmd, fix_includes_cmd): def run_iwyu(source_file, iwyu_cmd, fix_includes_cmd):
"""Run `include-what-you-use` on <source_file>, an update that file's header includes by """Run IWYU and fix_includes on a single source file
Parameters:
source_file (str): The name of a file to run IWYU on
iwyu_cmd (list): The command name and options list for `include-what-you-use`
fix_includes_cmd (list): The command name and options list for `fix_headers.py`
Returns:
True on success, False if either IWYU or fix_headers returns an error code
Run `include-what-you-use` on <source_file>, an update that file's header includes by
sending the output to `fix_includes.py`. If either command returns an error code, return sending the output to `fix_includes.py`. If either command returns an error code, return
`False`, otherwise return `True`.""" `False`, otherwise return `True`.
logging.info("Processing file: %s", source_file) """
logging.info("Fixing headers in file: %s", source_file)
# Run IWYU on <source_file> # Run IWYU on <source_file>
iwyu_proc = subprocess.run(iwyu_cmd + [source_file], capture_output=True, check=False) iwyu_proc = subprocess.run(iwyu_cmd + [source_file], capture_output=True, check=False)

@ -132,6 +132,7 @@ Our style guide is based on the [Google C++ style guide][goog:c++-guide] which w
- [Vertical Whitespace](#vertical-whitespace) - [Vertical Whitespace](#vertical-whitespace)
- [Exceptions to the Rules](#exceptions-to-the-rules) - [Exceptions to the Rules](#exceptions-to-the-rules)
- [Existing Non-conformant Code](#existing-non-conformant-code) - [Existing Non-conformant Code](#existing-non-conformant-code)
- [Maintenance Tools](#maintenance-tools)
- [Parting Words](#parting-words) - [Parting Words](#parting-words)
## Background ## Background
@ -3155,6 +3156,40 @@ The coding conventions described above are mandatory. However, like all good rul
If you find yourself modifying code that was written to specifications other than those presented by this guide, you may have to diverge from these rules in order to stay consistent with the local conventions in that code. If you are in doubt about how to do this, ask the original author or the person currently responsible for the code. Remember that *consistency* includes local consistency, too. If you find yourself modifying code that was written to specifications other than those presented by this guide, you may have to diverge from these rules in order to stay consistent with the local conventions in that code. If you are in doubt about how to do this, ask the original author or the person currently responsible for the code. Remember that *consistency* includes local consistency, too.
## Maintenance Tools
Kaleidoscope uses some automated tools to enforce compliance with this code style guide. Primarily, we use `clang-format` to format source files, `cpplint` to check for potential problems, and `include-what-you-use` to update header includes. These are invoked using python scripts that supply all of the necessary command-line parameters to both utilities. For convenience, there are also some shell scripts and makefile targets that further simplify the process of running these utilities properly.
### Code Formatting
We use `clang-format`(version 12 or higher) to automatically format Kaleidoscope source files. There is a top-level `.clang-format` config file that contains the settings that best match the style described in this guide. However, there are some files in the repository that are, for one reason or another, exempt from formatting in this way, so we use a wrapper script, `format-code.py`, instead of invoking it directly.
`format-code.py` takes a list of target filenames, either as command-line parameters or from standard input (if reading from a pipe, with each line treated as a filename), and formats those files. If given a directory target, it will recursively search that directory for source files, and format all of the ones it finds.
By default, `format-code.py` will first check for unstaged changes in the Kaleidoscope git working tree, and exit before formatting source files if any are found. This is meant to make it easier for developers to see the changes made by the formatter in isolation. It can be given the `--force` option to skip this check.
It also has a `--check` option, which will cause `format-code.py` to check for unstaged git working tree changes after running `clang-format` on the target source files, and return an error code if there are any, allowing us to automatically verify that code submitted complies with the formatting rules.
The easiest way to invoke the formatter on the whole repository is by running `make format` at the top level of the Kaleidoscope repository.
For automated checking of PRs in a CI tool, there is also `make check-code-style`.
### Linting
We use a copy of `cpplint.py` (with a modification that allows us to set the config file) to check for potential problems in the code. This can be invoked by running `make cpplint` at the top level of the Kaleidoscope repository.
### Header Includes
We use `include-what-you-use` (version 18 or higher) to automatically manage header includes in Kaleidoscope source files. Because of the peculiarities of Kaleidoscope's build system, we use a wrapper script, `iwyu.py`, instead of invoking it directly.
`iwyu.py` takes a list of target filenames, either as command-line parameters or from standard input (if reading from a pipe, with each line treated as a filename), and makes changes to the header includes. If given a directory target, it will recursively search that directory for source files, and run `include-what-you-use` on all of the ones it finds.
A number of files can't be processed this way, and are enumerated in `.iwyu_ignore`. Files in the `testing` directory (for the test simulator) require different include path items, so cannot be combined in the same call to `iwyu.py`.
The easiest way to invoke `iwyu.py` is by running `make check-includes`, which will check all files that differ between the git working tree and the current master branch of the main Kaleidoscope repository, update their headers, and return a non-zero exit code if there were any errors processing the file(s) or any changes made.
For automated checking of header includes in a CI tool, there is also `make check-all-includes`, that checks the whole repository, not just the current branch's changes.
## Parting Words ## Parting Words
Use common sense and *BE CONSISTENT*. Use common sense and *BE CONSISTENT*.

@ -16,12 +16,11 @@
#include "testing/AbsoluteMouseReport.h" #include "testing/AbsoluteMouseReport.h"
#include "Kaleidoscope.h" #include <cstring> // for memcpy
#include "testing/fix-macros.h" #include <vector> // for vector
#include <cstring> #include "MouseButtons.h" // for MOUSE_LEFT, MOUSE_MIDDLE, MOUSE_NEXT, MOUSE_PREV, MOUS...
#include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "MouseButtons.h"
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint16_t, uint32_t, uint8_t, int8_t
#include <vector> #include <vector> // for vector
#include "DeviceAPIs/AbsoluteMouseAPI.h" #include "DeviceAPIs/AbsoluteMouseAPI.h" // for HID_MouseAbsoluteReport_Data_t
#include "HID-Settings.h" #include "HID-Settings.h" // for HID_REPORTID_MOUSE_ABSOLUTE
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,10 +16,10 @@
#include "testing/ConsumerControlReport.h" #include "testing/ConsumerControlReport.h"
#include "Kaleidoscope.h" #include <cstring> // for memcpy
#include "testing/fix-macros.h" #include <vector> // for vector
#include <cstring> #include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint32_t, uint16_t, uint8_t
#include <vector> #include <vector> // for vector
#include "HID-Settings.h" #include "HID-Settings.h" // for HID_REPORTID_CONSUMERCONTROL
#include "MultiReport/ConsumerControl.h" #include "MultiReport/ConsumerControl.h" // for HID_ConsumerControlReport_Data_t
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,6 +16,9 @@
#include "testing/ExpectedKeyboardReport.h" #include "testing/ExpectedKeyboardReport.h"
#include <cstdint> // for uint8_t, uint32_t
#include <set> // for set
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,10 +16,10 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint8_t, uint32_t
#include <vector> #include <set> // for set
#include <set>
#include <string> #include "testing/iostream.h" // for string
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,6 +16,9 @@
#include "testing/ExpectedMouseReport.h" #include "testing/ExpectedMouseReport.h"
#include "MultiReport/Mouse.h" // for (anonymous union)::(anonymous)
#include "testing/MouseReport.h" // for MouseReport::ReportData
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,12 +16,10 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for int8_t, uint32_t, uint8_t
#include <vector>
#include <set>
#include <string>
#include "MouseReport.h" #include "MouseReport.h" // for MouseReport, MouseReport::ReportData
#include "testing/iostream.h" // for string
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,12 +16,14 @@
#include "testing/HIDState.h" #include "testing/HIDState.h"
#include "HID-Settings.h" // IWYU pragma: no_include <__utility/move.h>
#include "testing/fix-macros.h" #include <utility> // IWYU pragma: keep
#include <vector> // for vector
#include "HID-Settings.h" // for HID_REPORTID_CONSUMERCONTROL, HID_REPORTID_GAMEPAD, HID_RE...
#include "testing/iostream.h" // for operator<<, char_traits, cout, ostream, basic_ostream
// TODO(epan): Add proper logging.
#include <iostream>
#define LOG(x) std::cout #define LOG(x) std::cout
namespace kaleidoscope { namespace kaleidoscope {

@ -16,15 +16,18 @@
#pragma once #pragma once
#include "testing/AbsoluteMouseReport.h" // IWYU pragma: no_include <__memory/unique_ptr.h>
#include "testing/ConsumerControlReport.h"
#include "testing/KeyboardReport.h" #include <stddef.h> // for size_t
#include "testing/MouseReport.h" #include <stdint.h> // for uint8_t
#include "testing/SystemControlReport.h" #include <memory> // IWYU pragma: keep
#include <vector> // for vector
// Out of order due to macro conflicts.
#include "testing/fix-macros.h" #include "testing/AbsoluteMouseReport.h" // for AbsoluteMouseReport
#include <memory> #include "testing/ConsumerControlReport.h" // for ConsumerControlReport
#include "testing/KeyboardReport.h" // for KeyboardReport
#include "testing/MouseReport.h" // for MouseReport
#include "testing/SystemControlReport.h" // for SystemControlReport
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,10 +16,11 @@
#include "testing/KeyboardReport.h" #include "testing/KeyboardReport.h"
#include "Kaleidoscope.h" #include <cstring> // for memcpy
#include "testing/fix-macros.h" #include <vector> // for vector<>::iterator, vector
#include <cstring> #include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "kaleidoscope/key_defs.h" // for HID_KEYBOARD_FIRST_MODIFIER, HID_LAST_KEY
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint8_t, uint32_t
#include <vector> #include <vector> // for vector
#include "HID-Settings.h" #include "HID-Settings.h" // for HID_REPORTID_NKRO_KEYBOARD
#include "MultiReport/Keyboard.h" #include "MultiReport/Keyboard.h" // for HID_KeyboardReport_Data_t
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,12 +16,9 @@
#include "testing/MouseReport.h" #include "testing/MouseReport.h"
#include "Kaleidoscope.h" #include <cstring> // for memcpy
#include "testing/fix-macros.h"
#include <cstring> #include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "MouseButtons.h"
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint8_t, int8_t, uint32_t
#include <vector>
#include "HID-Settings.h" #include "HID-Settings.h" // for HID_REPORTID_MOUSE
#include "MultiReport/Mouse.h" #include "MouseButtons.h" // for MOUSE_LEFT, MOUSE_MIDDLE, MOUSE_NEXT, MOUSE_PREV, MOUSE_R...
#include "MultiReport/Mouse.h" // for HID_MouseReport_Data_t
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,8 +16,10 @@
#include "testing/SimHarness.h" #include "testing/SimHarness.h"
#include "testing/fix-macros.h" #include <Arduino.h> // for millis
#include <iostream>
#include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "kaleidoscope/device/device.h" // for Device, VirtualProps::KeyScanner
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {
@ -31,7 +33,7 @@ void SimHarness::RunCycle() {
millis(); millis();
} }
} }
Kaleidoscope.loop(); kaleidoscope::Runtime.loop();
} }
void SimHarness::RunCycles(size_t n) { void SimHarness::RunCycles(size_t n) {
@ -39,20 +41,20 @@ void SimHarness::RunCycles(size_t n) {
} }
void SimHarness::RunForMillis(size_t t) { void SimHarness::RunForMillis(size_t t) {
auto start_time = Kaleidoscope.millisAtCycleStart(); auto start_time = kaleidoscope::Runtime.millisAtCycleStart();
while (Kaleidoscope.millisAtCycleStart() - start_time < t) { while (kaleidoscope::Runtime.millisAtCycleStart() - start_time < t) {
RunCycle(); RunCycle();
} }
} }
void SimHarness::Press(KeyAddr key_addr) { void SimHarness::Press(KeyAddr key_addr) {
Kaleidoscope.device().keyScanner().setKeystate( kaleidoscope::Runtime.device().keyScanner().setKeystate(
key_addr, key_addr,
kaleidoscope::Device::Props::KeyScanner::KeyState::Pressed); kaleidoscope::Device::Props::KeyScanner::KeyState::Pressed);
} }
void SimHarness::Release(KeyAddr key_addr) { void SimHarness::Release(KeyAddr key_addr) {
Kaleidoscope.device().keyScanner().setKeystate( kaleidoscope::Runtime.device().keyScanner().setKeystate(
key_addr, key_addr,
kaleidoscope::Device::Props::KeyScanner::KeyState::NotPressed); kaleidoscope::Device::Props::KeyScanner::KeyState::NotPressed);
} }

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstddef> #include <cstddef> // for size_t
#include <cstdint> #include <cstdint> // for uint8_t
#include "Kaleidoscope.h" #include "kaleidoscope/KeyAddr.h" // for KeyAddr
#include "testing/fix-macros.h" #include "testing/gtest.h" // IWYU pragma: keep
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,15 +16,11 @@
#pragma once #pragma once
#include <cstddef> // IWYU pragma: no_include <__memory/unique_ptr.h>
#include <cstdint>
#include <vector>
#include "testing/HIDState.h" #include <memory> // IWYU pragma: keep
// Out of order due to macro conflicts. #include "testing/HIDState.h" // for HIDState
#include "testing/fix-macros.h"
#include <memory>
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,10 +16,9 @@
#include "testing/SystemControlReport.h" #include "testing/SystemControlReport.h"
#include "Kaleidoscope.h" #include <cstring> // for memcpy
#include "testing/fix-macros.h"
#include <cstring> #include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,11 +16,11 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint> // for uint8_t, uint32_t
#include <vector> #include <vector> // for vector
#include "HID-Settings.h" #include "HID-Settings.h" // for HID_REPORTID_SYSTEMCONTROL
#include "MultiReport/SystemControl.h" #include "MultiReport/SystemControl.h" // for HID_SystemControlReport_Data_t
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,10 +16,19 @@
#include "testing/VirtualDeviceTest.h" #include "testing/VirtualDeviceTest.h"
#include "HIDReportObserver.h" // IWYU pragma: no_include <__algorithm/max.h>
#include "testing/HIDState.h" // IWYU pragma: no_include <__tree>
#include <bitset> #include <bitset> // for bitset
#include <vector> // for vector
#include "HIDReportObserver.h" // for HIDReportObserver
#include "kaleidoscope/Runtime.h" // for Runtime, Runtime_
#include "testing/HIDState.h" // for HIDState, HIDStateBuilder
#include "testing/KeyboardReport.h" // for KeyboardReport
#include "testing/MouseReport.h" // for MouseReport
#include "testing/gtest.h" // for Message, TestPartResult, EXPECT_EQ, ElementsAreArray
#include "testing/iostream.h" // for operator<<, basic_ostream, char_traits, string, cerr
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -16,18 +16,24 @@
#pragma once #pragma once
#include <cstddef> // IWYU pragma: no_include <__memory/unique_ptr.h>
#include "testing/ExpectedKeyboardReport.h" #include <cstddef> // for size_t
#include "testing/ExpectedMouseReport.h" #include <cstdint> // for uint32_t, int8_t, uint8_t
#include "testing/SimHarness.h" #include <initializer_list> // for initializer_list
#include "testing/State.h" #include <memory> // IWYU pragma: keep
#include <set> // for set
// Out of order due to macro conflicts. #include <vector> // for vector
#include "testing/fix-macros.h"
#include "gmock/gmock.h" #include "kaleidoscope/KeyAddr.h" // for KeyAddr
#include "gtest/gtest.h" #include "kaleidoscope/key_defs.h" // for Key
#include <memory> #include "testing/ExpectedKeyboardReport.h" // for ExpectedKeyboardReport
#include "testing/ExpectedMouseReport.h" // for ExpectedMouseReport
#include "testing/HIDState.h" // for HIDState
#include "testing/SimHarness.h" // for SimHarness
#include "testing/State.h" // for State
#include "testing/gtest.h" // for Test
#include "testing/iostream.h" // for string
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -0,0 +1,36 @@
/* -*- mode: c++ -*-
* Copyright (C) 2022 Keyboardio, Inc.
*
* 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, version 3.
*
* 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/>.
*/
#pragma once
#undef TEST
// In order to include gtest files, we need to undefine some macros that
// Arduino.h unwisely exports. Rahter than including "gtest/gtest.h" (et al)
// directly, any simulator code should instead include "testing/gtest.h".
#undef min
#undef max
// The headers listed here other than "gtest/gtest.h" and "gmock/gmock.h" are
// only included to prevent IWYU from inserting them directly into simulator
// source files. This seems mildly preferable to modifying the gtest files
// themselves.
#include "gmock/gmock-matchers.h" // IWYU pragma: export
#include "gmock/gmock.h" // IWYU pragma: export
#include "gtest/gtest-message.h" // IWYU pragma: export
#include "gtest/gtest-test-part.h" // IWYU pragma: export
#include "gtest/gtest.h" // IWYU pragma: export
#include "gtest/gtest_pred_impl.h" // IWYU pragma: export

@ -1,5 +1,5 @@
/* -*- mode: c++ -*- /* -*- mode: c++ -*-
* Copyright (C) 2020 Eric Paniagua (epaniagua@google.com) * Copyright (C) 2022 Keyboardio, Inc.
* *
* This program is free software: you can redistribute it and/or modify it under * 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 * the terms of the GNU General Public License as published by the Free Software
@ -14,12 +14,12 @@
* this program. If not, see <http://www.gnu.org/licenses/>. * this program. If not, see <http://www.gnu.org/licenses/>.
*/ */
// No `#pragma once` since these undefs need to have every time a Kaleidoscope/ #pragma once
// Arduino header is included before non-Kaleidoscope/Arduino header. The undefs
// are needed, due to naming conflicts between Arduino and Googletest.
#undef min // In order to include certain standard library files, we need to undefine some
// macros that Arduino.h unwisely exports. Rahter than including <iostream>
// directly, any simulator code should instead include "testing/iostream.h".
#undef max #undef max
#undef T #undef min
#undef U
#undef TEST #include <iostream> // IWYU pragma: export

@ -16,13 +16,12 @@
#pragma once #pragma once
#include "kaleidoscope/key_defs.h" // For some reason, the `MATCHER_P` macro confuses IWYU into trying to include
#include "testing/SystemControlReport.h" // this file in itself, so we need to block it with the following:
// IWYU pragma: no_include "testing/matchers.h"
// Out of order because `fix-macros.h` clears the preprocessor environment for #include "kaleidoscope/key_defs.h" // for Key
// gtest and gmock. #include "testing/gtest.h" // for GMOCK_PP_INTERNAL_FOR_EACH_IMPL_1, GMOCK_PP_INTERNAL_...
#include "testing/fix-macros.h"
#include "gmock/gmock.h"
namespace kaleidoscope { namespace kaleidoscope {
namespace testing { namespace testing {

@ -18,17 +18,16 @@
#pragma once #pragma once
#include "kaleidoscope/key_defs/keyboard.h" #include <Kaleidoscope.h> // IWYU pragma: keep
#include "Kaleidoscope.h"
// Out of order because `fix-macros.h` clears the preprocessor environment for // Kaleidoscope.h includes Arduino, which unwisely defines `min` and `max` as
// gtest and gmock. // preprocessor macros. We need to undefine these macros before including any
#include "testing/fix-macros.h" // files from the standard library.
#include "gmock/gmock.h" #undef min
#include "gtest/gtest.h" #undef max
#include "testing/matchers.h" #include "testing/VirtualDeviceTest.h" // IWYU pragma: keep
#include "testing/VirtualDeviceTest.h" #include "testing/matchers.h" // IWYU pragma: keep
#define SETUP_GOOGLETEST() \ #define SETUP_GOOGLETEST() \
void executeTestFunction() { \ void executeTestFunction() { \

@ -15,7 +15,6 @@
*/ */
#include "Kaleidoscope.h" #include "Kaleidoscope.h"
#include "testing/fix-macros.h"
#include "testing/setup-googletest.h" #include "testing/setup-googletest.h"

Loading…
Cancel
Save