61 Commits

Author SHA1 Message Date
3198f8905e v2.3.2: fixed symlink creation 2025-11-19 13:47:34 +05:00
458652d0c5 default project config change 2025-11-12 15:13:10 +05:00
7ef47c567c default user config change 2025-11-12 15:11:32 +05:00
1a4ad2a3f0 default user config change 2025-11-12 15:10:40 +05:00
f0ee39084e v2.3.1 2025-11-12 14:34:08 +05:00
e655f05da6 stopped writing {GRAY} before each message in myprint() 2025-11-12 14:33:22 +05:00
5cc2e1c2ad load user config before project config 2025-11-12 14:31:22 +05:00
fca94ffe4c chmod 755 scripts 2025-11-12 13:49:05 +05:00
7197f09ca4 updated gprof task 2025-11-09 23:44:39 +05:00
f884925788 added strip_exec.sh 2025-11-09 23:35:12 +05:00
baf45a4b10 minor log style change 2025-11-09 23:26:43 +05:00
6bfc0f5e9a added file_copy_default_if_not_present() and replace_var_value_in_script() 2025-11-09 23:09:34 +05:00
579dd5916e moved srcipts to include/ and changed function include() 2025-11-09 22:20:58 +05:00
05f3b9a0a0 v2.3.0: project user config 2025-11-09 21:45:10 +05:00
49ccc76933 added default_vscode/c_cpp_properties.json 2025-11-09 20:32:34 +05:00
d1660e05cb v2.2.4 2025-11-09 19:47:52 +05:00
259d9873fb c99 and more warnings 2025-11-09 19:44:18 +05:00
46029cd01f typo fixes 2025-07-18 02:21:12 +03:00
5953b84cff fixed detect_os.sh again 2025-06-20 19:29:52 +05:00
4c34c127a6 removed '\r' AGAIN 2025-06-20 19:21:34 +05:00
cae29d6395 removed \r characters from detect_os.sh 2025-06-20 19:11:08 +05:00
596c570756 v2.2.2 2025-04-26 03:58:53 +05:00
1c93d4eb73 bootstrap can show help for cbuild installations again 2025-04-26 03:57:09 +05:00
6126001e5a copy dependency's out files only if they are newer or don't exist 2025-04-26 00:23:13 +05:00
912b348f57 v2.2.1 2025-04-25 21:11:12 +05:00
d3d7867736 bootstrap v1.0.2 2025-04-25 21:09:52 +05:00
4488776afe v2.2.0 2025-04-25 20:40:52 +05:00
683414c59b removed CONFIG_VERSION variable from config 2025-04-25 20:34:23 +05:00
c7b590907d added LINKER_LIBS variable to config 2025-04-25 20:33:57 +05:00
653d459999 added default vscode launch config 2025-04-25 20:29:19 +05:00
f0038dd7c7 refactored sed calls to not use -i 2025-04-25 19:49:09 +05:00
4d06f57758 added detect_arch function and ARCH global variable 2025-04-25 19:23:41 +05:00
fa15e15758 fixed valgrind sources path trimming 2025-04-21 18:46:12 +05:00
d60c86ec3b readme fix 2024-09-13 21:12:25 +05:00
1576021cf7 v2.1.4 2024-08-09 22:05:34 +03:00
646773f574 rpath 2024-08-09 21:05:30 +03:00
3b7f72c8b3 2.1.3 2024-08-09 05:59:19 +03:00
144b333b60 v2.1.2 2024-07-21 06:35:00 +03:00
0b5eed8b4c bootstrap help and version list 2024-07-21 05:01:40 +03:00
5c1b063399 objects deletion after link and pack_static_lib 2024-07-21 02:49:42 +03:00
c5f8f6ee09 changed compile_c and compile_cpp functions 2024-07-21 01:55:55 +03:00
91dfc8bc74 fixed command line arguments parsing 2024-07-21 01:39:25 +03:00
d021389637 callgrind config 2024-07-20 23:13:44 +03:00
9c4959d51e removed TESTS_C and TESTS_CPP 2024-07-20 23:07:35 +03:00
4fc62f65b4 v2.1.0 2024-07-19 01:56:33 +03:00
f4c12e9b0e optimized line printing 2024-07-19 01:43:57 +03:00
4b4794e253 parse_version 2024-07-19 00:13:12 +03:00
7d60219c33 v2.0.2 2024-07-18 22:25:23 +03:00
de9a63f84b rebuild_dependencies 2024-07-18 22:06:35 +03:00
b7109ef9fa dependency compilation 2024-07-15 23:01:38 +03:00
023f0b1d40 2.0.1 2024-07-12 03:23:28 +03:00
6d0a819982 v2.0.1 2024-07-12 03:22:47 +03:00
ec923ef5c1 task bugfix 2024-07-12 03:19:09 +03:00
821cf0794e empty task check 2024-07-12 02:55:37 +03:00
ace31427c3 readme 2024-07-12 02:42:44 +03:00
fc833508b8 --new-project 2024-07-12 02:32:59 +03:00
0ebc1c47ee fixed myprint and added --version 2024-07-12 02:09:25 +03:00
9b1bbffbc4 v2.0.0 2024-07-12 01:55:58 +03:00
6d285a88d8 changed url in readme 2024-07-11 14:59:49 +03:00
574ce6eab3 fixes to dependency resolution 2024-02-22 00:59:54 +06:00
c20e1e8f1c fixed typos 2024-01-02 12:28:04 +06:00
40 changed files with 1363 additions and 655 deletions

2
.gitattributes vendored Normal file
View File

@@ -0,0 +1,2 @@
# git add --renormalize .
*.sh text eol=lf

11
.gitignore vendored
View File

@@ -1,8 +1,6 @@
# build results # build results
bin/ bin/
obj/ obj/
*.log
*.tmp
# IDE files # IDE files
.vs/ .vs/
@@ -13,4 +11,11 @@ obj/
# other files # other files
.old*/ .old*/
current.config old/
tmp/
temp/
*.tmp
*.temp
logs/
log/
*.log

1
CBUILD_VERSION Normal file
View File

@@ -0,0 +1 @@
2.3.2

View File

@@ -1,4 +1,91 @@
# v7 ## 2.3.2
+ fixed symlink creation in `OBJDIR/static_libs`
## 2.3.1
+ user config loads before project config
## 2.3.0
+ **CONFIG:** Added new file `./project.config.user.default`.
+ **CONFIG:** Changed `cbuild/default_tasks` to `@cbuild/default_tasks`.
+ **CONFIG:** Updated `gprof` task.
+ Changed `include` function: replaced prefix `cbuild/` with `@cbuild/`.
+ Moved most scripts to `include/`.
+ Renamed default config to `./project.config.default`.
+ Added functions:
+ `file_copy_default_if_not_present()`
+ `replace_var_value_in_script()`
+ Added task scripts:
+ `strip_exec.sh` - enabled in default config in task `build_exec`
## 2.2.4
+ *default config*: C standard changed to C99
+ *default config*: enabled more warnings
+ added file `default_vscode/c_cpp_properties.json`
+ fixed copying of `default_vscode` files
## 2.2.3
+ removed `\r` characters from `detect_os.sh`
## 2.2.2
+ `bootstrap.sh` can show help for cbuild installations again
+ copy dependency out files only if they are newer or don't exist
## 2.2.1
+ updated `bootsrap.sh` to 1.0.2
## 2.2.0
+ **config**: removed slash after `--fullpath-after` in `VALGRIND_ARGS`
+ **config**: removed `CONFIG_VERSION` variable
+ **config**: added `LINKER_LIBS` variable
+ added `detect_arch` function and `ARCH` global variable
+ added [default vscode launch config](./default_vscode/)
+ refactored `sed` calls to not use `-i` argument because it caused errors on NTFS
## v2.1.4
+ added `-Wl,-rpath` argument generation in `link()`. It forces ld to link to local shared library in `OUTDIR`
## v2.1.3
+ added `DEP_OTHER_OUT_FILES` to dependency configs
+ if `PRESERVE_OUT_DIRECTORY_STRUCTURE=true` then `DEP_DYNAMIC_OUT_FILES` and `DEP_OTHER_OUT_FILES` are copied to `OUTDIR` preserving directory structure (example_dir/lib1.so -> $OUTDIR/example_dir/lib1.so)
## v2.1.2
+ changed compile_c and compile_cpp functions
+ bootstrap now can print help and list of installed versions
## v2.1.1
+ **config**: removed `TESTS_C` and `TESTS_CPP` compilation
## v2.1.0
+ **config**: no more `current.config` and `default.config`, just `project.config`
+ improved version checking
+ `setup.sh` now installs each minor version in separate dicectory
+ added `bootstrap.sh` which automaticly selects cbuild version specified in project config
## v2.0.2
+ new dependency resolution system (see **config** and `example_dependency_configs`)
+ **config**: changed description of `OBJDIR`
+ **config**: added task `rebuild_dependencies`
+ added variable `TASK_ARGS` which can be used in task scripts
## v2.0.1
+ updated `.gitignore`
+ **config**: added `pwd` call to `valgrind` task
+ **config**: added `""` empty task check
## 2.0.0
+ updated setup.sh to do system-wide installation
+ deleted makefile (call `./cbuild.sh` or installed `cbuild`)
+ added command line arguments:
+ `--help`
+ `--version`
+ `--current-config`
+ `--default-config`
+ `--new-project`
+ all shebang changed to `#!/usr/bin/env bash`
+ `init.sh` became `config.sh` with function `load_config`
+ moved color variables and print functions to `myprint.sh`
## v1.7.0
+ added function `resolve_dependencies` to `link` + added function `resolve_dependencies` to `link`
+ added variables `DEPS_BASEDIR` and `DEPS` to config + added variables `DEPS_BASEDIR` and `DEPS` to config
+ added script `rebuild_dep.sh` which can be called through `Makefile` + added script `rebuild_dep.sh` which can be called through `Makefile`
@@ -7,7 +94,7 @@
+ added task `no_task` which is been set in `init.sh` when `TASK` is empty + added task `no_task` which is been set in `init.sh` when `TASK` is empty
+ now `STATIC_LIB_FILE` starts with "lib" + now `STATIC_LIB_FILE` starts with "lib"
# v6 ## v1.6.0
+ `build_profile` task was split to `profile` and `gprof` + `build_profile` task was split to `profile` and `gprof`
+ added task `sanitize` + added task `sanitize`
+ default C++ standard set to `c++11` + default C++ standard set to `c++11`
@@ -16,7 +103,7 @@
+ added function `try_delete_dir_or_file` for `clean` task + added function `try_delete_dir_or_file` for `clean` task
+ dead code removal in `build_exec` and `build_static_lib` + dead code removal in `build_exec` and `build_static_lib`
# v5 ## v1.5.0
+ added task `clean` + added task `clean`
+ added task `exec_dbg` + added task `exec_dbg`
+ added task `build_profile` + added task `build_profile`

