diff --git a/.cmake-format.yaml b/.cmake-format.yaml new file mode 100644 index 0000000000..1193ec79aa --- /dev/null +++ b/.cmake-format.yaml @@ -0,0 +1,289 @@ +_help_parse: Options affecting listfile parsing +parse: + _help_additional_commands: + - Specify structure for custom cmake functions + additional_commands: + iox_add_executable: + pargs: + nargs: '*' + flags: + - USE_C_LANGUAGE + - PLACE_IN_BUILD_ROOT + kwargs: + TARGET: 1 + STACK_SIZE: 1 + FILES: + + LIBS: + + INCLUDE_DIRECTORIES: + + LIBS_QNX: + + LIBS_LINUX: + + LIBS_UNIX: + + LIBS_WIN32: + + LIBS_APPLE: + + BUILD_INTERFACE: + + INSTALL_INTERFACE: + + iox_add_library: + pargs: + nargs: '*' + flags: + - USE_C_LANGUAGE + - NO_EXPORT + - NO_PACKAGE_SETUP + - NO_FIND_PACKAGE_SUPPORT + - STATIC + kwargs: + TARGET: 1 + NAMESPACE: 1 + PROJECT_PREFIX: 1 + FILES: + + PUBLIC_LIBS: + + PRIVATE_LIBS: + + BUILD_INTERFACE: + + INSTALL_INTERFACE: + + ADDITIONAL_EXPORT_TARGETS: + + PUBLIC_INCLUDES: + + PRIVATE_INCLUDES: + + PUBLIC_LIBS_LINUX: + + PRIVATE_LIBS_LINUX: + + PUBLIC_LIBS_QNX: + + PRIVATE_LIBS_QNX: + + PUBLIC_LIBS_UNIX: + + PRIVATE_LIBS_UNIX: + + PUBLIC_LIBS_WIN32: + + PRIVATE_LIBS_WIN32: + + PUBLIC_LIBS_APPLE: + + PRIVATE_LIBS_APPLE: + + EXPORT_INCLUDE_DIRS: + + _help_override_spec: + - Override configurations per-command where available + override_spec: {} + _help_vartags: + - Specify variable tags. + vartags: [] + _help_proptags: + - Specify property tags. + proptags: [] +_help_format: Options affecting formatting. +format: + _help_disable: + - Disable formatting entirely, making cmake-format a no-op + disable: false + _help_line_width: + - How wide to allow formatted cmake files + line_width: 80 + _help_tab_size: + - How many spaces to tab for indent + tab_size: 4 + _help_use_tabchars: + - If true, lines are indented using tab characters (utf-8 + - 0x09) instead of space characters (utf-8 0x20). + - In cases where the layout would require a fractional tab + - character, the behavior of the fractional indentation is + - governed by + use_tabchars: false + _help_fractional_tab_policy: + - If is True, then the value of this variable + - indicates how fractional indentions are handled during + - whitespace replacement. If set to 'use-space', fractional + - indentation is left as spaces (utf-8 0x20). If set to + - '`round-up` fractional indentation is replaced with a single' + - tab character (utf-8 0x09) effectively shifting the column + - to the next tabstop + fractional_tab_policy: use-space + _help_max_subgroups_hwrap: + - If an argument group contains more than this many sub-groups + - (parg or kwarg groups) then force it to a vertical layout. + max_subgroups_hwrap: 2 + _help_max_pargs_hwrap: + - If a positional argument group contains more than this many + - arguments, then force it to a vertical layout. + max_pargs_hwrap: 6 + _help_max_rows_cmdline: + - If a cmdline positional group consumes more than this many + - lines without nesting, then invalidate the layout (and nest) + max_rows_cmdline: 2 + _help_separate_ctrl_name_with_space: + - If true, separate flow control names from their parentheses + - with a space + separate_ctrl_name_with_space: false + _help_separate_fn_name_with_space: + - If true, separate function names from parentheses with a + - space + separate_fn_name_with_space: false + _help_dangle_parens: + - If a statement is wrapped to more than one line, than dangle + - the closing parenthesis on its own line. + dangle_parens: true + _help_dangle_align: + - If the trailing parenthesis must be 'dangled' on its on + - 'line, then align it to this reference: `prefix`: the start' + - 'of the statement, `prefix-indent`: the start of the' + - 'statement, plus one indentation level, `child`: align to' + - the column of the arguments + dangle_align: prefix + _help_min_prefix_chars: + - If the statement spelling length (including space and + - parenthesis) is smaller than this amount, then force reject + - nested layouts. + min_prefix_chars: 4 + _help_max_prefix_chars: + - If the statement spelling length (including space and + - parenthesis) is larger than the tab width by more than this + - amount, then force reject un-nested layouts. + max_prefix_chars: 10 + _help_max_lines_hwrap: + - If a candidate layout is wrapped horizontally but it exceeds + - this many lines, then reject the layout. + max_lines_hwrap: 2 + _help_line_ending: + - What style line endings to use in the output. + line_ending: unix + _help_command_case: + - Format command names consistently as 'lower' or 'upper' case + command_case: canonical + _help_keyword_case: + - Format keywords consistently as 'lower' or 'upper' case + keyword_case: unchanged + _help_always_wrap: + - A list of command names which should always be wrapped + always_wrap: [] + _help_enable_sort: + - If true, the argument lists which are known to be sortable + - will be sorted lexicographicall + enable_sort: true + _help_autosort: + - If true, the parsers may infer whether or not an argument + - list is sortable (without annotation). + autosort: false + _help_require_valid_layout: + - By default, if cmake-format cannot successfully fit + - everything into the desired linewidth it will apply the + - last, most agressive attempt that it made. If this flag is + - True, however, cmake-format will print error, exit with non- + - zero status code, and write-out nothing + require_valid_layout: false + _help_layout_passes: + - A dictionary mapping layout nodes to a list of wrap + - decisions. See the documentation for more information. + layout_passes: {} +_help_markup: Options affecting comment reflow and formatting. +markup: + _help_bullet_char: + - What character to use for bulleted lists + bullet_char: '*' + _help_enum_char: + - What character to use as punctuation after numerals in an + - enumerated list + enum_char: . + _help_first_comment_is_literal: + - If comment markup is enabled, don't reflow the first comment + - block in each listfile. Use this to preserve formatting of + - your copyright/license statements. + first_comment_is_literal: false + _help_literal_comment_pattern: + - If comment markup is enabled, don't reflow any comment block + - which matches this (regex) pattern. Default is `None` + - (disabled). + literal_comment_pattern: null + _help_fence_pattern: + - Regular expression to match preformat fences in comments + - default= ``r'^\s*([`~]{3}[`~]*)(.*)$'`` + fence_pattern: ^\s*([`~]{3}[`~]*)(.*)$ + _help_ruler_pattern: + - Regular expression to match rulers in comments default= + - '``r''^\s*[^\w\s]{3}.*[^\w\s]{3}$''``' + ruler_pattern: ^\s*[^\w\s]{3}.*[^\w\s]{3}$ + _help_explicit_trailing_pattern: + - If a comment line matches starts with this pattern then it + - is explicitly a trailing comment for the preceeding + - argument. Default is '#<' + explicit_trailing_pattern: '#<' + _help_hashruler_min_length: + - If a comment line starts with at least this many consecutive + - hash characters, then don't lstrip() them off. This allows + - for lazy hash rulers where the first hash char is not + - separated by space + hashruler_min_length: 10 + _help_canonicalize_hashrulers: + - If true, then insert a space between the first hash char and + - remaining hash chars in a hash ruler, and normalize its + - length to fill the column + canonicalize_hashrulers: true + _help_enable_markup: + - enable comment markup parsing and reflow + enable_markup: true +_help_lint: Options affecting the linter +lint: + _help_disabled_codes: + - a list of lint codes to disable + disabled_codes: [] + _help_function_pattern: + - regular expression pattern describing valid function names + function_pattern: '[0-9a-z_]+' + _help_macro_pattern: + - regular expression pattern describing valid macro names + macro_pattern: '[0-9A-Z_]+' + _help_global_var_pattern: + - regular expression pattern describing valid names for + - variables with global (cache) scope + global_var_pattern: '[A-Z][0-9A-Z_]+' + _help_internal_var_pattern: + - regular expression pattern describing valid names for + - variables with global scope (but internal semantic) + internal_var_pattern: _[A-Z][0-9A-Z_]+ + _help_local_var_pattern: + - regular expression pattern describing valid names for + - variables with local scope + local_var_pattern: '[a-z][a-z0-9_]+' + _help_private_var_pattern: + - regular expression pattern describing valid names for + - privatedirectory variables + private_var_pattern: _[0-9a-z_]+ + _help_public_var_pattern: + - regular expression pattern describing valid names for public + - directory variables + public_var_pattern: '[A-Z][0-9A-Z_]+' + _help_argument_var_pattern: + - regular expression pattern describing valid names for + - function/macro arguments and loop variables. + argument_var_pattern: '[a-z][a-z0-9_]+' + _help_keyword_pattern: + - regular expression pattern describing valid names for + - keywords used in functions or macros + keyword_pattern: '[A-Z][0-9A-Z_]+' + _help_max_conditionals_custom_parser: + - In the heuristic for C0201, how many conditionals to match + - within a loop in before considering the loop a parser. + max_conditionals_custom_parser: 2 + _help_min_statement_spacing: + - Require at least this many newlines between statements + min_statement_spacing: 1 + _help_max_statement_spacing: + - Require no more than this many newlines between statements + max_statement_spacing: 2 + max_returns: 6 + max_branches: 12 + max_arguments: 5 + max_localvars: 15 + max_statements: 50 +_help_encode: Options affecting file encoding +encode: + _help_emit_byteorder_mark: + - If true, emit the unicode byte-order mark (BOM) at the start + - of the file + emit_byteorder_mark: false + _help_input_encoding: + - Specify the encoding of the input file. Defaults to utf-8 + input_encoding: utf-8 + _help_output_encoding: + - Specify the encoding of the output file. Defaults to utf-8. + - Note that cmake only claims to support utf-8 so be careful + - when using anything else + output_encoding: utf-8 +_help_misc: Miscellaneous configurations options. +misc: + _help_per_command: + - A dictionary containing any per-command configuration + - overrides. Currently only `command_case` is supported. + per_command: {} + diff --git a/iceoryx_hoofs/CMakeLists.txt b/iceoryx_hoofs/CMakeLists.txt index 0dd446b1ac..91abc868b5 100644 --- a/iceoryx_hoofs/CMakeLists.txt +++ b/iceoryx_hoofs/CMakeLists.txt @@ -1,3 +1,4 @@ +# cmake-format: off # Copyright (c) 2019 - 2021 by Robert Bosch GmbH. All rights reserved. # Copyright (c) 2020 - 2022 by Apex.AI Inc. All rights reserved. # @@ -14,6 +15,7 @@ # limitations under the License. # # SPDX-License-Identifier: Apache-2.0 +# cmake-format: on cmake_minimum_required(VERSION 3.16) @@ -36,90 +38,90 @@ endif() set(PREFIX iceoryx/v${CMAKE_PROJECT_VERSION}) -# +# cmake-format: off ########## build iceoryx hoofs lib ########## -# +# cmake-format: on + iox_add_library( - TARGET iceoryx_hoofs - NAMESPACE iceoryx_hoofs - PROJECT_PREFIX ${PREFIX} - PUBLIC_LIBS iceoryx_platform::iceoryx_platform - PRIVATE_LIBS ${ICEORYX_SANITIZER_FLAGS} - PRIVATE_LIBS_LINUX acl atomic ${CODE_COVERAGE_LIBS} - BUILD_INTERFACE ${PROJECT_SOURCE_DIR}/include - ${PROJECT_SOURCE_DIR}/legacy/include - ${PROJECT_SOURCE_DIR}/memory/include - ${PROJECT_SOURCE_DIR}/container/include - ${PROJECT_SOURCE_DIR}/vocabulary/include - ${PROJECT_SOURCE_DIR}/utility/include - ${PROJECT_SOURCE_DIR}/primitives/include - ${PROJECT_SOURCE_DIR}/design/include - ${PROJECT_SOURCE_DIR}/time/include - ${PROJECT_SOURCE_DIR}/posix/time/include - ${PROJECT_SOURCE_DIR}/buffer/include - ${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include - INSTALL_INTERFACE include/${PREFIX} - EXPORT_INCLUDE_DIRS include/ - legacy/include/ - memory/include/ - container/include/ - vocabulary/include/ - utility/include/ - primitives/include/ - design/include/ - time/include/ - posix/time/include/ - buffer/include/ - FILES - source/concurrent/loffli.cpp - posix/time/source/adaptive_wait.cpp - posix/time/source/deadline_timer.cpp - source/cxx/filesystem.cpp - source/cxx/requires.cpp - source/cxx/type_traits.cpp - source/cxx/unique_id.cpp - source/error_handling/error_handler.cpp - source/error_handling/error_handling.cpp - source/log/building_blocks/console_logger.cpp - source/log/building_blocks/logger.cpp - source/posix_wrapper/access_control.cpp - source/posix_wrapper/file_lock.cpp - source/posix_wrapper/mutex.cpp - source/posix_wrapper/named_semaphore.cpp - source/posix_wrapper/posix_access_rights.cpp - source/posix_wrapper/scheduler.cpp - source/posix_wrapper/semaphore_interface.cpp - source/posix_wrapper/shared_memory_object.cpp - source/posix_wrapper/shared_memory_object/memory_map.cpp - source/posix_wrapper/shared_memory_object/shared_memory.cpp - source/posix_wrapper/signal_handler.cpp - source/posix_wrapper/system_configuration.cpp - source/posix_wrapper/thread.cpp - source/posix_wrapper/types.cpp - source/posix_wrapper/unnamed_semaphore.cpp - source/posix_wrapper/unix_domain_socket.cpp - source/memory/relative_pointer_data.cpp - source/units/duration.cpp - memory/source/bump_allocator.cpp - memory/source/memory.cpp - design/source/functional_interface.cpp + TARGET iceoryx_hoofs + NAMESPACE iceoryx_hoofs + PROJECT_PREFIX ${PREFIX} + PUBLIC_LIBS iceoryx_platform::iceoryx_platform + PRIVATE_LIBS ${ICEORYX_SANITIZER_FLAGS} + PRIVATE_LIBS_LINUX acl atomic ${CODE_COVERAGE_LIBS} + BUILD_INTERFACE + ${PROJECT_SOURCE_DIR}/include + ${PROJECT_SOURCE_DIR}/legacy/include + ${PROJECT_SOURCE_DIR}/memory/include + ${PROJECT_SOURCE_DIR}/container/include + ${PROJECT_SOURCE_DIR}/vocabulary/include + ${PROJECT_SOURCE_DIR}/utility/include + ${PROJECT_SOURCE_DIR}/primitives/include + ${PROJECT_SOURCE_DIR}/design/include + ${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include + INSTALL_INTERFACE include/${PREFIX} + EXPORT_INCLUDE_DIRS + include/ + legacy/include/ + memory/include/ + container/include/ + vocabulary/include/ + utility/include/ + primitives/include/ + design/include/ + FILES source/concurrent/loffli.cpp + source/cxx/adaptive_wait.cpp + source/cxx/deadline_timer.cpp + source/cxx/filesystem.cpp + source/cxx/requires.cpp + source/cxx/type_traits.cpp + source/cxx/unique_id.cpp + source/error_handling/error_handler.cpp + source/error_handling/error_handling.cpp + source/log/building_blocks/console_logger.cpp + source/log/building_blocks/logger.cpp + source/posix_wrapper/access_control.cpp + source/posix_wrapper/file_lock.cpp + source/posix_wrapper/mutex.cpp + source/posix_wrapper/named_semaphore.cpp + source/posix_wrapper/posix_access_rights.cpp + source/posix_wrapper/scheduler.cpp + source/posix_wrapper/semaphore_interface.cpp + source/posix_wrapper/shared_memory_object.cpp + source/posix_wrapper/shared_memory_object/memory_map.cpp + source/posix_wrapper/shared_memory_object/shared_memory.cpp + source/posix_wrapper/signal_handler.cpp + source/posix_wrapper/system_configuration.cpp + source/posix_wrapper/thread.cpp + source/posix_wrapper/types.cpp + source/posix_wrapper/unnamed_semaphore.cpp + source/posix_wrapper/unix_domain_socket.cpp + source/memory/relative_pointer_data.cpp + source/units/duration.cpp + memory/source/bump_allocator.cpp + memory/source/memory.cpp + design/source/functional_interface.cpp ) -configure_file("${CMAKE_CURRENT_SOURCE_DIR}/cmake/iceoryx_hoofs_deployment.hpp.in" - "${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include/iceoryx_hoofs/iceoryx_hoofs_deployment.hpp" @ONLY) +configure_file( + "${CMAKE_CURRENT_SOURCE_DIR}/cmake/iceoryx_hoofs_deployment.hpp.in" + "${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include/iceoryx_hoofs/iceoryx_hoofs_deployment.hpp" + @ONLY +) install( - FILES ${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include/${PROJECT_NAME}/iceoryx_hoofs_deployment.hpp + FILES + ${CMAKE_BINARY_DIR}/generated/iceoryx_hoofs/include/${PROJECT_NAME}/iceoryx_hoofs_deployment.hpp DESTINATION include/${PREFIX}/${PROJECT_NAME}/ COMPONENT dev ) -# +# cmake-format: off ########## hoofs testing ########## -# +# cmake-format: on -# Finding gtest and adding the subdirectories is split to support the use case of -# building the testing lib without the tests by providing gtest externally +# Finding gtest and adding the subdirectories is split to support the use case +# of building the testing lib without the tests by providing gtest externally if(NOT GTest_FOUND AND BUILD_TEST) find_package(GTest CONFIG REQUIRED) endif() @@ -132,9 +134,6 @@ if(GTest_FOUND) endif() endif() -install( - FILES - cmake/IceoryxPlatform.cmake - cmake/IceoryxVersion.cmake - DESTINATION ${DESTINATION_CONFIGDIR} +install(FILES cmake/IceoryxPlatform.cmake cmake/IceoryxVersion.cmake + DESTINATION ${DESTINATION_CONFIGDIR} ) diff --git a/tools/git-hooks/pre-commit b/tools/git-hooks/pre-commit index f5ffde34a9..851617be63 100755 --- a/tools/git-hooks/pre-commit +++ b/tools/git-hooks/pre-commit @@ -50,6 +50,18 @@ if ! tools/scripts/clang_format.sh hook; then exit 1 fi +## cmake-lint +if ! tools/scripts/cmake_lint.sh hook; then + echo "Error while executing cmake-lint" + exit 1 +fi + +## cmake-format +if ! tools/scripts/cmake_format.sh hook; then + echo "Error while executing cmake-format" + exit 1 +fi + ## check for test IDs if ! tools/scripts/check_test_ids.sh; then echo "Error checking test IDs" diff --git a/tools/scripts/cmake_format.sh b/tools/scripts/cmake_format.sh new file mode 100755 index 0000000000..840a7bcc1f --- /dev/null +++ b/tools/scripts/cmake_format.sh @@ -0,0 +1,68 @@ +#!/bin/bash + +# Copyright (c) 2023 by Apex.AI Inc. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# SPDX-License-Identifier: Apache-2.0 + +set -e + +SCOPE=${1:-full} # Can be either `full` for all files or `hook` for formatting with git hooks + + +fail() { + printf "\033[1;31merror: %s: %s\033[0m\n" ${FUNCNAME[1]} "${1:-"Unknown error"}" + exit 1 +} + +hash git || fail "git not found" + +# Check if we have at least a specific cmake-format version installed +CMAKE_FORMAT_VERSION="0.6.13" +CMAKE_FORMAT_CMD="cmake-format" +if ! command -v $CMAKE_FORMAT_CMD &> /dev/null +then + CMAKE_FORMAT_MAJOR_VERSION=$(cmake-format --version | sed -rn 's/.*([0-9][0-9])\.[0-9].*/\1/p') + if [[ $CMAKE_FORMAT_MAJOR_VERSION -lt "$CMAKE_FORMAT_VERSION" ]]; then + echo "Warning: cmake-format version $CMAKE_FORMAT_VERSION or higher is not installed." + echo "Code will not be formatted." + exit 0 + else + CMAKE_FORMAT_CMD="cmake-format" + fi +fi + +cd "$(git rev-parse --show-toplevel)" + +if [[ "$SCOPE" == "hook"* ]]; then + cmake_files=$(git diff --cached --name-only --diff-filter=ACMRT | grep -E "CMakeLists.txt|\.cmake" | cat) + if [[ -n ${cmake_files} ]]; then + + echo "Running cmake-format on the following file(s):" + echo "--------------------------------------------" + echo "${cmake_files}" | sed 's/^/\ -\ /g' + echo + + for file in $cmake_files ; do + if [ -f "$file" ]; then + $CMAKE_FORMAT_CMD -i -l error "${file}" + git add "${file}" + fi + done + fi +elif [[ "$SCOPE" == "full"* ]]; then + git ls-files | grep -E "CMakeLists.txt|\.cmake" | xargs cmake-format -i -l error +elif [[ "$SCOPE" == "check"* ]]; then + git ls-files | grep -E "CMakeLists.txt|\.cmake" | xargs cmake-format -i -l error +fi diff --git a/tools/scripts/cmake_lint.sh b/tools/scripts/cmake_lint.sh new file mode 100755 index 0000000000..4ccdca26ae --- /dev/null +++ b/tools/scripts/cmake_lint.sh @@ -0,0 +1,68 @@ +#!/bin/bash + +# Copyright (c) 2023 by Apex.AI Inc. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +# SPDX-Licensedentifier: Apache-2.0 + +set -e + +SCOPE=${1:-full} # Can be either `full` for all files or `hook` for lintting with git hooks + + +fail() { + printf "\033[1;31merror: %s: %s\033[0m\n" ${FUNCNAME[1]} "${1:-"Unknown error"}" + exit 1 +} + +hash git || fail "git not found" + +# Check if we have at least a specific cmake-lint version installed +CMAKE_LINT_VERSION="0.6.13" +CMAKE_LINT_CMD="cmake-lint" +if ! command -v $CMAKE_LINT_CMD &> /dev/null +then + CMAKE_LINT_MAJOR_VERSION=$(cmake-lint --version | sed -rn 's/.*([0-9][0-9])\.[0-9].*/\1/p') + if [[ $CMAKE_LINT_MAJOR_VERSION -lt "$CMAKE_LINT_VERSION" ]]; then + echo "Warning: cmake-lint version $CMAKE_LINT_VERSION or higher is not installed." + echo "Code will not be lintted." + exit 0 + else + CMAKE_LINT_CMD="cmake-lint" + fi +fi + +cd "$(git rev-parse --show-toplevel)" + +if [[ "$SCOPE" == "hook"* ]]; then + cmake_files=$(git diff --cached --name-only --diff-filter=ACMRT | grep -E "CMakeLists.txt|\.cmake" | cat) + if [[ -n ${cmake_files} ]]; then + + echo "Running cmake-lint on the following file(s):" + echo "--------------------------------------------" + echo "${cmake_files}" | sed 's/^/\ -\ /g' + echo + + for file in $cmake_files ; do + if [ -f "$file" ]; then + $CMAKE_LINT_CMD -l error "${file}" + git add "${file}" + fi + done + fi +elif [[ "$SCOPE" == "full"* ]]; then + git ls-files | grep -E "CMakeLists.txt|\.cmake" | xargs cmake-lint -l error +elif [[ "$SCOPE" == "check"* ]]; then + git ls-files | grep -E "CMakeLists.txt|\.cmake" | xargs cmake-lint -l error +fi