2019-08-28 17:20:42 +00:00
|
|
|
#!/bin/sh -e
|
2019-08-23 16:43:29 +00:00
|
|
|
|
|
|
|
# Tool to bundle multiple C/C++ source files, inlining any includes.
|
2019-08-25 20:49:01 +00:00
|
|
|
#
|
2019-08-28 17:20:42 +00:00
|
|
|
# Note: this POSIX-compliant script is many times slower than the original bash
|
|
|
|
# implementation (due to the grep calls) but it runs and works everywhere.
|
|
|
|
#
|
2020-04-03 17:07:46 +00:00
|
|
|
# TODO: ROOTS, FOUND, etc., as arrays (since they fail on paths with spaces)
|
|
|
|
# TODO: revert to Bash-only regex (the grep ones being too slow)
|
2019-09-02 16:02:50 +00:00
|
|
|
#
|
2020-04-03 18:50:54 +00:00
|
|
|
# Author: Carl Woffenden, Numfum GmbH (this script is released under a CC0 license/Public Domain)
|
2019-08-23 16:43:29 +00:00
|
|
|
|
|
|
|
# Common file roots
|
2020-04-03 17:07:46 +00:00
|
|
|
ROOTS="."
|
|
|
|
|
|
|
|
# -x option excluded includes
|
|
|
|
XINCS=""
|
|
|
|
|
|
|
|
# -k option includes to keep as include directives
|
|
|
|
KINCS=""
|
2019-08-23 16:43:29 +00:00
|
|
|
|
|
|
|
# Files previously visited
|
|
|
|
FOUND=""
|
|
|
|
|
2019-08-25 20:49:01 +00:00
|
|
|
# Optional destination file (empty string to write to stdout)
|
|
|
|
DESTN=""
|
|
|
|
|
2020-04-03 17:07:46 +00:00
|
|
|
# Whether the "#pragma once" directives should be written to the output
|
|
|
|
PONCE=0
|
|
|
|
|
2019-08-23 16:43:29 +00:00
|
|
|
# Prints the script usage then exits
|
2019-08-28 17:20:42 +00:00
|
|
|
usage() {
|
2020-04-03 17:07:46 +00:00
|
|
|
echo "Usage: $0 [-r <path>] [-x <header>] [-k <header>] [-o <outfile>] infile"
|
|
|
|
echo " -r file root search path"
|
|
|
|
echo " -x file to completely exclude from inlining"
|
|
|
|
echo " -k file to exclude from inlining but keep the include directive"
|
|
|
|
echo " -p keep any '#pragma once' directives (removed by default)"
|
2019-08-25 20:49:01 +00:00
|
|
|
echo " -o output file (otherwise stdout)"
|
|
|
|
echo "Example: $0 -r ../my/path - r ../other/path -o out.c in.c"
|
2019-08-23 16:43:29 +00:00
|
|
|
exit 1
|
|
|
|
}
|
|
|
|
|
2019-09-02 16:02:50 +00:00
|
|
|
# Tests that the grep implementation works as expected (older OSX grep fails)
|
|
|
|
test_grep() {
|
2020-04-03 17:07:46 +00:00
|
|
|
if ! echo '#include "foo"' | grep -Eq '^\s*#\s*include\s*".+"'; then
|
|
|
|
echo "Aborting: the grep implementation fails to parse include lines"
|
|
|
|
exit 1
|
|
|
|
fi
|
2019-09-02 16:02:50 +00:00
|
|
|
}
|
|
|
|
|
2019-08-28 17:20:42 +00:00
|
|
|
# Tests if list $1 has item $2 (returning zero on a match)
|
|
|
|
list_has_item() {
|
|
|
|
if echo "$1" | grep -Eq "(^|\s*)$2(\$|\s*)"; then
|
2019-08-23 16:43:29 +00:00
|
|
|
return 0
|
2019-08-28 17:20:42 +00:00
|
|
|
else
|
|
|
|
return 1
|
2019-08-23 16:43:29 +00:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2019-08-25 20:49:01 +00:00
|
|
|
# Adds a new line with the supplied arguments to $DESTN (or stdout)
|
2019-08-28 17:20:42 +00:00
|
|
|
write_line() {
|
2019-08-25 20:49:01 +00:00
|
|
|
if [ -n "$DESTN" ]; then
|
2019-08-28 17:20:42 +00:00
|
|
|
printf '%s\n' "$@" >> "$DESTN"
|
2019-08-25 20:49:01 +00:00
|
|
|
else
|
2019-08-28 17:20:42 +00:00
|
|
|
printf '%s\n' "$@"
|
2019-08-25 20:49:01 +00:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2019-08-23 16:43:29 +00:00
|
|
|
# Adds the contents of $1 with any of its includes inlined
|
2019-08-28 17:20:42 +00:00
|
|
|
add_file() {
|
2019-08-23 16:43:29 +00:00
|
|
|
# Match the path
|
|
|
|
local file=
|
2020-04-03 17:07:46 +00:00
|
|
|
for root in $ROOTS; do
|
|
|
|
if [ -f "$root/$1" ]; then
|
|
|
|
file="$root/$1"
|
|
|
|
fi
|
|
|
|
done
|
2019-08-25 20:49:01 +00:00
|
|
|
if [ -n "$file" ]; then
|
2020-04-03 17:07:46 +00:00
|
|
|
if [ -n "$DESTN" ]; then
|
|
|
|
# Log but only if not writing to stdout
|
|
|
|
echo "Processing: $file"
|
|
|
|
fi
|
2019-08-23 16:43:29 +00:00
|
|
|
# Read the file
|
2019-08-28 17:20:42 +00:00
|
|
|
local line=
|
2019-08-23 16:43:29 +00:00
|
|
|
while IFS= read -r line; do
|
2019-08-28 17:20:42 +00:00
|
|
|
if echo "$line" | grep -Eq '^\s*#\s*include\s*".+"'; then
|
|
|
|
# We have an include directive so strip the (first) file
|
|
|
|
local inc=$(echo "$line" | grep -Eo '".*"' | grep -Eo '\w*(\.?\w+)+' | head -1)
|
2020-04-03 17:07:46 +00:00
|
|
|
if list_has_item "$XINCS" "$inc"; then
|
|
|
|
# The file was excluded so error if the source attempts to use it
|
|
|
|
write_line "#error Using excluded file: $inc"
|
2019-08-23 16:43:29 +00:00
|
|
|
else
|
2020-04-03 17:07:46 +00:00
|
|
|
if ! list_has_item "$FOUND" "$inc"; then
|
|
|
|
# The file was not previously encountered
|
|
|
|
FOUND="$FOUND $inc"
|
|
|
|
if list_has_item "$KINCS" "$inc"; then
|
|
|
|
# But the include was flagged to keep as included
|
|
|
|
write_line "/**** *NOT* inlining $inc ****/"
|
|
|
|
write_line "$line"
|
|
|
|
else
|
|
|
|
# The file was neither excluded nor seen before so inline it
|
|
|
|
write_line "/**** start inlining $inc ****/"
|
|
|
|
add_file "$inc"
|
|
|
|
write_line "/**** ended inlining $inc ****/"
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
write_line "/**** skipping file: $inc ****/"
|
|
|
|
fi
|
2019-08-23 16:43:29 +00:00
|
|
|
fi
|
|
|
|
else
|
2020-04-03 17:07:46 +00:00
|
|
|
# Skip any 'pragma once' directives, otherwise write the source line
|
|
|
|
local write=$PONCE
|
|
|
|
if [ $write -eq 0 ]; then
|
|
|
|
if echo "$line" | grep -Eqv '^\s*#\s*pragma\s*once\s*'; then
|
|
|
|
write=1
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
if [ $write -ne 0 ]; then
|
|
|
|
write_line "$line"
|
|
|
|
fi
|
2019-08-23 16:43:29 +00:00
|
|
|
fi
|
|
|
|
done < "$file"
|
|
|
|
else
|
2019-08-25 20:49:01 +00:00
|
|
|
write_line "#error Unable to find \"$1\""
|
2019-08-23 16:43:29 +00:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2020-04-03 17:07:46 +00:00
|
|
|
while getopts ":r:x:k:po:" opts; do
|
2019-08-23 16:43:29 +00:00
|
|
|
case $opts in
|
|
|
|
r)
|
2020-04-03 17:07:46 +00:00
|
|
|
ROOTS="$ROOTS $OPTARG"
|
|
|
|
;;
|
|
|
|
x)
|
|
|
|
XINCS="$XINCS $OPTARG"
|
|
|
|
;;
|
|
|
|
k)
|
|
|
|
KINCS="$KINCS $OPTARG"
|
|
|
|
;;
|
|
|
|
p)
|
|
|
|
PONCE=1
|
2019-08-25 20:49:01 +00:00
|
|
|
;;
|
|
|
|
o)
|
|
|
|
DESTN="$OPTARG"
|
2019-08-23 16:43:29 +00:00
|
|
|
;;
|
|
|
|
*)
|
|
|
|
usage
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
shift $((OPTIND-1))
|
|
|
|
|
2019-08-25 20:49:01 +00:00
|
|
|
if [ -n "$1" ]; then
|
|
|
|
if [ -f "$1" ]; then
|
|
|
|
if [ -n "$DESTN" ]; then
|
|
|
|
printf "" > "$DESTN"
|
|
|
|
fi
|
2019-09-02 16:02:50 +00:00
|
|
|
test_grep
|
2019-08-25 20:49:01 +00:00
|
|
|
add_file $1
|
|
|
|
else
|
2019-08-28 17:20:42 +00:00
|
|
|
echo "Input file not found: \"$1\""
|
2019-08-25 20:49:01 +00:00
|
|
|
exit 1
|
|
|
|
fi
|
2019-08-23 16:43:29 +00:00
|
|
|
else
|
|
|
|
usage
|
|
|
|
fi
|
2019-08-25 20:49:01 +00:00
|
|
|
exit 0
|