View File

@@ -1,13 +1,31 @@
# cbuild # cbuild
My C/C++ build system written in bash. My C/C++ build system written in sh.
Repo contains some functions, which can be used in your custom task scripts. There are also some default tasks.
All tasks can be launched through `Makefile` or `cbuild/call_task.sh`. Tasks can be configured in `current.config`. ## Installation
```sh
## How to set up git clone https://timerix.ddns.net:3322/Timerix/cbuild.git
```bash cd cbuild
git clone http://github.com/Timerix22/cbuild.git && \ sudo ./setup.sh
cbuild/setup.sh submodule ```
Can be installed to `~/.local/` if you have no root rights.
- ```sh
./setup.sh --local
```
- Then add `~/.local/bin` to `PATH` in your shell config.
## Usage
1. Initialize cbuild project in some directory:
```sh
cd some_project
cbuild --new-project
```
2. Edit `project.config`.
3. Call some tasks:
```sh
cbuild build_exec_dbg exec
```
P.S. See help
```sh
cbuild -h
``` ```
Than create your project `default.config` based on `cbuild/default.config`.

125
bootstrap.sh Executable file
View File

@@ -0,0 +1,125 @@
#!/usr/bin/env bash
CBUILD_BOOTSTRAP_VERSION=1.0.3
set -eo pipefail
function version_parse {
local value="$1"
var_name="$2"
IFS_backup="$IFS"
IFS='.'
local v_array=($value)
IFS="$IFS_backup"
eval ${var_name}_version_major=${v_array[0]}
eval ${var_name}_version_minor=${v_array[1]}
eval ${var_name}_version_patch=${v_array[2]}
}
function safeprint {
printf "%s" "$@"
}
function exec_script_line {
local script="$1"
local line_num="$2"
local line_str="$(sed $line_num'!d' $script)"
if [ -z "$line_str" ]; then
echo "script line is empty"
exit 1
fi
eval "$line_str"
}
function print_version_list {
dir_local="$HOME/.local/share/cbuild"
dir_global="/usr/local/share/cbuild"
files=""
if [ -d "$dir_local" ]; then
files+=$(find "$dir_local" -maxdepth 2 -name 'CBUILD_VERSION')
fi
if [ -d "$dir_global" ]; then
files+=$(find "$dir_global" -maxdepth 2 -name 'CBUILD_VERSION')
fi
for f in $files; do
cat $f
echo " at $(dirname $f)"
done | sort -V
}
function print_help {
echo "cbuild-bootstrap v$CBUILD_BOOTSTRAP_VERSION"
echo "Script that launches a specific cbuild version selected by user or defined in a project config."
echo "Usage: cbuild [OPTIONS]"
echo "Options:"
echo " -h, --help Show this message"
echo " -v, --version Shows version"
echo " -c, --config FILE Set project config file path (default=./project.config)"
echo " --list-versions Shows list of installed cbuild versions"
}
# parse command line arguments
project_config_path="./project.config"
args=($@)
args_count=${#args[@]}
i=0
if [ $args_count -eq 0 ]; then
print_help
fi
while [ $i -lt $args_count ]
do
case "${args[i]}" in
'-v' | '--version')
echo "cbuild-bootstrap v$CBUILD_BOOTSTRAP_VERSION"
echo "list of installed cbuild versions:"
print_version_list
exit 0
;;
'-h' | '--help')
print_help
;;
'-c' | '--config')
i=$((i+1))
project_config_path="${args[i]}"
;;
*)
;;
esac
i=$((i+1))
done
if [ -z "$CBUILD_VERSION" ]; then
if [ -f "$project_config_path" ]; then
# read version from project config
exec_script_line "$project_config_path" 2
else
echo "project config not found"
echo "list of installed cbuild versions:"
print_version_list
# cut installation directories from list
versions=$(print_version_list | cut -d ' ' -f 1)
latest_version=$(echo "$versions" | tail -n 1)
echo "select version (default=$latest_version):"
read -r CBUILD_VERSION
[ -z "$CBUILD_VERSION" ] && CBUILD_VERSION=$latest_version
fi
fi
version_parse "$CBUILD_VERSION"
cbuild_ommand="cbuild${_version_major}.${_version_minor} $@"
LOG_FILE="$(realpath cbuild.log)"
set +eo pipefail
# enable logging stdout and stderr to file
$cbuild_ommand 2>&1 | tee "$LOG_FILE"
# log file can be deleted by clean task
if [ -f "$LOG_FILE" ]; then
# remove terminal escape codes
sed -e 's/[^[:blank:][:print:]]//g' \
-e 's/\[0;[0-9][0-9]m//g' \
-e 's/\[0;[0-9]m//g' \
-e 's/\[[0-9][0-9]m//g' \
-e 's/\[[0-9]m//g' \
-e 's/ H //g' \
-e 's/\[3gH //g' \
"$LOG_FILE" > "$LOG_FILE.clean"
mv "$LOG_FILE.clean" "$LOG_FILE"
fi

View File

@@ -1,26 +0,0 @@
#!/bin/bash
function exec_script {
myprint "${BLUE}executing $1"
source "$1"
}
function try_exec_script {
if [ -f "$1" ]; then
exec_script "$1"
fi
}
function call_task {
TASK="$1"
source cbuild/init.sh
myprint "${CYAN}===========[$TASK]==========="
myprint "${WHITE}project: ${CYAN}$PROJECT"
try_exec_script "$PRE_TASK_SCRIPT"
exec_script "$TASK_SCRIPT"
try_exec_script "$POST_TASK_SCRIPT"
}
time call_task "$1"
# new line
echo

176
cbuild.sh Executable file
View File

