annotate vext @ 53:f8110ba54f1b

Update vext
author Chris Cannam <c.cannam@qmul.ac.uk>
date Mon, 17 Jul 2017 16:07:46 +0100
parents ec5b5a9adac2
children
rev   line source
c@50 1 #!/bin/bash
c@50 2
c@50 3 # Disable shellcheck warnings for useless-use-of-cat. UUOC is good
c@50 4 # practice, not bad: clearer, safer, less error-prone.
c@50 5 # shellcheck disable=SC2002
c@50 6
c@50 7 sml="$VEXT_SML"
c@50 8
c@50 9 set -eu
c@50 10
c@50 11 mydir=$(dirname "$0")
c@50 12 program="$mydir/vext.sml"
c@50 13
c@53 14 hasher=
c@53 15 local_install=
c@53 16 if [ -w "$mydir" ]; then
c@53 17 if echo | sha256sum >/dev/null 2>&1 ; then
c@53 18 hasher=sha256sum
c@53 19 local_install=true
c@53 20 elif echo | shasum >/dev/null 2>&1 ; then
c@53 21 hasher=shasum
c@53 22 local_install=true
c@53 23 else
c@53 24 echo "WARNING: sha256sum or shasum program not found" 1>&2
c@53 25 fi
c@53 26 fi
c@53 27
c@53 28 if [ -n "$local_install" ]; then
c@53 29 hash=$(echo "$sml" | cat "$program" - | $hasher | cut -c1-16)
c@53 30 gen_sml=$mydir/.vext-$hash.sml
c@53 31 gen_out=$mydir/.vext-$hash.bin
c@53 32 trap 'rm -f $gen_sml' 0
c@53 33 else
c@53 34 gen_sml=$(mktemp /tmp/vext-XXXXXXXX.sml)
c@53 35 gen_out=$(mktemp /tmp/vext-XXXXXXXX.bin)
c@53 36 trap 'rm -f $gen_sml $gen_out' 0
c@53 37 fi
c@53 38
c@53 39 if [ -x "$gen_out" ]; then
c@53 40 exec "$gen_out" "$@"
c@53 41 fi
c@53 42
c@50 43 # We need one of Poly/ML, SML/NJ, or MLton. Since we're running a
c@50 44 # single-file SML program as if it were a script, our order of
c@53 45 # preference is based on startup speed, except in the local_install
c@53 46 # case where we retain a persistent binary.
c@50 47
c@50 48 if [ -z "$sml" ]; then
c@53 49 if [ -n "$local_install" ] && mlton 2>&1 | grep -q 'MLton'; then
c@53 50 sml="mlton"
c@53 51 elif sml -h 2>&1 | grep -q 'Standard ML of New Jersey'; then
c@50 52 sml="smlnj"
c@50 53 # We would prefer Poly/ML to SML/NJ, except that Poly v5.7 has a
c@50 54 # nasty bug that occasionally causes it to deadlock on startup.
c@50 55 # That appears to be fixed in their repo, so we could promote it
c@50 56 # up the order again at some point in future
c@50 57 elif echo | poly -v 2>/dev/null | grep -q 'Poly/ML'; then
c@50 58 sml="poly"
c@50 59 elif mlton 2>&1 | grep -q 'MLton'; then
c@50 60 sml="mlton"
c@50 61 else cat 1>&2 <<EOF
c@50 62
c@50 63 ERROR: No supported SML compiler or interpreter found
c@50 64 EOF
c@50 65 cat <<EOF
c@50 66
c@50 67 The Vext external source code manager needs a Standard ML (SML)
c@50 68 compiler or interpreter to run.
c@50 69
c@50 70 Please ensure you have one of the following SML implementations
c@50 71 installed and present in your PATH, and try again.
c@50 72
c@50 73 1. Standard ML of New Jersey
c@50 74 - often found in a distribution package called: smlnj
c@50 75 - executable name: sml
c@50 76
c@50 77 2. Poly/ML
c@50 78 - often found in a distribution package called: polyml
c@50 79 - executable name: poly
c@50 80
c@50 81 3. MLton
c@50 82 - often found in a distribution package called: mlton
c@50 83 - executable name: mlton
c@50 84
c@50 85 EOF
c@50 86 exit 2
c@50 87 fi
c@50 88 fi
c@50 89
c@50 90 arglist=""
c@50 91 for arg in "$@"; do
c@50 92 if [ -n "$arglist" ]; then arglist="$arglist,"; fi
c@50 93 if echo "$arg" | grep -q '[^a-z]' ; then
c@50 94 arglist="$arglist\"usage\""
c@50 95 else
c@50 96 arglist="$arglist\"$arg\""
c@50 97 fi
c@50 98 done
c@50 99
c@50 100 case "$sml" in
c@53 101 poly)
c@53 102 if [ -n "$local_install" ] && polyc --help >/dev/null 2>&1 ; then
c@53 103 if [ ! -x "$gen_out" ]; then
c@53 104 polyc -o "$gen_out" "$program"
c@53 105 fi
c@53 106 "$gen_out" "$@"
c@53 107 else
c@53 108 echo 'use "'"$program"'"; vext ['"$arglist"'];' |
c@53 109 poly -q --error-exit
c@53 110 fi ;;
c@50 111 mlton)
c@53 112 if [ ! -x "$gen_out" ]; then
c@53 113 echo "val _ = main ()" | cat "$program" - > "$gen_sml"
c@53 114 mlton -output "$gen_out" "$gen_sml"
c@53 115 fi
c@53 116 "$gen_out" "$@" ;;
c@50 117 smlnj)
c@50 118 cat "$program" | (
c@50 119 cat <<EOF
c@50 120 val smlrun__cp =
c@50 121 let val x = !Control.Print.out in
c@50 122 Control.Print.out := { say = fn _ => (), flush = fn () => () };
c@50 123 x
c@50 124 end;
c@50 125 val smlrun__prev = ref "";
c@50 126 Control.Print.out := {
c@50 127 say = fn s =>
c@50 128 (if String.isSubstring " Error" s
c@50 129 then (Control.Print.out := smlrun__cp;
c@50 130 (#say smlrun__cp) (!smlrun__prev);
c@50 131 (#say smlrun__cp) s)
c@50 132 else (smlrun__prev := s; ())),
c@50 133 flush = fn s => ()
c@50 134 };
c@50 135 EOF
c@50 136 cat -
c@50 137 cat <<EOF
c@50 138 val _ = vext [$arglist];
c@50 139 val _ = OS.Process.exit (OS.Process.success);
c@50 140 EOF
c@53 141 ) > "$gen_sml"
c@53 142 CM_VERBOSE=false sml "$gen_sml" ;;
c@50 143 *)
c@50 144 echo "Unknown SML implementation name: $sml";
c@50 145 exit 2 ;;
c@50 146 esac
c@50 147