summaryrefslogtreecommitdiffstats
path: root/ship/build
blob: 1c6f2e758ced8045287bf2efc1f0941dc9ac3469 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#! /bin/sh
set -euf

## SYNOPSIS
# build compile SRCFILE DSTFILE
# build deps SRCFILE
build() {
  case "$1" in
    compile) build_compile "$2" "$3";;
    deps) build_deps "$2";;
    *) echo "build: $1: bad command" >&2; return 23;;
  esac
}

## build directives
build_info_directive='#@info'
build_include_directive='#@include \([0-9A-Za-z]\+\)'

## usage: build_compile SRCFILE DSTFILE
build_compile() {
  cp "$1" "$2"

  while needs_compilation "$2"; do
    script="$(make_sedscript_maker_shellscript "$2" | sh)"
    sed -n "$script" "$2" >"$2.tmp"
    mv "$2.tmp" "$2"
  done

  chmod +x "$2"
}

## usage: needs_compilation SRCFILE
# Returns true if SRCFILE contains compilation directives.
needs_compilation() {
  grep -q "^\\($build_include_directive\\|$build_info_directive\\)$" "$1"
}

## usage: make_sedscript_maker_shellscript SRCFILE
# Print a shellscript that creates a sedscript that resolves all the build
# directives in SRCFILE.
make_sedscript_maker_shellscript() {
  echo 'set -euf'

  echo "_build_info='$(
    echo "# This file was generated by //ship/build\\"
    echo "#   Date: $(date -u --rfc-3339=s)\\"
    echo "#   Git-Commit: $(git show --pretty=oneline | awk '{print$1}')"
  )'"

  deps="$(build_deps "$1")"
  for lib in $deps; do
    echo "_build_include_$(basename $lib)=$lib"
  done

  nl -b a -s ' ' "$1" |
  sed '
    s:^ *::

    s:^\([0-9]\+\) '"$build_info_directive"'$:\1a\\\\\
$_build_info\
      :

    s:^\([0-9]\+\) '"$build_include_directive"'$:\1a\\\\\
# BEGIN \2\
      \1r\$_build_include_\2\
      \1a\\\\\
# END \2:

    s:^\([0-9]\+\) .*:\1p:
  
    1s:^:echo ":
    $s:$:":
  '
}

## usage: build_deps SRCFILE
build_deps() {
  for libname in $(sed -n 's:^'"$build_include_directive"'$:\1:p' "$1"); do
    build_resolve "$libname"
  done
}

## usage: build_resolve LIBNAME
build_resolve() {
  echo "$BUILD_PATH" | tr : \\n |
    xargs -I: printf '%s/%s\n' : "$1" |
    xargs -I: ls -d : 2>/dev/null |
    head -n 1 |
    grep . ||
  {
    echo "build resolve: $1: library not found" >&2
    return 23
  }
}

build "$@"