@@ -0,0 +1,176 @@
#!/usr/bin/env bash
INSTALLED_CBUILD_VERSION=2.3.2
# set \t size to 4 spaces
tabs 4
# exit on errors
set -eo pipefail
function version_parse {
local value="$1"
var_name="$2"
IFS_backup="$IFS"
IFS='.'
local v_array=($value)
IFS="$IFS_backup"
eval ${var_name}_version_major=${v_array[0]}
eval ${var_name}_version_minor=${v_array[1]}
eval ${var_name}_version_patch=${v_array[2]}
}
version_parse $INSTALLED_CBUILD_VERSION installed
if [ -z "$CBUILD_INSTALL_DIR" ]; then
CBUILD_INSTALL_DIR="$HOME/.local/share/cbuild/${installed_version_major}.${installed_version_minor}"
if [ ! -f "$CBUILD_INSTALL_DIR/cbuild.sh" ]; then
CBUILD_INSTALL_DIR="/usr/local/share/cbuild/${installed_version_major}.${installed_version_minor}"
if [ ! -f "$CBUILD_INSTALL_DIR/cbuild.sh" ]; then
echo "CBUILD_INSTALL_DIR '$CBUILD_INSTALL_DIR' doesn't exist"
exit 1
fi
fi
fi
function include {
local script_path="$1"
if [[ "$script_path" == @cbuild/* ]]; then
script_path="$CBUILD_INSTALL_DIR/$(echo $script_path | sed 's,^@cbuild/,,')"
fi
# echp "including script $script_path"
. "$script_path"
}
include "@cbuild/include/myprint.sh"
include "@cbuild/include/functions.sh"
include "@cbuild/include/config.sh"
function print_help {
myprint "${GRAY}cbuild v$INSTALLED_CBUILD_VERSION"
myprint "C/C++ project build system written in bash."
myprint "Usage: cbuild [OPTIONS] [TASKS]"
myprint "Options:"
myprint " -h, --help Show this message"
myprint " -v, --version Shows version"
myprint " -c, --config FILE Set project config file path (default=./project.config)"
myprint " -n, --new-project [DIR] Initialize new cbuild project directory (default=./)"
}
# parse command line arguments
project_config_path="./project.config"
args=($@)
args_count=${#args[@]}
selected_tasks_array=()
i=0
if [ $args_count -eq 0 ]; then
print_help
exit 1
fi
while [ $i -lt $args_count ]
do
case "${args[i]}" in
'-h' | '--help')
print_help
exit 0
;;
'-v' | '--version')
myprint "${GRAY}cbuild v$INSTALLED_CBUILD_VERSION"
exit 0
;;
'-c' | '--config')
i=$((i+1))
project_config_path="${args[i]}"
;;
'-n' | '--new-project')
i=$((i+1))
new_project_dir="${args[i]}"
if [ -z "$new_project_dir" ]; then
new_project_dir="."
else
mkdir -p "$new_project_dir"
fi
# create project config
project_config_path="$new_project_dir/project.config"
cp "$CBUILD_INSTALL_DIR/project.config.default" "$project_config_path.temp"
myprint "${WHITE}Enter project name: "
read -r project_name
sed "s,\%PROJECT_NAME\%,$project_name,g" \
"$project_config_path.temp" > "$project_config_path"
rm "$project_config_path.temp"
myprint "${GREEN}Created '$project_config_path'"
# create project user default config
project_user_config_path="$new_project_dir/project.config.user.default"
cp "$CBUILD_INSTALL_DIR/project.config.user.default" "$project_user_config_path"
myprint "${GREEN}Created '$project_user_config_path'"
if ask_yn "${WHITE}Copy default .gitignore?"; then
cp -v "$CBUILD_INSTALL_DIR/.gitignore" "$new_project_dir/"
fi
if ask_yn "${WHITE}Copy default .vscode launch tasks?"; then
new_project_vscode_dir="$new_project_dir/.vscode"
mkdir -pv "$new_project_vscode_dir"
for vscode_dir_f in $(find "$CBUILD_INSTALL_DIR/default_vscode/" -type f); do
cp -vr "$vscode_dir_f" "$new_project_vscode_dir/"
done
sed "s,\%PROJECT_NAME\%,$project_name,g" \
"$new_project_vscode_dir/launch.json" > "$new_project_vscode_dir/launch.json.temp"
mv "$new_project_vscode_dir/launch.json.temp" "$new_project_vscode_dir/launch.json"
fi
exit 0
;;
*)
selected_tasks_array+=(${args[i]})
;;
esac
i=$((i+1))
done
function call_task {
local project_config_path="$1"
local task="$2"
TASK_ARGS="$3"
print_header "${CYAN}" "─" "$PROJECT/$task"
load_config "$project_config_path" "$task" true
if [ ! -z "$PRE_TASK_SCRIPT" ]; then
myprint "${BLUE}executing ${WHITE}'$PRE_TASK_SCRIPT'"
include "$PRE_TASK_SCRIPT"
fi
if [ ! -z "$TASK_SCRIPT" ]; then
myprint "${BLUE}executing ${WHITE}'$TASK_SCRIPT'"
include "$TASK_SCRIPT"
fi
if [ ! -z "$POST_TASK_SCRIPT" ]; then
myprint "${BLUE}executing ${WHITE}'$POST_TASK_SCRIPT'"
include "$POST_TASK_SCRIPT"
fi
}
function call_tasks {
local tasks="$@"
load_config "$project_config_path" "" false
print_header "${WHITE}" "═" "$PROJECT"
project_dir="$(pwd)"
for task_str in $tasks ; do
task=$(safeprint "$task_str" | sed 's/=.*//g')
local args_str=$(safeprint "$task_str" | grep -oP '(?<=\=).*' || echo "")
IFS_backup="$IFS"
IFS=',;'
args_array=($args_str)
IFS="$IFS_backup"
call_task "$project_config_path" "$task" "${args_array[@]}"
cd "$project_dir"
done
}
print_hline "${WHITE}" "═"
selected_tasks_count=${#selected_tasks_array[@]}
if [ $selected_tasks_count -gt 0 ]; then
call_tasks "${selected_tasks_array[@]}"
fi

View File

@@ -1,7 +1,7 @@
#!/bin/bash #!/usr/bin/env bash
SCRIPTS="$(find ./ -name '*.sh')"
for F in $SCRIPTS for f in $(find ./ -name '*.sh')
do do
echo "$F" chmod a+x "$f"
chmod +x "$F" ls -lh "$f"
done done

View File

@@ -1,11 +0,0 @@
#!/bin/bash
BLACK='\033[0;30m'
GRAY='\033[0;37m'
WHITE='\033[0;97m'
RED='\033[0;91m'
GREEN='\033[0;92m'
YELLOW='\033[0;93m'
BLUE='\033[0;94m'
PURPLE='\033[0;95m'
CYAN='\033[0;96m'

View File

@@ -1,91 +0,0 @@
######################################
###### Build tasks #######
######################################
all: build_exec_dbg
# creates executable using profiling info generated by profile
build_exec: profile
@cbuild/call_task.sh build_exec 2>&1 | tee -a make_raw.log
# creates executable with debug info and no optimizations
build_exec_dbg:
@cbuild/call_task.sh build_exec_dbg 2>&1 | tee make_raw.log
# creates shared library
build_shared_lib:
@cbuild/call_task.sh build_shared_lib 2>&1 | tee make_raw.log
# creates shared library with debug symbols and no optimizations
build_shared_lib_dbg:
@cbuild/call_task.sh build_shared_lib_dbg 2>&1 | tee make_raw.log
# creates static library
build_static_lib:
@cbuild/call_task.sh build_static_lib 2>&1 | tee make_raw.log
# creates static library with debug symbols and no optimizations
build_static_lib_dbg:
@cbuild/call_task.sh build_static_lib_dbg 2>&1 | tee make_raw.log
# recompile libsome_dep.a in the next build task
#rebuild_some_dep:
# @cbuild/rebuild_dep.sh libsome_dep.a 2>&1 | tee make_raw.log
#rebuild_all: rebuild_some_dep
######################################
###### Launch tasks #######
######################################
# executes $EXEC_FILE
exec: build_exec
@cbuild/call_task.sh exec 2>&1 | tee -a make_raw.log
# executes $EXEC_FILE
exec_dbg: build_exec_dbg
@cbuild/call_task.sh exec 2>&1 | tee -a make_raw.log
# executes $EXEC_FILE with valgrind memory checker
valgrind: build_exec_dbg
@cbuild/call_task.sh valgrind 2>&1 | tee -a make_raw.log
# generates profiling info
profile:
@cbuild/call_task.sh profile 2>&1 | tee make_raw.log
# compiles program with -pg and runs it with gprof
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
gprof:
@cbuild/call_task.sh gprof 2>&1 | tee make_raw.log
# compiles program and runs it with callgrind (part of valgrind)
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
# P.S. detailed rezults can be viewed in KCacheGrind
callgrind:
@cbuild/call_task.sh callgrind 2>&1 | tee make_raw.log
# compiles executable with sanitizers and executes it to find errors and warnings
sanitize:
@cbuild/call_task.sh sanitize 2>&1 | tee make_raw.log
######################################
###### Other tasks #######
######################################
# deletes generated files
clean:
@cbuild/call_task.sh clean 2>&1 | tee make_raw.log
# removes all unreadable characters copied from stdio
fix_log:
sed 's/[^[:blank:][:print:]]//g' make_raw.log \
| sed 's/\[0;[0-9][0-9]m//g' \
| sed 's/\[0;[0-9]m//g' \
| sed 's/\[[0-9][0-9]m//g' \
| sed 's/\[[0-9]m//g' \
| sed 's/ H //g' \
| sed 's/\[3gH //g' \
> make_fixed.log

View File

@@ -1,186 +0,0 @@
#!/bin/bash
CBUILD_VERSION=7
CONFIG_VERSION=1
PROJECT="NULL"
CMP_C="gcc"
CMP_CPP="g++"
STD_C="c11"
STD_CPP="c++11"
WARN_C="-Wall -Wno-discarded-qualifiers -Wextra -Wno-unused-parameter"
WARN_CPP="-Wall -Wextra -Wno-unused-parameter"
SRC_C="$( find src -name '*.c')"
SRC_CPP="$( find src -name '*.cpp')"
TESTS_C="$( find tests -name '*.c')"
TESTS_CPP="$(find tests -name '*.cpp')"
# dir with dependeicy dirs
DEPS_BASEDIR="."
# EXAMPLE: "dependency_dir='build_task out_dir lib_file'
# other_depndency_dir=..."
# Dependencies must be declared on separate lines
# Values can be override by resetting one of dependencies:
# DEPS="$DEPS
# dependency_dir='...'"
DEPS=""
# OBJDIR structure:
# ├── objects/ - dir where compiled *.o files are stored. cleans every call of build task
# ├── profile/ - dir where gcc *.gcda profiling info files stored
# └── libs/ - there you can put static libs and linker will find them
OBJDIR="obj"
OUTDIR="bin"
STATIC_LIB_FILE="lib$PROJECT.a"
# OS-specific options
case "$OS" in
WINDOWS)
EXEC_FILE="$PROJECT.exe"
SHARED_LIB_FILE="$PROJECT.dll"
# example: "-I./"
INCLUDE=""
;;
LINUX)
EXEC_FILE="$PROJECT"
SHARED_LIB_FILE="$PROJECT.so"
INCLUDE=""
;;
*)
error "operating system $OS has no configuration variants"
;;
esac
# TASKS
case "$TASK" in
# creates executable using profiling info if it exists
build_exec)
# -flto applies more optimizations across object files
# -flto=auto is needed to multithreaded copilation
# -fuse-linker-plugin is required to use static libs with lto
# -fprofile-use enables compiler to use profiling info files to optimize executable
# -fprofile-prefix-path sets path where profiling info about objects are be saved
# -fdata-sections -ffunction-sections -Wl,--gc-sections removes unused code
C_ARGS="-O2 -flto=auto -fuse-linker-plugin -fprofile-use -fprofile-prefix-path=$(realpath $OBJDIR)/objects -fdata-sections -ffunction-sections -Wl,--gc-sections"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
POST_TASK_SCRIPT=
;;
# creates executable with debug info and no optimizations
build_exec_dbg)
C_ARGS="-O0 -g3"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
POST_TASK_SCRIPT=
;;
# creates shared library
build_shared_lib)
C_ARGS="-O2 -fpic -flto -shared"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS -Wl,-soname,$SHARED_LIB_FILE"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_shared_lib.sh
POST_TASK_SCRIPT=
;;
# creates shared library with debug symbols and no optimizations
build_shared_lib_dbg)
C_ARGS="-O0 -g3 -fpic -shared"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS -Wl,-soname,$SHARED_LIB_FILE"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_shared_lib.sh
POST_TASK_SCRIPT=
;;
# creates static library
build_static_lib)
C_ARGS="-O2 -fpic -fdata-sections -ffunction-sections"
CPP_ARGS="$C_ARGS"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_static_lib.sh
POST_TASK_SCRIPT=
;;
# creates static library with debug symbols and no optimizations
build_static_lib_dbg)
C_ARGS="-O0 -g3"
CPP_ARGS="$C_ARGS"
PRE_TASK_SCRIPT=
TASK_SCRIPT=cbuild/default_tasks/build_static_lib.sh
POST_TASK_SCRIPT=
;;
# executes $EXEC_FILE
exec)
TASK_SCRIPT=cbuild/default_tasks/exec.sh
;;
# executes $EXEC_FILE with valgrind memory checker
valgrind)
VALGRIND_ARGS="-s --read-var-info=yes --track-origins=yes --fullpath-after=$PROJECT/ --leak-check=full --show-leak-kinds=all"
TASK_SCRIPT=cbuild/default_tasks/valgrind.sh
;;
# generates profiling info
profile)
OUTDIR="$OUTDIR/profile"
# -flto applies more optimizations across object files
# -flto=auto is needed to multithreaded copilation
# -fuse-linker-plugin is required to use static libs with lto
# -pg adds code to executable, that generates file containing function call info (gmon.out)
# -fprofile-generate generates executable with profiling code
# -fprofile-prefix-path sets path where profiling info about objects will be saved
C_ARGS="-O2 -flto=auto -fuse-linker-plugin -fprofile-generate -fprofile-prefix-path=$(realpath $OBJDIR)/objects"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
TASK_SCRIPT=cbuild/default_tasks/profile.sh
POST_TASK_SCRIPT=
;;
# compiles program with -pg and runs it with gprof
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
gprof)
OUTDIR="$OUTDIR/gprof"
# -pg adds code to executable, that generates file containing function call info (gmon.out)
C_ARGS="-O2 -flto=auto -fuse-linker-plugin -pg"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
TASK_SCRIPT=cbuild/default_tasks/gprof.sh
POST_TASK_SCRIPT=
;;
# compiles program and runs it with callgrind (part of valgrind)
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
# P.S. detailed rezults can be viewed in KCacheGrind
callgrind)
OUTDIR="$OUTDIR/callgrind"
# -pg adds code to executable, that generates file containing function call info (gmon.out)
C_ARGS="-O2 -flto=auto -fuse-linker-plugin"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=tasks/pre_build.sh
TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
POST_TASK_SCRIPT=cbuild/default_tasks/callgrind.sh
;;
# compiles executable with sanitizers and executes it to find errors and warnings
sanitize)
OUTDIR="$OUTDIR/sanitize"
C_ARGS="-O0 -g3 -fsanitize=undefined,address"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS"
PRE_TASK_SCRIPT=cbuild/default_tasks/build_exec.sh
TASK_SCRIPT=cbuild/default_tasks/exec.sh
POST_TASK_SCRIPT=
;;
# deletes generated files
clean)
TASK_SCRIPT=cbuild/default_tasks/clean.sh
;;
# nothing to do
no_task)
;;
# unknown task
*)
error "task <$TASK> not found"
;;
esac

