Home | History | Annotate | Download | only in etc
      1 #!/bin/bash
      2 #
      3 # Copyright (C) 2008 The Android Open Source Project
      4 #
      5 # Licensed under the Apache License, Version 2.0 (the "License");
      6 # you may not use this file except in compliance with the License.
      7 # You may obtain a copy of the License at
      8 #
      9 #     http://www.apache.org/licenses/LICENSE-2.0
     10 #
     11 # Unless required by applicable law or agreed to in writing, software
     12 # distributed under the License is distributed on an "AS IS" BASIS,
     13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14 # See the License for the specific language governing permissions and
     15 # limitations under the License.
     16 
     17 # Stop if something fails.
     18 set -e
     19 
     20 function fail() {
     21   echo "$*" >&2
     22   exit 1
     23 }
     24 
     25 if [[ $# -le 0 ]]; then
     26   echo 'Error:' '$0 should have the parameters from the "build" script forwarded to it' >&2
     27   fail 'Error: An example of how do it correctly is ./default-build "$@"'
     28   exit 1
     29 fi
     30 
     31 # Set default values for directories.
     32 if [ -d smali ]; then
     33   HAS_SMALI=true
     34 else
     35   HAS_SMALI=false
     36 fi
     37 
     38 # .j files in jasmin get compiled into classes.jar
     39 if [ -d jasmin ]; then
     40   HAS_JASMIN=true
     41 else
     42   HAS_JASMIN=false
     43 fi
     44 
     45 if [ -d src ]; then
     46   HAS_SRC=true
     47 else
     48   HAS_SRC=false
     49 fi
     50 
     51 # .java files in src-art get compiled with libcore on the bootclasspath
     52 if [ -d src-art ]; then
     53   HAS_SRC_ART=true
     54 else
     55   HAS_SRC_ART=false
     56 fi
     57 
     58 if [ -d src2 ]; then
     59   HAS_SRC2=true
     60 else
     61   HAS_SRC2=false
     62 fi
     63 
     64 if [ -d src-multidex ]; then
     65   HAS_SRC_MULTIDEX=true
     66 else
     67   HAS_SRC_MULTIDEX=false
     68 fi
     69 
     70 if [ -d smali-multidex ]; then
     71   HAS_SMALI_MULTIDEX=true
     72 else
     73   HAS_SMALI_MULTIDEX=false
     74 fi
     75 
     76 # .j files in jasmin-multidex get compiled into classes2.jar
     77 if [ -d jasmin-multidex ]; then
     78   HAS_JASMIN_MULTIDEX=true
     79 else
     80   HAS_JASMIN_MULTIDEX=false
     81 fi
     82 
     83 if [ -d smali-ex ]; then
     84   HAS_SMALI_EX=true
     85 else
     86   HAS_SMALI_EX=false
     87 fi
     88 
     89 if [ -d src-ex ]; then
     90   HAS_SRC_EX=true
     91 else
     92   HAS_SRC_EX=false
     93 fi
     94 
     95 if [ -d src-dex2oat-unresolved ]; then
     96   HAS_SRC_DEX2OAT_UNRESOLVED=true
     97 else
     98   HAS_SRC_DEX2OAT_UNRESOLVED=false
     99 fi
    100 
    101 if [ -f hiddenapi-flags.csv ]; then
    102   HAS_HIDDENAPI_SPEC=true
    103 else
    104   HAS_HIDDENAPI_SPEC=false
    105 fi
    106 
    107 # USE_HIDDENAPI=false run-test... will disable hiddenapi.
    108 if [ -z "${USE_HIDDENAPI}" ]; then
    109   USE_HIDDENAPI=true
    110 fi
    111 
    112 # DESUGAR=false run-test... will disable desugaring.
    113 if [[ "$DESUGAR" == false ]]; then
    114   USE_DESUGAR=false
    115 fi
    116 
    117 # Allow overriding ZIP_COMPRESSION_METHOD with e.g. 'store'
    118 ZIP_COMPRESSION_METHOD="deflate"
    119 # Align every ZIP file made by calling $ZIPALIGN command?
    120 WITH_ZIP_ALIGN=false
    121 ZIP_ALIGN_BYTES="-1"
    122 
    123 BUILD_MODE="target"
    124 DEV_MODE="no"
    125 
    126 DEFAULT_EXPERIMENT="no-experiment"
    127 
    128 # The key for default arguments if no experimental things are enabled.
    129 EXPERIMENTAL=$DEFAULT_EXPERIMENT
    130 
    131 # Setup experimental API level mappings in a bash associative array.
    132 declare -A EXPERIMENTAL_API_LEVEL
    133 EXPERIMENTAL_API_LEVEL[${DEFAULT_EXPERIMENT}]="26"
    134 EXPERIMENTAL_API_LEVEL["default-methods"]="24"
    135 EXPERIMENTAL_API_LEVEL["parameter-annotations"]="25"
    136 EXPERIMENTAL_API_LEVEL["agents"]="26"
    137 EXPERIMENTAL_API_LEVEL["method-handles"]="26"
    138 EXPERIMENTAL_API_LEVEL["var-handles"]="28"
    139 
    140 while true; do
    141   if [ "x$1" = "x--no-src" ]; then
    142     HAS_SRC=false
    143     shift
    144   elif [ "x$1" = "x--no-src2" ]; then
    145     HAS_SRC2=false
    146     shift
    147   elif [ "x$1" = "x--no-src-multidex" ]; then
    148     HAS_SRC_MULTIDEX=false
    149     shift
    150   elif [ "x$1" = "x--no-smali-multidex" ]; then
    151     HAS_SMALI_MULTIDEX=false
    152     shift
    153   elif [ "x$1" = "x--no-src-ex" ]; then
    154     HAS_SRC_EX=false
    155     shift
    156   elif [ "x$1" = "x--no-smali" ]; then
    157     HAS_SMALI=false
    158     shift
    159   elif [ "x$1" = "x--no-jasmin" ]; then
    160     HAS_JASMIN=false
    161     shift
    162   elif [ "x$1" = "x--api-level" ]; then
    163     shift
    164     EXPERIMENTAL_API_LEVEL[${EXPERIMENTAL}]=$1
    165     shift
    166   elif [ "x$1" = "x--experimental" ]; then
    167     shift
    168     # We have a specific experimental configuration so don't use the default.
    169     EXPERIMENTAL="$1"
    170     shift
    171   elif [ "x$1" = "x--zip-compression-method" ]; then
    172     # Allow using different zip compression method, e.g. 'store'
    173     shift
    174     ZIP_COMPRESSION_METHOD="$1"
    175     shift
    176   elif [ "x$1" = "x--zip-align" ]; then
    177     # Align ZIP entries to some # of bytes.
    178     shift
    179     WITH_ZIP_ALIGN=true
    180     ZIP_ALIGN_BYTES="$1"
    181     shift
    182   elif [ "x$1" = "x--host" ]; then
    183     BUILD_MODE="host"
    184     shift
    185   elif [ "x$1" = "x--target" ]; then
    186     BUILD_MODE="target"
    187     shift
    188   elif [ "x$1" = "x--jvm" ]; then
    189     BUILD_MODE="jvm"
    190     shift
    191   elif [ "x$1" = "x--dev" ]; then
    192     DEV_MODE="yes"
    193     shift
    194   elif expr "x$1" : "x--" >/dev/null 2>&1; then
    195     fail "unknown $0 option: $1"
    196   else
    197     break
    198   fi
    199 done
    200 
    201 if [[ $BUILD_MODE == jvm ]]; then
    202   # Does not need desugaring on jvm because it supports the latest functionality.
    203   USE_DESUGAR=false
    204   # Do not attempt to build src-art directories on jvm, it would fail without libcore.
    205   HAS_SRC_ART=false
    206 fi
    207 
    208 # Set API level for smali and d8.
    209 API_LEVEL="${EXPERIMENTAL_API_LEVEL[${EXPERIMENTAL}]}"
    210 
    211 # Add API level arguments to smali and dx
    212 SMALI_ARGS="${SMALI_ARGS} --api $API_LEVEL"
    213 D8_FLAGS="${D8_FLAGS} --min-api $API_LEVEL"
    214 
    215 #########################################
    216 
    217 # Catch all commands to 'ZIP' and prepend extra flags.
    218 # Optionally, zipalign results to some alignment.
    219 function zip() {
    220   local zip_target="$1"
    221   local entry_src="$2"
    222   shift 2
    223 
    224   command zip --compression-method "$ZIP_COMPRESSION_METHOD" "$zip_target" "$entry_src" "$@"
    225 
    226   if "$WITH_ZIP_ALIGN"; then
    227     # zipalign does not operate in-place, so write results to a temp file.
    228     local tmp_file="$(mktemp)"
    229     "$ZIPALIGN" -f "$ZIP_ALIGN_BYTES" "$zip_target" "$tmp_file"
    230     # replace original zip target with our temp file.
    231     mv "$tmp_file" "$zip_target"
    232   fi
    233 }
    234 
    235 function make_jasmin() {
    236   local out_directory="$1"
    237   shift
    238   local jasmin_sources=("$@")
    239 
    240   mkdir -p "$out_directory"
    241 
    242   if [[ $DEV_MODE == yes ]]; then
    243     echo ${JASMIN} -d "$out_directory" "${jasmin_sources[@]}"
    244     ${JASMIN} -d "$out_directory" "${jasmin_sources[@]}"
    245   else
    246     ${JASMIN} -d "$out_directory" "${jasmin_sources[@]}" >/dev/null
    247   fi
    248 }
    249 
    250 # Like regular javac but may include libcore on the bootclasspath.
    251 function javac_with_bootclasspath {
    252   local helper_args="--mode=$BUILD_MODE"
    253 
    254   if [[ $DEV_MODE == yes ]]; then
    255     helper_args="$helper_args --show-commands"
    256   fi
    257 
    258   # build with libcore for host and target, or openjdk for jvm
    259   "$ANDROID_BUILD_TOP/art/tools/javac-helper.sh" --core-only $helper_args ${JAVAC_ARGS} "$@"
    260 }
    261 
    262 # Make a "dex" file given a directory of classes in $1. This will be
    263 # packaged in a jar file.
    264 function make_dex() {
    265   local name="$1"
    266   local d8_inputs=$(find $name -name '*.class' -type f)
    267   local d8_output=${name}.jar
    268   local dex_output=${name}.dex
    269   local d8_local_flags=""
    270   if [[ "$USE_DESUGAR" = "true" ]]; then
    271     local boot_class_path_list=$($ANDROID_BUILD_TOP/art/tools/bootjars.sh --$BUILD_MODE --core --path)
    272     for boot_class_path_element in $boot_class_path_list; do
    273       d8_local_flags="$d8_local_flags --classpath $boot_class_path_element"
    274     done
    275   else
    276     d8_local_flags="$d8_local_flags --no-desugaring"
    277   fi
    278   if [ "$DEV_MODE" = "yes" ]; then
    279     echo ${D8} ${D8_FLAGS} $d8_local_flags --output $d8_output $d8_inputs
    280   fi
    281   ${D8} ${D8_FLAGS} $d8_local_flags --output $d8_output $d8_inputs
    282 
    283   # D8 outputs to JAR files today rather than DEX files as DX used
    284   # to. To compensate, we extract the DEX from d8's output to meet the
    285   # expectations of make_dex callers.
    286   if [ "$DEV_MODE" = "yes" ]; then
    287     echo unzip -p $d8_output classes.dex \> $dex_output
    288   fi
    289   unzip -p $d8_output classes.dex > $dex_output
    290 }
    291 
    292 # Merge all the dex files in $1..$N into $1. Skip non-existing files, but at least 1 file must exist.
    293 function make_dexmerge() {
    294   # Dex file that acts as the destination.
    295   local dst_file="$1"
    296 
    297   # Dex files that act as the source.
    298   local dex_files_to_merge=()
    299 
    300   # Skip any non-existing files.
    301   while [[ $# -gt 0 ]]; do
    302     if [[ -e "$1" ]]; then
    303       dex_files_to_merge+=("$1")
    304     fi
    305     shift
    306   done
    307 
    308   # Skip merge if we are not merging anything. IE: input = output.
    309   if [[ "${#dex_files_to_merge[@]}" -eq "1" ]]; then
    310     local single_input=${dex_files_to_merge[0]}
    311     if [[ "$dst_file" != "$single_input" ]]; then
    312      mv "$single_input" "$dst_file";
    313      return
    314     fi
    315   fi
    316 
    317   # We assume the dexer did all the API level checks and just merge away.
    318   mkdir d8_merge_out
    319   ${DEXMERGER} --min-api 1000 --output ./d8_merge_out "${dex_files_to_merge[@]}"
    320 
    321   if [[ -e "./d8_merge_out/classes2.dex" ]]; then
    322     fail "Cannot merge all dex files into a single dex"
    323   fi
    324 
    325   mv ./d8_merge_out/classes.dex "$dst_file";
    326   rmdir d8_merge_out
    327 }
    328 
    329 function make_hiddenapi() {
    330   local args=( "encode" )
    331   while [[ $# -gt 0 ]]; do
    332     args+=("--input-dex=$1")
    333     args+=("--output-dex=$1")
    334     shift
    335   done
    336   args+=("--api-flags=hiddenapi-flags.csv")
    337   args+=("--no-force-assign-all")
    338   ${HIDDENAPI} "${args[@]}"
    339 }
    340 
    341 # Print the directory name only if it exists.
    342 function maybe_dir() {
    343   local dirname="$1"
    344   if [[ -d "$dirname" ]]; then
    345     echo "$dirname"
    346   fi
    347 }
    348 
    349 if [ -e classes.dex ]; then
    350   zip $TEST_NAME.jar classes.dex
    351   exit 0
    352 fi
    353 
    354 # Helper function for a common test. Evaluate with $(has_mutlidex).
    355 function has_multidex() {
    356   echo [ ${HAS_SRC_MULTIDEX} = "true" \
    357          -o ${HAS_JASMIN_MULTIDEX} = "true" \
    358          -o ${HAS_SMALI_MULTIDEX} = "true" ]
    359 }
    360 
    361 if [ ${HAS_SRC_DEX2OAT_UNRESOLVED} = "true" ]; then
    362   mkdir -p classes
    363   mkdir classes-ex
    364   javac_with_bootclasspath -implicit:none -sourcepath src-dex2oat-unresolved -d classes `find src -name '*.java'`
    365   javac_with_bootclasspath -implicit:none -sourcepath src -d classes-ex `find src-dex2oat-unresolved -name '*.java'`
    366   if [ ${NEED_DEX} = "true" ]; then
    367     make_dex classes-ex
    368     mv classes-ex.dex classes.dex   # rename it so it shows up as "classes.dex" in the zip file.
    369     zip ${TEST_NAME}-ex.jar classes.dex
    370     make_dex classes
    371   fi
    372 else
    373   if [ "${HAS_SRC}" = "true" ]; then
    374     mkdir -p classes
    375     javac_with_bootclasspath -implicit:none -classpath src-multidex -d classes `find src -name '*.java'`
    376   fi
    377 
    378   if [ "${HAS_SRC_ART}" = "true" ]; then
    379     mkdir -p classes
    380     javac_with_bootclasspath -implicit:none -classpath src-multidex -d classes `find src-art -name '*.java'`
    381   fi
    382 
    383   if [ "${HAS_SRC_MULTIDEX}" = "true" ]; then
    384     mkdir classes2
    385     javac_with_bootclasspath -implicit:none -classpath src -d classes2 `find src-multidex -name '*.java'`
    386     if [ ${NEED_DEX} = "true" ]; then
    387       make_dex classes2
    388     fi
    389   fi
    390 
    391   if [ "${HAS_SRC2}" = "true" ]; then
    392     mkdir -p classes
    393     javac_with_bootclasspath -classpath classes -d classes `find src2 -name '*.java'`
    394   fi
    395 
    396   # If the classes directory is not-empty, package classes in a DEX file. NB some
    397   # tests provide classes rather than java files.
    398   if [ "$(ls -A classes)" ]; then
    399     if [ ${NEED_DEX} = "true" ]; then
    400       make_dex classes
    401     fi
    402   fi
    403 fi
    404 
    405 if [[ "${HAS_JASMIN}" == true ]]; then
    406   # Compile Jasmin classes as if they were part of the classes.dex file.
    407   make_jasmin jasmin_classes $(find 'jasmin' -name '*.j')
    408   if [[ "${NEED_DEX}" == "true" ]]; then
    409     make_dex jasmin_classes
    410     make_dexmerge classes.dex jasmin_classes.dex
    411   else
    412     # Move jasmin classes into classes directory so that they are picked up with -cp classes.
    413     mkdir -p classes
    414     mv jasmin_classes/* classes
    415   fi
    416 fi
    417 
    418 if [ "${HAS_SMALI}" = "true" -a ${NEED_DEX} = "true" ]; then
    419   # Compile Smali classes
    420   ${SMALI} -JXmx512m assemble ${SMALI_ARGS} --output smali_classes.dex `find smali -name '*.smali'`
    421   if [[ ! -s smali_classes.dex ]] ; then
    422     fail "${SMALI} produced no output."
    423   fi
    424   # Merge smali files into classes.dex, this takes priority over any jasmin files.
    425   make_dexmerge classes.dex smali_classes.dex
    426 fi
    427 
    428 # Compile Jasmin classes in jasmin-multidex as if they were part of the classes2.jar
    429 if [[ "$HAS_JASMIN_MULTIDEX" == true ]]; then
    430   make_jasmin jasmin_classes2 $(find 'jasmin-multidex' -name '*.j')
    431 
    432   if [[ "${NEED_DEX}" == "true" ]]; then
    433     make_dex jasmin_classes2
    434     make_dexmerge classes2.dex jasmin_classes2.dex
    435   else
    436     # Move jasmin classes into classes2 directory so that they are picked up with -cp classes2.
    437     mkdir -p classes2
    438     mv jasmin_classes2/* classes2
    439   fi
    440 fi
    441 
    442 if [ "${HAS_SMALI_MULTIDEX}" = "true" -a ${NEED_DEX} = "true" ]; then
    443   # Compile Smali classes
    444   ${SMALI} -JXmx512m assemble ${SMALI_ARGS} --output smali_classes2.dex `find smali-multidex -name '*.smali'`
    445 
    446   # Merge smali_classes2.dex into classes2.dex
    447   make_dexmerge classes2.dex smali_classes2.dex
    448 fi
    449 
    450 if [ ${HAS_SRC_EX} = "true" ]; then
    451   # Build src-ex into classes-ex.
    452   # Includes 'src', 'src-art' source when compiling classes-ex, but exclude their .class files.
    453   if [[ "${HAS_SRC}" == "true" ]]; then
    454     mkdir -p classes-tmp-for-ex
    455     javac_with_bootclasspath -d classes-tmp-for-ex `find src -name '*.java'`
    456     src_tmp_for_ex="-cp classes-tmp-for-ex"
    457   fi
    458   if [[ "${HAS_SRC_ART}" == "true" ]]; then
    459     mkdir -p classes-tmp-for-ex
    460     javac_with_bootclasspath -d classes-tmp-for-ex `find src-art -name '*.java'`
    461     src_tmp_for_ex="-cp classes-tmp-for-ex"
    462   fi
    463   mkdir -p classes-ex
    464   javac_with_bootclasspath -d classes-ex $src_tmp_for_ex `find src-ex -name '*.java'`
    465 fi
    466 
    467 if [[ -d classes-ex ]] && [ ${NEED_DEX} = "true" ]; then
    468   make_dex classes-ex
    469 fi
    470 
    471 if [ "${HAS_SMALI_EX}" = "true" -a ${NEED_DEX} = "true" ]; then
    472   # Compile Smali classes
    473   ${SMALI} -JXmx512m assemble ${SMALI_ARGS} --output smali_classes-ex.dex `find smali-ex -name '*.smali'`
    474   if [[ ! -s smali_classes-ex.dex ]] ; then
    475     fail "${SMALI} produced no output."
    476   fi
    477   # Merge smali files into classes-ex.dex.
    478   make_dexmerge classes-ex.dex smali_classes-ex.dex
    479 fi
    480 
    481 if [[ -f classes-ex.dex ]]; then
    482   # Apply hiddenapi on the dex files if the test has API list file(s).
    483   if [ ${USE_HIDDENAPI} = "true" -a ${HAS_HIDDENAPI_SPEC} = "true" ]; then
    484     make_hiddenapi classes-ex.dex
    485   fi
    486 
    487   # quick shuffle so that the stored name is "classes.dex"
    488   mv classes.dex classes-1.dex
    489   mv classes-ex.dex classes.dex
    490   zip $TEST_NAME-ex.jar classes.dex
    491   mv classes.dex classes-ex.dex
    492   mv classes-1.dex classes.dex
    493 fi
    494 
    495 # Apply hiddenapi on the dex files if the test has API list file(s).
    496 if [ ${NEED_DEX} = "true" -a ${USE_HIDDENAPI} = "true" -a ${HAS_HIDDENAPI_SPEC} = "true" ]; then
    497   if $(has_multidex); then
    498     make_hiddenapi classes.dex classes2.dex
    499   else
    500     make_hiddenapi classes.dex
    501   fi
    502 fi
    503 
    504 # Create a single dex jar with two dex files for multidex.
    505 if [ ${NEED_DEX} = "true" ]; then
    506   if [ -f classes2.dex ] ; then
    507     zip $TEST_NAME.jar classes.dex classes2.dex
    508   else
    509     zip $TEST_NAME.jar classes.dex
    510   fi
    511 fi
    512