| #!/bin/bash |
| # |
| # Copyright (C) 2009 The Android Open Source Project |
| # |
| # 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. |
| # |
| |
| # |
| # This script imports new versions of scrypt (http://www.tarsnap.com/scrypt/) into the |
| # Android source tree. To run, (1) fetch the appropriate tarball from the scrypt repository, |
| # (2) check the gpg/pgp signature, and then (3) run: |
| # ./import_scrypt.sh import scrypt-*.tar.gz |
| # |
| # IMPORTANT: See README.android for additional details. |
| |
| # turn on exit on error as well as a warning when it happens |
| set -e |
| set -x |
| trap "echo WARNING: Exiting on non-zero subprocess exit code" ERR; |
| |
| # Ensure consistent sorting order / tool output. |
| export LANG=C |
| export LC_ALL=C |
| |
| export DIRNAME=$(dirname $0) |
| |
| function die() { |
| declare -r message=$1 |
| |
| echo $message |
| exit 1 |
| } |
| |
| function usage() { |
| declare -r message=$1 |
| |
| if [ ! "$message" = "" ]; then |
| echo $message |
| fi |
| echo "Usage:" |
| echo " ./import_scrypt.sh import </path/to/scrypt-*.tar.gz>" |
| echo " ./import_scrypt.sh regenerate <patch/*.patch>" |
| echo " ./import_scrypt.sh generate <patch/*.patch> </path/to/scrypt-*.tar.gz>" |
| exit 1 |
| } |
| |
| function main() { |
| if [ ! -d patches ]; then |
| die "scrypt patch directory patches/ not found" |
| fi |
| |
| if [ ! -f scrypt.version ]; then |
| die "scrypt.version not found" |
| fi |
| |
| source $DIRNAME/scrypt.version |
| if [ "$SCRYPT_VERSION" == "" ]; then |
| die "Invalid scrypt.version; see README.android for more information" |
| fi |
| |
| SCRYPT_DIR=scrypt-$SCRYPT_VERSION |
| SCRYPT_DIR_ORIG=$SCRYPT_DIR.orig |
| |
| if [ ! -f scrypt.config ]; then |
| die "scrypt.config not found" |
| fi |
| |
| source $DIRNAME/scrypt.config |
| if [ "$CONFIGURE_ARGS" == "" -o "$UNNEEDED_SOURCES" == "" -o "$NEEDED_SOURCES" == "" ]; then |
| die "Invalid scrypt.config; see README.android for more information" |
| fi |
| |
| declare -r command=$1 |
| shift || usage "No command specified. Try import, regenerate, or generate." |
| if [ "$command" = "import" ]; then |
| declare -r tar=$1 |
| shift || usage "No tar file specified." |
| import $tar |
| elif [ "$command" = "regenerate" ]; then |
| declare -r patch=$1 |
| shift || usage "No patch file specified." |
| [ -d $SCRYPT_DIR ] || usage "$SCRYPT_DIR not found, did you mean to use generate?" |
| [ -d $SCRYPT_DIR_ORIG_ORIG ] || usage "$SCRYPT_DIR_ORIG not found, did you mean to use generate?" |
| regenerate $patch |
| elif [ "$command" = "generate" ]; then |
| declare -r patch=$1 |
| shift || usage "No patch file specified." |
| declare -r tar=$1 |
| shift || usage "No tar file specified." |
| generate $patch $tar |
| else |
| usage "Unknown command specified $command. Try import, regenerate, or generate." |
| fi |
| } |
| |
| # Compute the name of an assembly source file generated by one of the |
| # gen_asm_xxxx() functions below. The logic is the following: |
| # - if "$2" is not empty, output it directly |
| # - otherwise, change the file extension of $1 from .pl to .S and output |
| # it. |
| # Usage: default_asm_file "$1" "$2" |
| # or default_asm_file "$@" |
| # |
| # $1: generator path (perl script) |
| # $2: optional output file name. |
| function default_asm_file () { |
| if [ "$2" ]; then |
| echo "$2" |
| else |
| echo "${1%%.pl}.S" |
| fi |
| } |
| |
| # Generate an ARM assembly file. |
| # $1: generator (perl script) |
| # $2: [optional] output file name |
| function gen_asm_arm () { |
| local OUT |
| OUT=$(default_asm_file "$@") |
| perl "$1" > "$OUT" |
| } |
| |
| function gen_asm_mips () { |
| local OUT |
| OUT=$(default_asm_file "$@") |
| # The perl scripts expect to run the target compiler as $CC to determine |
| # the endianess of the target. Setting CC to true is a hack that forces the scripts |
| # to generate little endian output |
| CC=true perl "$1" o32 > "$OUT" |
| } |
| |
| function gen_asm_x86 () { |
| local OUT |
| OUT=$(default_asm_file "$@") |
| perl "$1" elf -fPIC > "$OUT" |
| } |
| |
| function gen_asm_x86_64 () { |
| local OUT |
| OUT=$(default_asm_file "$@") |
| perl "$1" elf "$OUT" > "$OUT" |
| } |
| |
| |
| # Filter all items in a list that match a given pattern. |
| # $1: space-separated list |
| # $2: egrep pattern. |
| # Out: items in $1 that match $2 |
| function filter_by_egrep() { |
| declare -r pattern=$1 |
| shift |
| echo "$@" | tr ' ' '\n' | grep -e "$pattern" | tr '\n' ' ' |
| } |
| |
| # Sort and remove duplicates in a space-separated list |
| # $1: space-separated list |
| # Out: new space-separated list |
| function uniq_sort () { |
| echo "$@" | tr ' ' '\n' | sort -u | tr '\n' ' ' |
| } |
| |
| function print_autogenerated_header() { |
| echo "# Auto-generated - DO NOT EDIT!" |
| echo "# To regenerate, edit scrypt.config, then run:" |
| echo "# ./import_scrypt.sh import /path/to/scrypt-$SCRYPT_VERSION.tar.gz" |
| echo "#" |
| } |
| |
| function generate_build_config_mk() { |
| ./configure $CONFIGURE_ARGS |
| #rm -f apps/CA.pl.bak crypto/scryptconf.h.bak |
| |
| declare -r tmpfile=$(mktemp) |
| (grep -e -D Makefile | grep -v CONFIGURE_ARGS= | grep -v OPTIONS=) > $tmpfile |
| |
| declare -r cflags=$(filter_by_egrep "^-D" $(grep -e "^CFLAG=" $tmpfile)) |
| declare -r depflags=$(filter_by_egrep "^-D" $(grep -e "^DEPFLAG=" $tmpfile)) |
| rm -f $tmpfile |
| |
| echo "Generating $(basename $1)" |
| ( |
| print_autogenerated_header |
| |
| echo "scrypt_cflags := \\" |
| for cflag in $cflags $depflags; do |
| echo " $cflag \\" |
| done |
| echo "" |
| ) > $1 |
| } |
| |
| # Return the value of a computed variable name. |
| # E.g.: |
| # FOO=foo |
| # BAR=bar |
| # echo $(var_value FOO_$BAR) -> prints the value of ${FOO_bar} |
| # $1: Variable name |
| # Out: variable value |
| var_value() { |
| # Note: don't use 'echo' here, because it's sensitive to values |
| # that begin with an underscore (e.g. "-n") |
| eval printf \"%s\\n\" \$$1 |
| } |
| |
| # Same as var_value, but returns sorted output without duplicates. |
| # $1: Variable name |
| # Out: variable value (if space-separated list, sorted with no duplicates) |
| var_sorted_value() { |
| uniq_sort $(var_value $1) |
| } |
| |
| # Print the definition of a given variable in a GNU Make build file. |
| # $1: Variable name (e.g. common_src_files) |
| # $2+: Variable value (e.g. list of sources) |
| print_vardef_in_mk() { |
| declare -r varname=$1 |
| shift |
| if [ -z "$1" ]; then |
| echo "$varname :=" |
| else |
| echo "$varname := \\" |
| for src; do |
| echo " $src \\" |
| done |
| fi |
| echo "" |
| } |
| |
| # Same as print_vardef_in_mk, but print a CFLAGS definition from |
| # a list of compiler defines. |
| # $1: Variable name (e.g. common_c_flags) |
| # $2: List of defines (e.g. SCRYPT_NO_DONKEYS ...) |
| print_defines_in_mk() { |
| declare -r varname=$1 |
| shift |
| if [ -z "$1" ]; then |
| echo "$varname :=" |
| else |
| echo "$varname := \\" |
| for def; do |
| echo " -D$def \\" |
| done |
| fi |
| echo "" |
| } |
| |
| # Generate a configuration file like Scrypt-config.mk |
| # This uses variable definitions from scrypt.config to build a config |
| # file that can compute the list of target- and host-specific sources / |
| # compiler flags for a given component. |
| # |
| # $1: Target file name. (e.g. Scrypt-config.mk) |
| function generate_config_mk() { |
| declare -r output="$1" |
| declare -r all_archs="arm arm_neon x86 x86_64 mips" |
| |
| echo "Generating $(basename $output)" |
| ( |
| print_autogenerated_header |
| echo \ |
| "# Before including this file, the local Android.mk must define the following |
| # variables: |
| # |
| # local_c_flags |
| # local_c_includes |
| # local_additional_dependencies |
| # |
| # This script will define the following variables: |
| # |
| # target_c_flags |
| # target_c_includes |
| # target_src_files |
| # |
| # host_c_flags |
| # host_c_includes |
| # host_src_files |
| # |
| |
| # Ensure these are empty. |
| unknown_arch_c_flags := |
| unknown_arch_src_files := |
| unknown_arch_exclude_files := |
| |
| " |
| common_defines=$(var_sorted_value SCRYPT_DEFINES) |
| print_defines_in_mk common_c_flags $common_defines |
| |
| common_sources=$(var_sorted_value SCRYPT_SOURCES) |
| print_vardef_in_mk common_src_files $common_sources |
| |
| common_includes=$(var_sorted_value SCRYPT_INCLUDES) |
| print_vardef_in_mk common_c_includes $common_includes |
| |
| for arch in $all_archs; do |
| arch_defines=$(var_sorted_value SCRYPT_DEFINES_${arch}) |
| print_defines_in_mk ${arch}_c_flags $arch_defines |
| |
| arch_sources=$(var_sorted_value SCRYPT_SOURCES_${arch}) |
| print_vardef_in_mk ${arch}_src_files $arch_sources |
| |
| arch_exclude_sources=$(var_sorted_value SCRYPT_SOURCES_EXCLUDES_${arch}) |
| print_vardef_in_mk ${arch}_exclude_files $arch_exclude_sources |
| |
| done |
| |
| echo "\ |
| target_arch := \$(TARGET_ARCH) |
| ifeq (\$(target_arch)-\$(TARGET_HAS_BIGENDIAN),mips-true) |
| target_arch := unknown_arch |
| endif |
| |
| target_c_flags := \$(common_c_flags) \$(\$(target_arch)_c_flags) \$(local_c_flags) |
| target_c_includes := \$(addprefix external/scrypt/,\$(common_c_includes)) \$(local_c_includes) |
| target_src_files := \$(common_src_files) \$(\$(target_arch)_src_files) |
| target_src_files := \$(filter-out \$(\$(target_arch)_exclude_files), \$(target_src_files)) |
| |
| # Hacks for ARM NEON support |
| ifeq (\$(target_arch),arm) |
| ifeq (\$(ARCH_ARM_HAVE_NEON),true) |
| target_c_flags += \$(arm_neon_c_flags) |
| target_src_files += \$(arm_neon_src_files) |
| target_src_files := \$(filter-out \$(arm_neon_exclude_files), \$(target_src_files)) |
| endif |
| endif |
| |
| ifeq (\$(HOST_OS)-\$(HOST_ARCH),linux-x86) |
| host_arch := x86 |
| else |
| host_arch := unknown_arch |
| endif |
| |
| host_c_flags := \$(common_c_flags) \$(\$(host_arch)_c_flags) \$(local_c_flags) |
| host_c_includes := \$(addprefix external/scrypt/,\$(common_c_includes)) \$(local_c_includes) |
| host_src_files := \$(common_src_files) \$(\$(host_arch)_src_files) |
| host_src_files := \$(filter-out \$(\$(host_arch)_exclude_files), \$(host_src_files)) |
| |
| local_additional_dependencies += \$(LOCAL_PATH)/$(basename $output) |
| " |
| |
| ) > "$output" |
| } |
| |
| function import() { |
| declare -r SCRYPT_SOURCE=$1 |
| |
| untar $SCRYPT_SOURCE readonly |
| applypatches $SCRYPT_DIR |
| |
| cd $SCRYPT_DIR |
| |
| generate_build_config_mk ../build-config.mk |
| |
| touch ../MODULE_LICENSE_BSD_LIKE |
| |
| cd .. |
| |
| generate_config_mk Scrypt-config.mk |
| |
| # Prune unnecessary sources |
| prune |
| |
| NEEDED_SOURCES="$NEEDED_SOURCES" |
| for i in $NEEDED_SOURCES; do |
| echo "Updating $i" |
| rm -r $i |
| mv $SCRYPT_DIR/$i . |
| done |
| |
| cleantar |
| } |
| |
| function regenerate() { |
| declare -r patch=$1 |
| |
| generatepatch $patch |
| } |
| |
| function generate() { |
| declare -r patch=$1 |
| declare -r SCRYPT_SOURCE=$2 |
| |
| untar $SCRYPT_SOURCE |
| applypatches $SCRYPT_DIR_ORIG $patch |
| prune |
| |
| for i in $NEEDED_SOURCES; do |
| echo "Restoring $i" |
| rm -r $SCRYPT_DIR/$i |
| cp -rf $i $SCRYPT_DIR/$i |
| done |
| |
| generatepatch $patch |
| cleantar |
| } |
| |
| # Find all files in a sub-directory that are encoded in ISO-8859 |
| # $1: Directory. |
| # Out: list of files in $1 that are encoded as ISO-8859. |
| function find_iso8859_files() { |
| find $1 -type f -print0 | xargs -0 file | fgrep "ISO-8859" | cut -d: -f1 |
| } |
| |
| # Convert all ISO-8859 files in a given subdirectory to UTF-8 |
| # $1: Directory name |
| function convert_iso8859_to_utf8() { |
| declare -r iso_files=$(find_iso8859_files "$1") |
| for iso_file in $iso_files; do |
| iconv --from-code iso-8859-1 --to-code utf-8 $iso_file > $iso_file.tmp |
| rm -f $iso_file |
| mv $iso_file.tmp $iso_file |
| done |
| } |
| |
| function untar() { |
| declare -r SCRYPT_SOURCE=$1 |
| declare -r readonly=$2 |
| |
| # Remove old source |
| cleantar |
| |
| # Process new source |
| tar -zxf $SCRYPT_SOURCE |
| convert_iso8859_to_utf8 $SCRYPT_DIR |
| cp -rfP $SCRYPT_DIR $SCRYPT_DIR_ORIG |
| if [ ! -z $readonly ]; then |
| find $SCRYPT_DIR_ORIG -type f -print0 | xargs -0 chmod a-w |
| fi |
| } |
| |
| function prune() { |
| echo "Removing $UNNEEDED_SOURCES" |
| (cd $SCRYPT_DIR_ORIG && rm -rf $UNNEEDED_SOURCES) |
| (cd $SCRYPT_DIR && rm -r $UNNEEDED_SOURCES) |
| } |
| |
| function cleantar() { |
| rm -rf $SCRYPT_DIR_ORIG |
| rm -rf $SCRYPT_DIR |
| } |
| |
| function applypatches () { |
| declare -r dir=$1 |
| declare -r skip_patch=$2 |
| |
| cd $dir |
| |
| # Apply appropriate patches |
| for i in $SCRYPT_PATCHES; do |
| if [ ! "$skip_patch" = "patches/$i" ]; then |
| echo "Applying patch $i" |
| patch -p1 --merge < ../patches/$i || die "Could not apply patches/$i. Fix source and run: $0 regenerate patches/$i" |
| else |
| echo "Skiping patch $i" |
| fi |
| |
| done |
| |
| # Cleanup patch output |
| find . \( -type f -o -type l \) -name "*.orig" -print0 | xargs -0 rm -f |
| |
| cd .. |
| } |
| |
| function generatepatch() { |
| declare -r patch=$1 |
| |
| # Cleanup stray files before generating patch |
| find $SCRYPT_DIR -type f -name "*.orig" -print0 | xargs -0 rm -f |
| find $SCRYPT_DIR -type f -name "*~" -print0 | xargs -0 rm -f |
| |
| declare -r variable_name=SCRYPT_PATCHES_`basename $patch .patch | sed s/-/_/`_SOURCES |
| # http://tldp.org/LDP/abs/html/ivr.html |
| eval declare -r sources=\$$variable_name |
| rm -f $patch |
| touch $patch |
| for i in $sources; do |
| LC_ALL=C TZ=UTC0 diff -aup $SCRYPT_DIR_ORIG/$i $SCRYPT_DIR/$i >> $patch && die "ERROR: No diff for patch $path in file $i" |
| done |
| echo "Generated patch $patch" |
| echo "NOTE To make sure there are not unwanted changes from conflicting patches, be sure to review the generated patch." |
| } |
| |
| main $@ |