View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
# delete old objects # delete old objects
clean_dir "$OBJDIR/objects" clean_dir "$OBJDIR/objects"
@@ -15,6 +15,6 @@ else
done done
fi fi
compile_c "$C_ARGS" "$SRC_C $TESTS_C" [ ! -z "$SRC_C" ] && compile_c "$CMP_C" "$STD_C" "$WARN_C" "$C_ARGS" "$INCLUDE" "$SRC_C"
compile_cpp "$CPP_ARGS" "$SRC_CPP $TESTS_CPP" [ ! -z "$SRC_CPP" ] && compile_cpp "$CMP_CPP" "$STD_CPP" "$WARN_CPP" "$CPP_ARGS" "$INCLUDE" "$SRC_CPP"
link "$LINKER_ARGS" "$EXEC_FILE" link "$LINKER_ARGS" "$EXEC_FILE"

View File

@@ -1,7 +1,7 @@
#!/bin/bash #!/usr/bin/env bash
# delete old objects # delete old objects
clean_dir "$OBJDIR/objects" clean_dir "$OBJDIR/objects"
compile_c "$C_ARGS" "$SRC_C" [ ! -z "$SRC_C" ] && compile_c "$CMP_C" "$STD_C" "$WARN_C" "$C_ARGS" "$INCLUDE" "$SRC_C"
compile_cpp "$CPP_ARGS" "$SRC_CPP" [ ! -z "$SRC_CPP" ] && compile_cpp "$CMP_CPP" "$STD_CPP" "$WARN_CPP" "$CPP_ARGS" "$INCLUDE" "$SRC_CPP"
link "$LINKER_ARGS" "$SHARED_LIB_FILE" link "$LINKER_ARGS" "$SHARED_LIB_FILE"

View File

@@ -1,7 +1,7 @@
#!/bin/bash #!/usr/bin/env bash
# delete old objects # delete old objects
clean_dir "$OBJDIR/objects" clean_dir "$OBJDIR/objects"
compile_c "$C_ARGS" "$SRC_C" [ ! -z "$SRC_C" ] && compile_c "$CMP_C" "$STD_C" "$WARN_C" "$C_ARGS" "$INCLUDE" "$SRC_C"
compile_cpp "$CPP_ARGS" "$SRC_CPP" [ ! -z "$SRC_CPP" ] && compile_cpp "$CMP_CPP" "$STD_CPP" "$WARN_CPP" "$CPP_ARGS" "$INCLUDE" "$SRC_CPP"
pack_static_lib "$STATIC_LIB_FILE" pack_static_lib "$STATIC_LIB_FILE"

2
default_tasks/callgrind.sh Normal file → Executable file
View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
cd "$OUTDIR" cd "$OUTDIR"

23
default_tasks/clean.sh Normal file → Executable file
View File

@@ -1,23 +1,14 @@
#!/usr/bin/bash #!/usr/bin/env bash
try_delete_dir_or_file "$OBJDIR" try_delete_dir_or_file "$OBJDIR"
try_delete_dir_or_file "$OUTDIR" try_delete_dir_or_file "$OUTDIR"
myprint "${WHITE}deleting build logs" myprint "${WHITE}deleting build logs"
rm -rf *.log rm -rf *.log
for tmpfile in $(ls -a | grep -e '\.rebuild.*\.tmp'); do project_dir="$(pwd)"
try_delete_dir_or_file "$tmpfile" for dep in $ENABLED_DEPENDENCIES; do
load_dependency_config "$DEPENDENCY_CONFIGS_DIR/$dep.config"
cd "$DEP_WORKING_DIR"
exec_command "$DEP_CLEAN_COMMAND"
cd "$project_dir"
done done
set +e
OLDIFS="$IFS"
IFS=$'\n'
for dep in $DEPS; do
dep_dir=$(echo ${dep/=*/} | tr -d '[:blank:]')
myprint "${CYAN}--------------[$dep_dir]--------------"
cd "$DEPS_BASEDIR/$dep_dir"
make clean
cd ..
done
IFS="$OLDIFS"
set -e

View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
cd "$OUTDIR" cd "$OUTDIR"
./$EXEC_FILE ./$EXEC_FILE

2
default_tasks/gprof.sh Normal file → Executable file
View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
cd "$OUTDIR" cd "$OUTDIR"

2
default_tasks/profile.sh Normal file → Executable file
View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
cd "$OUTDIR" cd "$OUTDIR"

View File

@@ -0,0 +1,11 @@
#!/usr/bin/env bash
dependencies="$TASK_ARGS"
if [ "$dependencies" = 'all' ]; then
dependencies="$ENABLED_DEPENDENCIES"
fi
if [ ! -z "$dependencies" ]; then
myprint "${BLUE}dependencies to be rebuild: $dependencies"
build_dependencies "$dependencies" true
else
myprint "${YELLOW}no dependencies specified"
fi

4
default_tasks/strip_exec.sh Executable file
View File

@@ -0,0 +1,4 @@
#!/usr/bin/env bash
exe_path="$OUTDIR/$EXEC_FILE"
myprint "${BLUE}stripping symbols from ${WHITE}$exe_path"
strip -s "$exe_path"

View File

@@ -1,4 +1,4 @@
#!/bin/bash #!/usr/bin/env bash
cd "$OUTDIR" cd "$OUTDIR"
rm -f "valgrind.log" rm -f "valgrind.log"

1
default_vscode/.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
settings.json

View File

@@ -0,0 +1,14 @@
{
"configurations": [
{
"name": "all",
"defines": [],
"includePath": [
// "include",
"${default}"
],
"cStandard": "c99"
}
],
"version": 4
}

View File

@@ -0,0 +1,29 @@
{
"version": "0.2.0",
"configurations": [
{
"name": "gdb_debug",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/bin/%PROJECT_NAME%",
"windows": { "program": "${workspaceFolder}/bin/%PROJECT_NAME%.exe" },
"preLaunchTask": "build_exec_dbg",
"stopAtEntry": false,
"cwd": "${workspaceFolder}/bin",
"externalConsole": false,
"internalConsoleOptions": "neverOpen",
"MIMode": "gdb",
"miDebuggerPath": "gdb",
"setupCommands": [
{
"text": "-enable-pretty-printing",
"ignoreFailures": true
},
{
"text": "-gdb-set disassembly-flavor intel",
"ignoreFailures": true
}
]
}
]
}

31
default_vscode/tasks.json Normal file
View File

@@ -0,0 +1,31 @@
{
"version": "2.0.0",
"tasks": [
{
"label": "build_exec_dbg",
"detail": "build project with debug symbols",
"type": "cppbuild",
"command": "bash",
"args": [
"-c",
"cbuild build_exec_dbg"
],
"options": {
"cwd": "${workspaceFolder}"
},
"problemMatcher": ["$gcc"],
"group": {
"kind": "build"
},
"presentation": {
"echo": true,
"reveal": "always",
"focus": true,
"panel": "shared",
"showReuseMessage": false,
"clear": true
}
}
]
}

View File

@@ -1,26 +0,0 @@
#!/bin/bash
source cbuild/colors.sh
uname_rezult="$(uname -o)"
myprint "${GRAY}uname rezult: '$uname_rezult'"
case "$uname_rezult" in
Msys | Cygwin | "MS/Windows")
OS=WINDOWS
;;
Linux | GNU/Linux | Android)
OS=LINUX
;;
FreeBSD)
OS=FREEBSD
;;
Darwin)
OS=MACOS
;;
*)
error "unknown operating system: $uname_rezult"
;;
esac
myprint "${GREEN}detected OS: $OS"

View File

@@ -0,0 +1,13 @@
#!/usr/bin/env bash
DEP_WORKING_DIR='dependencies/libexample1'
DEP_PRE_BUILD_COMMAND=''
DEP_BUILD_COMMAND='make libexample1.a'
DEP_POST_BUILD_COMMAND=''
DEP_CLEAN_COMMAND='make clean'
# won't be copied to project $OUTDIR
DEP_STATIC_OUT_FILES='libexample1.a libexample1_addon.a'
PRESERVE_OUT_DIRECTORY_STRUCTURE=false;
# will be copied to project $OUTDIR
DEP_DYNAMIC_OUT_FILES='libexample1.config.json'
# will be copied to project $OUTDIR
DEP_OTHER_OUT_FILES=''

View File

@@ -0,0 +1,22 @@
#!/usr/bin/env bash
DEP_WORKING_DIR='dependencies/libexample2'
DEP_PRE_BUILD_COMMAND=''
DEP_POST_BUILD_COMMAND=''
DEP_CLEAN_COMMAND='make clean'
DEP_STATIC_OUT_FILES=''
DEP_OTHER_OUT_FILES=''
case $OS in
WINDOWS)
DEP_BUILD_COMMAND='make libexample2.dll && mkdir -p win-x64 && mv libexample.dll win-x64'
DEP_DYNAMIC_OUT_FILES='win-x64/libexample2.dll'
PRESERVE_OUT_DIRECTORY_STRUCTURE=true # library will be copied to $OUTDIR/win-x64
;;
LINUX)
DEP_BUILD_COMMAND='make libexample2.so && mkdir -p linux-x64 && mv libexample.so linux-x64'
DEP_DYNAMIC_OUT_FILES='linux-x64/libexample2.so'
PRESERVE_OUT_DIRECTORY_STRUCTURE=true # library will be copied to $OUTDIR/linux-x64
;;
*)
error "operating system $OS has no configuration variants"
;;
esac

View File

@@ -1,154 +0,0 @@
#!/bin/bash
function myprint {
# first gray is needed to print string trarting with -
printf "${GRAY}$@${GRAY}\n"
}
function error {
myprint "${RED}$1"
exit 1
}
function clean_dir {
local dir="$1"
myprint "${WHITE}cleaning $dir"
rm -rf "$dir"
mkdir "$dir"
}
function delete_dir {
local dir="$1"
myprint "${WHITE}deleting $dir"
rm -rf "$dir"
}
function try_delete_dir_or_file {
local path="$1"
if [ -f "$path" ] || [ -d "$path" ]; then
rm -rf "$path"
myprint "${WHITE}deleting $path"
fi
}
function compile {
local cmp="$1"
myprint "${BLUE}compiler: ${GRAY}$cmp"
local std="$2"
myprint "${BLUE}standard: ${GRAY}$std"
local warn="$3"
myprint "${BLUE}warnings: ${GRAY}$warn"
local args="$4"
myprint "${BLUE}args: ${GRAY}$args"
local include="$5"
myprint "${BLUE}include dirs: ${GRAY}$include"
local sources="$6"
myprint "${BLUE}sources: ${GRAY}$sources"
for srcfile in $sources
do (
local object="$OBJDIR/objects/$(basename $srcfile).o"
if ! $($cmp -std=$std $warn $args $include -c -o $object $srcfile)
then
error "some error happened"
# TODO stop all threads
fi
) & done
wait
}
# (args, sources)
function compile_c {
myprint "${CYAN}-------------[compile_c]--------------"
compile "$CMP_C" "$STD_C" "$WARN_C" "$1" "$INCLUDE" "$2"
}
# (args, sources)
function compile_cpp {
myprint "${CYAN}------------[compile_cpp]-------------"
compile "$CMP_CPP" "$STD_CPP" "$WARN_CPP" "$1" "$INCLUDE" "$2"
}
# (outfile)
function pack_static_lib {
myprint "${CYAN}----------[pack_static_lib]-----------"
local outfile="$1"
myprint "${BLUE}outfile: ${GRAY}$outfile"
local objects="$(find $OBJDIR/objects -name *.o)
$(find $OBJDIR/libs -name '*.a')"
myprint "${BLUE}objects: ${GRAY}$objects"
if ar rcs "$OUTDIR/$outfile" $(echo "$objects" | tr '\n' ' ')
then
myprint "${GREEN}file $CYAN$outfile ${GREEN}created"
else
error "some error happened"
fi
}
# if $lib_file doesn't exist or rebuild_* task was executed, builds static lib
function handle_static_dependency {
local deps_basedir="${1%/}"
local lib_project_dir="$2"
local lib_build_task="$3"
local lib_build_dir="$4"
local lib_file="$5"
if [ ! -f "$OBJDIR/libs/$lib_file" ] || [ -f .rebuild_$lib_file.tmp ]; then
[[ -z "$lib_build_task" ]] && error "lib_build_task is empty"
myprint "${BLUE}making $lib_file by task $lib_build_task"
cd "$deps_basedir/$lib_project_dir"
if ! make "$lib_build_task"; then
exit 1
fi
cd ..
cp "$deps_basedir/$lib_project_dir/$lib_build_dir/$lib_file" "$OBJDIR/libs/"
myprint "${GREEN}copied ${CYAN}$lib_file to $OBJDIR/libs/"
rm -f .rebuild_$lib_file.tmp
fi
}
function resolve_dependencies {
deps_basedir=$1
deps=$2
[[ -z "$deps_basedir" ]] && deps_basedir=.
OLDIFS="$IFS"
IFS=$'\n'
# Evalueting dependency expressions.
# Necessary for overriding dependency configurations.
for dep in $deps; do
eval $dep
done
# handling dependencies
for dep in $deps; do
IFS="$OLDIFS"
dep_dir=$(echo ${dep/=*/} | tr -d '[:blank:]')
eval 'dep_params=$'$dep_dir
f_args="$deps_basedir $dep_dir $dep_params"
myprint "${BLUE}resolving dependency ${WHITE}$dep_dir${BLUE}: ${GRAY}$f_args"
handle_static_dependency $f_args
IFS=$'\n'
done
IFS="$OLDIFS"
}
function link {
myprint "${CYAN}----------------[link]----------------"
local args="$1"
local outfile="$2"
resolve_dependencies "$DEPS_BASEDIR" "$DEPS"
myprint "${BLUE}args: ${GRAY}$args"
myprint "${BLUE}outfile: ${GRAY}$outfile"
local objects="$(find $OBJDIR/objects -name '*.o')
$(find $OBJDIR/libs -name '*.a')"
myprint "${BLUE}objects: ${GRAY}$objects"
local command="$CMP_CPP $(echo "$objects" | tr '\n' ' ') $args -o $OUTDIR/$outfile"
myprint "$command"
if $command
then
myprint "${GREEN}file $CYAN$outfile ${GREEN}created"
else
error "some error happened"
fi
}

73
include/config.sh Executable file
View File

@@ -0,0 +1,73 @@
#!/usr/bin/env bash
include "@cbuild/include/myprint.sh"
include "@cbuild/include/functions.sh"
include "@cbuild/include/detect_os.sh"
function load_config {
local proj_conf_file="$1"
TASK="$2"
#true or false
local quiet=$3
if [ -z "$proj_conf_file" ]; then
error "config path is null"
fi
if [ ! -f "$proj_conf_file" ]; then
error "${YELLOW}$proj_conf_file doesn't exist"
fi
# load project user config
local proj_conf_user_file="$proj_conf_file.user"
file_copy_default_if_not_present "$proj_conf_user_file" "$proj_conf_user_file.default"
myprint "${BLUE}loading ${WHITE}'$proj_conf_user_file'"
# throw error on undefined variable usage
set -u
include "$proj_conf_user_file"
# don't throw error on undefined variable usage
set +u
myprint "${BLUE}loading ${WHITE}'$proj_conf_file'"
OS=$(detect_os)
ARCH=$(detect_arch)
myprint_quiet $quiet "${GREEN}detected OS: $OS"
# getting version of cbuild installation
if [ -z "$INSTALLED_CBUILD_VERSION" ]; then
error "couldnt get current cbuild installation version"
fi
# undefined task
[ -z "$TASK" ] && TASK="no_task"
# getting cbuild version from config (CBUILD_VERSION declaration is at line 2)
exec_script_line "$proj_conf_file" 2 $quiet
myprint_quiet $quiet "${WHITE}'${proj_conf_file}' cbuild version: ${CYAN}$CBUILD_VERSION"
myprint_quiet $quiet "${WHITE}installed cbuild version: ${CYAN}$INSTALLED_CBUILD_VERSION"
# checking versions
version_parse $INSTALLED_CBUILD_VERSION installed
version_parse $CBUILD_VERSION config
if [ "$installed_version_major.$installed_version_minor" != "$config_version_major.$config_version_minor" ]; then
error "config was created for cbuild$config_version_major.$config_version_minor, but loaded whith cbuild$installed_version_major.$installed_version_minor which is incompatible"
fi
if [[ $installed_version_patch < $config_version_patch ]]; then
myprint "${YELLOW}New patch for cbuild$installed_version_major.$installed_version_minor is abaliable."
myprint "${YELLOW}Install it to get latest bugfixes."
fi
set -u
include "$proj_conf_file"
mkdir -p "$OUTDIR"
mkdir -p "$OBJDIR/objects"
mkdir -p "$OBJDIR/static_libs"
mkdir -p "$OBJDIR/dynamic_libs"
mkdir -p "$OBJDIR/profile"
set +u
myprint_quiet $quiet "${GREEN}config loading completed"
}

45
include/detect_os.sh Executable file
View File

@@ -0,0 +1,45 @@
#!/usr/bin/env bash
include "@cbuild/include/myprint.sh"
function detect_os {
local uname_result="$(uname -o)"
case "$uname_result" in
Msys | Cygwin | MS/Windows)
safeprint WINDOWS
;;
Linux | GNU/Linux | Android)
safeprint LINUX
;;
FreeBSD)
safeprint FREEBSD
;;
Darwin)
safeprint MACOS
;;
*)
error "unknown operating system: $uname_result"
;;
esac
}
function detect_arch {
local uname_result="$(uname -m)"
case "$uname_result" in
arm | arm32 | armhf | aarch32)
safeprint arm32
;;
arm64 | aarch64 | aarch64_be | armv8b | armv8l)
safeprint arm64
;;
x86 | i386 | i486 | i686)
safeprint x86
;;
x64 | x86_64 | amd64)
safeprint x64
;;
*)
error "unknown CPU architecture: $uname_result"
;;
esac
}

283
include/functions.sh Executable file
View File

@@ -0,0 +1,283 @@
#!/usr/bin/env bash
include "@cbuild/include/myprint.sh"
function exec_script_line {
local script="$1"
local line_num="$2"
#true or false
local quiet=$3
myprint_quiet $quiet "${BLUE}reading line $line_num from $script"
local line_str="$(sed $line_num'!d' $script)"
if [ -z "$line_str" ]; then
error "script line is empty"
fi
myprint_quiet $quiet "$line_str"
eval "$line_str"
}
function replace_var_value_in_script(){
local script="$1"
local var_name="$2"
local new_value="$3"
myprint "${BLUE}setting $var_name to ${CYAN}'$new_value' in '$script'"
cp "$script" "$script.tmp"
sed "s,$var_name=\".*\",$var_name=\"$new_value\",g" \
"$script.tmp" > "$script"
rm "$script.tmp"
}
function clean_dir {
local dir="$1"
myprint "${BLUE}cleaning ${WHITE}'$dir'"
rm -rf "$dir"
mkdir "$dir"
}
function delete_dir {
local dir="$1"
myprint "${BLUE}deleting ${WHITE}'$dir'"
rm -rf "$dir"
}
function try_delete_dir_or_file {
local path="$1"
if [ -f "$path" ] || [ -d "$path" ]; then
rm -rf "$path"
myprint "${BLUE}deleting ${WHITE}'$path'"
fi
}
file_copy_default_if_not_present(){
local file="$1"
local file_default="$2"
if [ ! -f "$file" ]; then
myprint "${YELLOW}creating default ${WHITE}'$file'"
cp -r "$file_default" "$file"
fi
}
function exec_command {
local command="$@"
if [ ! -z "$command" ]; then
myprint "${GRAY}$command"
$command || error "command returned eror"
fi
}
function load_dependency_config {
local dep_conf_file="$1"
myprint "${BLUE}loading dependency config ${WHITE}'${dep_conf_file}'"
include "$dep_conf_file"
}
# builds a dependency when $dep_out_files dont exist or rebuild task is executed
function build_dependency {
# path to *.config file
local dep_conf_file="$1"
# true or false
local force_build="$2"
load_dependency_config "$dep_conf_file"
local proj_root_dir="$(pwd)"
myprint "${BLUE}entering dependency directory '${DEP_WORKING_DIR}'"
cd "$DEP_WORKING_DIR"
local build_needed="$force_build"
if [ "$build_needed" != true ]; then
for file in $DEP_STATIC_OUT_FILES $DEP_DYNAMIC_OUT_FILES $DEP_OTHER_OUT_FILES; do
if [ ! -f "$file" ]; then
myprint "${GRAY}missing file '$file'"
local build_needed=true
fi
done
fi
if [ "$build_needed" = true ]; then
exec_command "$DEP_PRE_BUILD_COMMAND"
exec_command "$DEP_BUILD_COMMAND"
exec_command "$DEP_POST_BUILD_COMMAND"
myprint "${GRAY}dependency build finished"
else
myprint "${GRAY}dependency was built already"
fi
if [ ! -z "$DEP_DYNAMIC_OUT_FILES" ]; then
# copies each file to $OUTDIR and creates symbolic link in $OBJDIR/dynamic_libs
for file in $DEP_DYNAMIC_OUT_FILES; do
# doesnt return error if called not like this
real_file=$(realpath $file)
file_dir=$(dirname $file)
if [ "$PRESERVE_OUT_DIRECTORY_STRUCTURE" = true ] && [ "$file_dir" != '.' ]; then
mkdir -p "$proj_root_dir/$OUTDIR/$file_dir"
mkdir -p "$proj_root_dir/$OBJDIR/dynamic_libs/$file_dir"
cp -v -u --preserve=timestamps "$file" "$proj_root_dir/$OUTDIR/$file"
ln -sfv "$real_file" "$proj_root_dir/$OBJDIR/dynamic_libs/$file"
else
cp -v -u --preserve=timestamps "$file" "$proj_root_dir/$OUTDIR/"
ln -sfv "$real_file" "$proj_root_dir/$OBJDIR/dynamic_libs/"
fi
done
fi
if [ ! -z "$DEP_OTHER_OUT_FILES" ]; then
# copies each file to $OUTDIR
for file in $DEP_OTHER_OUT_FILES; do
if [ "$PRESERVE_OUT_DIRECTORY_STRUCTURE" = true ]; then
file_dir=$(dirname $file)
mkdir -p "$proj_root_dir/$OUTDIR/$file_dir"
cp -v -u --preserve=timestamps "$file" "$proj_root_dir/$OUTDIR/$file"
else
cp -v -u --preserve=timestamps "$file" "$proj_root_dir/$OUTDIR/"
fi
done
fi
if [ ! -z "$DEP_STATIC_OUT_FILES" ]; then
# creates symbolic link to each file in $OBJDIR/static_libs
for file in $DEP_STATIC_OUT_FILES; do
# doesnt return error if called not like this
real_file=$(realpath $file)
ln -sfv "$real_file" "$proj_root_dir/$OBJDIR/static_libs/"
done
fi
cd "$proj_root_dir"
# unsed all dependency config variables to not mess with next dependencies
unset DEP_WORKING_DIR
unset DEP_PRE_BUILD_COMMAND
unset DEP_BUILD_COMMAND
unset DEP_POST_BUILD_COMMAND
unset DEP_CLEAN_COMMAND
unset DEP_STATIC_OUT_FILES
unset PRESERVE_OUT_DIRECTORY_STRUCTURE
unset DEP_DYNAMIC_OUT_FILES
unset DEP_OTHER_OUT_FILES
}
function build_dependencies {
local dependencies="$1"
# true or false
local force_build="$2"
myprint "${BLUE}resolving dependencies"
clean_dir "$OBJDIR/static_libs"
clean_dir "$OBJDIR/dynamic_libs"
for dep in $dependencies; do
build_dependency "$DEPENDENCY_CONFIGS_DIR/$dep.config" "$force_build"
done
}
function compile {
build_dependencies "$ENABLED_DEPENDENCIES"
print_hline "${BLUE}" "─"
local cmp="$1"
local std="$2"
local warn="$3"
local args="$4"
local include="$5"
local sources="$6"
myprint "${BLUE}compiler: ${GRAY}$cmp"
myprint "${BLUE}standard: ${GRAY}$std"
myprint "${BLUE}warnings: ${GRAY}$warn"
myprint "${BLUE}args: ${GRAY}$args"
myprint "${BLUE}include dirs: ${GRAY}$include"
myprint "${BLUE}sources: ${GRAY}$sources"
for srcfile in $sources
do (
local object="$OBJDIR/objects/$(basename $srcfile).o"
if ! $($cmp -std=$std $warn $args $include -c -o $object $srcfile)
then
error "some error happened"
# TODO stop all threads
fi
) & done
wait
}
# (args, sources)
function compile_c {
local cmp="$1"
local std="$2"
local warn="$3"
local args="$4"
local include="$5"
local sources="$6"
print_header "${CYAN}" "─" "$PROJECT/$TASK/compile_c"
compile "$cmp" "$std" "$warn" "$args" "$include" "$sources"
}
# (args, sources)
function compile_cpp {
local cmp="$1"
local std="$2"
local warn="$3"
local args="$4"
local include="$5"
local sources="$6"
print_header "${CYAN}" "─" "$PROJECT/$TASK/compile_cpp"
compile "$cmp" "$std" "$warn" "$args" "$include" "$sources"
}
# (outfile)
function pack_static_lib {
print_header "${CYAN}" "─" "$PROJECT/$TASK/pack_static_lib"
local outfile="$1"
myprint "${BLUE}outfile: ${GRAY}$outfile"
local objects=$(find $OBJDIR/objects -type f,l | tr '\n' ' ')
myprint "${BLUE}objects: ${GRAY}$objects"
if [ -z "$objects" ]; then
error "no compiled objects found"
fi
local command="ar rcs $OUTDIR/$outfile $objects"
print_hline "${GRAY}" "─"
myprint "${GRAY}$command"
print_hline "${GRAY}" "─"
if $command
then
myprint "${GREEN}file $CYAN$outfile ${GREEN}created"
else
error "some error happened"
fi
clean_dir "$OBJDIR/objects"
}
function link {
print_header "${CYAN}" "─" "$PROJECT/$TASK/link"
local args="$1"
local outfile="$2"
myprint "${BLUE}args: ${GRAY}$args"
myprint "${BLUE}outfile: ${GRAY}$outfile"
local objects=$(find $OBJDIR/objects -type f,l | tr '\n' ' ')
myprint "${BLUE}objects: ${GRAY}$objects"
if [ -z "$objects" ]; then
error "no compiled objects found"
fi
local static_libs=$(find $OBJDIR/static_libs -type f,l | tr '\n' ' ')
myprint "${BLUE}static libraries: ${GRAY}$static_libs"
local dynamic_libs=$(find $OBJDIR/dynamic_libs -type f,l | tr '\n' ' '\
| sed "s,$OBJDIR/dynamic_libs/,,g")
myprint "${BLUE}dynamic libraries: ${GRAY}$dynamic_libs"
local dynamic_libs_args="-L./$OBJDIR/dynamic_libs -Wl,-Bdynamic"
for lib in $dynamic_libs; do
dynamic_libs_args="$dynamic_libs_args -Wl,-rpath=$(dirname $lib) -l:$lib"
done
local command="$CMP_CPP $objects $static_libs $args $dynamic_libs_args -o $OUTDIR/$outfile"
print_hline "${GRAY}" "─"
myprint "${GRAY}$command"
print_hline "${GRAY}" "─"
if $command
then
myprint "${GREEN}file $CYAN$outfile ${GREEN}created"
else
error "some error happened"
fi
clean_dir "$OBJDIR/objects"
}

97
include/myprint.sh Executable file
View File

@@ -0,0 +1,97 @@
#!/usr/bin/env bash
BLACK='\033[0;30m'
GRAY='\033[0;37m'
WHITE='\033[0;97m'
RED='\033[0;91m'
GREEN='\033[0;92m'
YELLOW='\033[0;93m'
BLUE='\033[0;94m'
PURPLE='\033[0;95m'
CYAN='\033[0;96m'
# prints text without special characters
# use it to return string values from functions
function safeprint {
printf "%s" "$@"
}
# prints text with special characters and resets color
function myprint {
printf "$@${GRAY}\n"
}
function myprint_quiet {
#true or false
local quiet=$1
local text="$2"
if [ "$quiet" != true ]; then
myprint "$text"
fi
}
# print message and exit
function error {
myprint "${RED}$@"
exit 1
}
# asks a question with two options and returns 0 or 1
# usage: `if ask_yn "do something?"`
function ask_yn {
local answ=""
myprint "$@ [y/n]"
read -r answ
return $([[ "$answ" = [Yy] ]]);
}
function char_multiply {
local character="$1"
local length="$2"
i=0
while [ $i -lt $length ]; do
printf $character
i=$((i+1))
done
}
# prints horizontal line occupying whole terminal row
# https://en.wikipedia.org/wiki/Box-drawing_characters
function print_hline {
local color="$1"
local character="$2"
if [ -z "$color" ]; then
color="${GRAY}"
fi
if [ -z "$character" ]; then
character="-";
fi
local term_width=$(tput cols)
local line_length=$((term_width - 1))
printf "${color}"
char_multiply "$character" $line_length
printf "${GRAY}\n"
}
# prints horizontal line occupying whole terminal row with a given label
function print_header {
local color="$1"
local character="$2"
local label="$3"
if [ -z "$color" ]; then
color="${GRAY}"
fi
if [ -z "$character" ]; then
character="-";
fi
local term_width=$(tput cols)
local label_length=${#label}
local line_characters_count=$((term_width - label_length - 2))
local left_line_length=$(( line_characters_count / 2 ))
local right_line_length=$(( left_line_length - 1 + line_characters_count % 2 ))
printf "${color}"
char_multiply "$character" $left_line_length
printf "[${label}]${color}"
char_multiply "$character" $right_line_length
printf "${GRAY}\n"
}

78
init.sh
View File

@@ -1,78 +0,0 @@
#!/bin/bash
tabs 4
# exit on errors
set -eo pipefail
source cbuild/colors.sh
source cbuild/functions.sh
source cbuild/detect_os.sh
# copying default config from cbuild if it not exists
if [ ! -f default.config ]; then
cp cbuild/default.config default.config
myprint "${YELLOW}Default config didn't exist, copied from cbuild."
fi
function exec_script_line {
local script="$1"
local line_num="$2"
myprint "${BLUE}reading line $line_num from $script"
local line_str="$(sed $line_num'!d' $script)"
myprint "$line_str"
eval "$line_str"
}
# getting version of cbuild installation
exec_script_line cbuild/default.config 2
INSTALLED_CBUILD_VERSION="$CBUILD_VERSION"
unset CBUILD_VERSION
# getting version of default config
exec_script_line default.config 3
DEFAULT_CONFIG_VERSION="$CONFIG_VERSION"
unset CONFIG_VERSION
# undefined task
[ -z "$TASK" ] && TASK="no_task"
# error on undefined
set -u
# reading current config or creating default
if [ ! -f current.config ]; then
myprint "${YELLOW}./current.config doesn't exist"
cp default.config current.config
myprint "${YELLOW}New config created from the default.\nEdit it."
exit
fi
myprint "${BLUE}reading ./current.config"
source current.config
myprint "${WHITE}project: ${CYAN}$PROJECT"
myprint "${WHITE}current.config cbuild version: ${CYAN}$CBUILD_VERSION"
myprint "${WHITE}installed cbuild version: ${CYAN}$INSTALLED_CBUILD_VERSION"
myprint "${WHITE}current.config version: ${CYAN}$CONFIG_VERSION"
myprint "${WHITE}default.config version: ${CYAN}$DEFAULT_CONFIG_VERSION"
# checking versions
if [ ! "$CBUILD_VERSION" -eq "$INSTALLED_CBUILD_VERSION" ]; then
error "config was created for outdated cbuild version"
fi
if [ ! "$CONFIG_VERSION" -eq "$DEFAULT_CONFIG_VERSION" ]; then
error "config version isn't correct"
fi
mkdir -p "$OUTDIR"
mkdir -p "$OBJDIR/libs"
mkdir -p "$OBJDIR/objects"
mkdir -p "$OBJDIR/profile"
# dont thorw error on undefined variable
set +u
myprint "${GREEN}cbuild initialized!"

203
project.config.default Executable file
View File

@@ -0,0 +1,203 @@
#!/usr/bin/env bash
CBUILD_VERSION=2.3.2
PROJECT="%PROJECT_NAME%"
CMP_C="gcc"
CMP_CPP="g++"
STD_C="c99"
STD_CPP="c++11"
WARN_C="-Wall -Wextra
-Wduplicated-branches
-Wduplicated-cond
-Wformat=2
-Wmissing-include-dirs
-Wshadow
-Werror=return-type
-Werror=pointer-arith
-Werror=init-self
-Werror=incompatible-pointer-types"
WARN_CPP="$WARN_C"
SRC_C="$(find src -name '*.c')"
SRC_CPP="$(find src -name '*.cpp')"
# Directory with dependency configs.
# See cbuild/example_dependency_configs
DEPENDENCY_CONFIGS_DIR='dependencies'
# List of dependency config files in DEPENDENCY_CONFIGS_DIR separated by space.
ENABLED_DEPENDENCIES=''
# OBJDIR structure:
# ├── objects/ - Compiled object files. Cleans on each call of build task
# ├── static_libs/ - Symbolic links to static libraries used by linker. Cleans on each call of build task.
# ├── static_libs/ - Symbolic links to dynamic libraries used by linker. Cleans on each call of build task.
# └── profile/ - gcc *.gcda profiling info files
OBJDIR="obj"
OUTDIR="bin"
STATIC_LIB_FILE="$PROJECT.a"
# example: "-I./include -I$DEPENDENCIES_DIR/libexample"
INCLUDE=""
# OS-specific options
case "$OS" in
WINDOWS)
EXEC_FILE="$PROJECT.exe"
SHARED_LIB_FILE="$PROJECT.dll"
INCLUDE="$INCLUDE "
# example: "-lSDL2 -lSDL2_image"
LINKER_LIBS=""
;;
LINUX)
EXEC_FILE="$PROJECT"
SHARED_LIB_FILE="$PROJECT.so"
INCLUDE="$INCLUDE "
LINKER_LIBS=""
;;
*)
error "operating system $OS has no configuration variants"
;;
esac
# TASKS
case "$TASK" in
# creates executable using profiling info if it exists
build_exec)
# -flto applies more optimizations across object files
# -flto=auto is needed to multithreaded copilation
# -fuse-linker-plugin is required to use static libs with lto
# -fprofile-use enables compiler to use profiling info files to optimize executable
# -fprofile-prefix-path sets path where profiling info about objects are be saved
# -fdata-sections -ffunction-sections -Wl,--gc-sections removes unused code
C_ARGS="-O2 -flto=auto -fuse-linker-plugin -fprofile-use -fprofile-prefix-path=$(realpath $OBJDIR)/objects -fdata-sections -ffunction-sections -Wl,--gc-sections"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
POST_TASK_SCRIPT="@cbuild/default_tasks/strip_exec.sh"
;;
# creates executable with debug info and no optimizations
build_exec_dbg)
C_ARGS="-O0 -g3"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
POST_TASK_SCRIPT=""
;;
# creates shared library
build_shared_lib)
C_ARGS="-O2 -fpic -flto -shared"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS -Wl,-soname,$SHARED_LIB_FILE"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_shared_lib.sh"
POST_TASK_SCRIPT=""
;;
# creates shared library with debug symbols and no optimizations
build_shared_lib_dbg)
C_ARGS="-O0 -g3 -fpic -shared"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS -Wl,-soname,$SHARED_LIB_FILE"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_shared_lib.sh"
POST_TASK_SCRIPT=""
;;
# creates static library
build_static_lib)
C_ARGS="-O2 -fpic -fdata-sections -ffunction-sections"
CPP_ARGS="$C_ARGS"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_static_lib.sh"
POST_TASK_SCRIPT=""
;;
# creates static library with debug symbols and no optimizations
build_static_lib_dbg)
C_ARGS="-O0 -g3"
CPP_ARGS="$C_ARGS"
PRE_TASK_SCRIPT=""
TASK_SCRIPT="@cbuild/default_tasks/build_static_lib.sh"
POST_TASK_SCRIPT=""
;;
# executes $EXEC_FILE
exec)
TASK_SCRIPT="@cbuild/default_tasks/exec.sh"
;;
# executes $EXEC_FILE with valgrind memory checker
valgrind)
VALGRIND_ARGS="-s --read-var-info=yes --track-origins=yes --fullpath-after=$(pwd)/ --leak-check=full --show-leak-kinds=all"
TASK_SCRIPT="@cbuild/default_tasks/valgrind.sh"
;;
# generates profiling info
profile)
OUTDIR="$OUTDIR/profile"
# -flto applies more optimizations across object files
# -flto=auto is needed to multithreaded copilation
# -fuse-linker-plugin is required to use static libs with lto
# -pg adds code to executable, that generates file containing function call info (gmon.out)
# -fprofile-generate generates executable with profiling code
# -fprofile-prefix-path sets path where profiling info about objects will be saved
C_ARGS="-O2 -flto=auto -fuse-linker-plugin -fprofile-generate -fprofile-prefix-path=$(realpath $OBJDIR)/objects"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
TASK_SCRIPT="@cbuild/default_tasks/profile.sh"
POST_TASK_SCRIPT=""
;;
# compiles program with -pg and runs it with gprof
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
gprof)
OUTDIR="$OUTDIR/gprof"
# arguments that emit some call counter code and disable optimizations to see function names
# https://github.com/msys2/MINGW-packages/issues/8503#issuecomment-1365475205
C_ARGS="-O0 -g -pg -no-pie -fno-omit-frame-pointer
-fno-inline-functions -fno-inline-functions-called-once
-fno-optimize-sibling-calls -fopenmp"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
TASK_SCRIPT="@cbuild/default_tasks/gprof.sh"
POST_TASK_SCRIPT=""
;;
# compiles program and runs it with callgrind (part of valgrind)
# uses gprof2dot python script to generate function call tree (pip install gprof2dot)
# requires graphviz (https://www.graphviz.org/download/source/)
# P.S. detailed results can be viewed in KCacheGrind
callgrind)
OUTDIR="$OUTDIR/callgrind"
# -pg adds code to executable, that generates file containing function call info (gmon.out)
C_ARGS="-O2 -flto=auto -fuse-linker-plugin"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
TASK_SCRIPT="@cbuild/default_tasks/callgrind.sh"
POST_TASK_SCRIPT=""
;;
# compiles executable with sanitizers and executes it to find errors and warnings
sanitize)
OUTDIR="$OUTDIR/sanitize"
C_ARGS="-O0 -g3 -fsanitize=undefined,address"
CPP_ARGS="$C_ARGS"
LINKER_ARGS="$CPP_ARGS $LINKER_LIBS"
PRE_TASK_SCRIPT="@cbuild/default_tasks/build_exec.sh"
TASK_SCRIPT="@cbuild/default_tasks/exec.sh"
POST_TASK_SCRIPT=""
;;
# rebuilds specified dependencies
# EXAMPLE: `cbuild rebuild_dependencies=libexample1,fonts`
# 'all' can be specified to rebuild all dependencies
rebuild_dependencies)
TASK_SCRIPT="@cbuild/default_tasks/rebuild_dependencies.sh"
;;
# deletes generated files
clean)
TASK_SCRIPT="@cbuild/default_tasks/clean.sh"
;;
# nothing to do
"" | no_task)
;;
# unknown task
*)
error "task <$PROJECT/$TASK> not found"
;;
esac

11
project.config.user.default Executable file
View File

@@ -0,0 +1,11 @@
#!/usr/bin/env bash
# Project user config is ignored by git.
# Here you can add variables that users might want to change
# on their local machine, without commiting to the repository.
# Directory where you install dependencies.
# Do not confuse with DEPENDENCY_CONFIGS_DIR
# Example:
# libexample source code is at `../libexample`, and dependency config
# that specifies how to build this lib is at `dependencies/libexample.config`
DEPENDENCIES_DIR=".."

View File

@@ -1,7 +0,0 @@
#!/bin/bash
source "cbuild/init.sh"
target_file="$1"
touch ".rebuild_$target_file.tmp"
rm -fv "$OBJDIR/libs/$target_file.a"
myprint "${YELLOW}dependency ${WHITE}$target_file ${YELLOW}will be rebuilt with the next build task"

97
setup.sh Normal file → Executable file
View File

@@ -1,32 +1,77 @@
#!/bin/bash #!/usr/bin/env bash
# USAGE: ./setup.sh install cbuild to /usr/local
# ./setup.sh --local install cbuild to $HOME/.local
# exit on errors # exit on errors
set -eo pipefail set -xeo pipefail
CBUILD_VERSION=$(<CBUILD_VERSION)
# help function version_parse {
if [ $# -eq 0 ] || [ "$1" = "h" ] || [ "$1" = "-h" ] || [ "$1" = "--help" ] || [ "$1" = "/?" ]; then local value="$1"
echo "usage: setup.sh [ submodule | standalone ]" var_name="$2"
echo " submodule - add to existing git repo as submodule" IFS_backup="$IFS"
echo " standalone - keep independent git repo" IFS='.'
local v_array=($value)
IFS="$IFS_backup"
eval ${var_name}_version_major=${v_array[0]}
eval ${var_name}_version_minor=${v_array[1]}
eval ${var_name}_version_patch=${v_array[2]}
}
function exec_script_line {
local script="$1"
local line_num="$2"
local line_str="$(sed $line_num'!d' $script)"
if [ -z "$line_str" ]; then
echo "script line is empty"
exit 1
fi
eval "$line_str"
}
version_parse $CBUILD_VERSION local
if [ -z "$CBUILD_INSTALL_DIR" ]; then
if [ "$1" = "--local" ]; then
if [ ! -d "$HOME" ]; then
echo "ERROR: home directory '$HOME' doesn't exist"
exit 1
fi
CBUILD_INSTALL_DIR="$HOME/.local/share/cbuild/${local_version_major}.${local_version_minor}"
CBUILD_BIN_DIR="$HOME/.local/bin"
mkdir -p "$HOME/.local"
mkdir -p "$HOME/.local/share"
mkdir -p "$HOME/.local/share/cbuild"
mkdir -p "$HOME/.local/bin"
else
CBUILD_INSTALL_DIR="/usr/local/share/cbuild/${local_version_major}.${local_version_minor}"
CBUILD_BIN_DIR="/usr/local/bin"
mkdir -p "/usr/local/share"
mkdir -p "/usr/local/share/cbuild"
mkdir -p "/usr/local/bin"
fi
fi fi
case "$1" in rm -rf "$CBUILD_INSTALL_DIR"
submodule) cp -r ./ "$CBUILD_INSTALL_DIR"
echo "mode - $1" rm -rf "$CBUILD_INSTALL_DIR/.git"
git submodule add ./cbuild ln -sf "$(realpath $CBUILD_INSTALL_DIR/cbuild.sh)" -T "$CBUILD_BIN_DIR/cbuild${local_version_major}.${local_version_minor}"
;;
standalone)
echo "mode - $1"
;;
*)
echo "invalid argument: $1"
exit -1
;;
esac
cp cbuild/default.Makefile Makefile bootstrap_install_path="$CBUILD_INSTALL_DIR/../bootstrap.sh"
cp cbuild/default.config ./ if [ -f "$bootstrap_install_path" ]; then
exec_script_line "$bootstrap_install_path" 2
echo "copy default .gitignore? [y/any]" installed_bootstrap_version=$CBUILD_BOOTSTRAP_VERSION
read answ installed_bootstrap_version_int=$(echo $installed_bootstrap_version | sed 's/\.//g')
[[ "$answ"="y" ]] && cp cbuild/.gitignore ./ exec_script_line "./bootstrap.sh" 2
local_bootstrap_version=$CBUILD_BOOTSTRAP_VERSION
local_bootstrap_version_int=$(echo $local_bootstrap_version | sed 's/\.//g')
if [[ $local_bootstrap_version_int > $installed_bootstrap_version_int ]]; then
echo "Found outdated bootstrap.sh in '$CBUILD_INSTALL_DIR', updating"
ln -sf "$(realpath $CBUILD_INSTALL_DIR/bootstrap.sh)" -T "$bootstrap_install_path"
fi
else
echo "Not found bootstrap.sh in '$CBUILD_INSTALL_DIR', installing"
ln -sf "$(realpath $CBUILD_INSTALL_DIR/bootstrap.sh)" -T "$bootstrap_install_path"
fi
echo "Link bootstrap.sh to $CBUILD_BIN_DIR"
ln -sf "$(realpath $bootstrap_install_path)" -T "$CBUILD_BIN_DIR/cbuild"