1#!/bin/sh
2# //===--------------------------- testit ---------------------------------===//
3# //
4# //                     The LLVM Compiler Infrastructure
5# //
6# // This file is distributed under the University of Illinois Open Source
7# // License. See LICENSE.TXT for details.
8# //
9# //===--------------------------------------------------------------------===//
10
11currentpath=`pwd`
12origpath=$currentpath
13currentdir=`basename $currentpath`
14while [ $currentdir != "test" ]; do
15	if [ $currentdir = "/" ]
16	then
17		echo "current directory must be in or under \"test\"."
18		exit 1
19	fi
20	cd ..
21	currentpath=`pwd`
22	currentdir=`basename $currentpath`
23done
24
25cd ..
26LIBCXX_ROOT=`pwd`
27cd $origpath
28
29VERBOSE=1
30
31run () {
32  if [ "$VERBOSE" -gt 1 ]; then
33    echo "COMMAND: $@"
34  fi
35  case $VERBOSE in
36    0|1)
37      # Hide command output and errors.
38      "$@" >/dev/null 2>&1
39      ;;
40    2)
41      # Only hide command output
42      "$@" >/dev/null
43      ;;
44    *)
45      # Show command output and errors.
46      "$@"
47      ;;
48  esac
49}
50
51run2 () {
52  if [ "$VERBOSE" -gt 2 ]; then
53    echo "COMMAND: $@"
54  fi
55  case $VERBOSE in
56    0|1)
57      # Hide command output and errors.
58      "$@" >/dev/null 2>&1
59      ;;
60    2)
61      # Only hide command output
62      "$@" >/dev/null
63      ;;
64    *)
65      # Show command output and errors.
66      "$@"
67      ;;
68  esac
69}
70
71# The list of valid target abis supported by this script.
72VALID_ABIS="armeabi armeabi-v7a x86 mips"
73
74HOST_OS=`uname -s`
75if [ "$HOST_OS" = "Darwin" ]; then
76    # darwin doesn't have readelf.  hard-code a known one
77    READELF=../../../../../toolchains/arm-linux-androideabi-4.8/prebuilt/darwin-x86/bin/arm-linux-androideabi-readelf
78else
79    READELF=readelf
80fi
81
82DO_HELP=
83DO_STATIC=
84TARGET_ABI=
85TARGET_ARCH=
86if [ -z "$TARGET_TEST_SUBDIR" ]; then
87    TARGET_TEST_SUBDIR="libcxx"
88fi
89TARGET_PATH=/data/local/tmp/$TARGET_TEST_SUBDIR
90CXX=
91WITH_COMPILER_RT=
92LIBCXX_NEEDS_COMPILER_RT=
93for OPT; do
94  case $OPT in
95    --help|-?)
96      DO_HELP=true
97      ;;
98    --abi=*)
99      TARGET_ABI=${OPT##--abi=}
100      ;;
101    --static)
102      DO_STATIC=true
103      ;;
104    --shared)
105      DO_STATIC=
106      ;;
107    --cxx=*)
108      CXX=${OPT##--cxx=}
109      ;;
110    --verbose)
111      VERBOSE=$(( $VERBOSE + 1 ))
112      ;;
113    --with-compiler-rt)
114      WITH_COMPILER_RT=yes
115      ;;
116    -*)
117      echo "Unknown option: $OPT. See --help."
118      exit 1
119      ;;
120    *)
121      echo "This script doesn't take parameters. See --help."
122      exit 1
123      ;;
124  esac
125done
126
127if [ "$DO_HELP" ]; then
128  echo \
129"Usage: $(basename $0) [options]
130
131This script is used to run the libc++ test suite for Android.
132You will need the following things:
133
134  - The prebuild libc++ libraries in your NDK install.
135  - A prebuilt Android toolchain in your path.
136  - The 'adb' tool in your path.
137  - An Android device connected to ADB.
138
139The toolchain and device must match your target ABI. For example, if
140you use --abi=armeabi-v7a, your device must run ARMv7-A Android binaries,
141and arm-linux-androideabi-g++ will be used to compile all tests, unless
142you use --cxx=<command> to override it.
143
144Valid options:
145  --help|-?        Display this message.
146  --abi=<name>     Specify target ABI. Use --abi=list for list.
147  --static         Link against static libc++ library.
148  --cxx=<program>  Override C++ compiler/linker.
149  --verbose        Increase verbosity.
150"
151  exit 0
152fi
153
154# Check target ABI.
155if [ "$TARGET_ABI" = "list" ]; then
156  echo "List of valid target ABIs:"
157  for ABI in $VALID_ABIS; do
158    printf " %s" $ABI
159  done
160  printf "\n"
161  exit 0
162fi
163
164if [ -z "$TARGET_ABI" ]; then
165  echo "ERROR: Please specify a target ABI (--abi=<name>)."
166  exit 1
167fi
168
169FOUND_ABI=
170for ABI in $VALID_ABIS; do
171  if [ "$ABI" = "$TARGET_ABI" ]; then
172    FOUND_ABI=true
173    break
174  fi
175done
176
177if [ -z "$FOUND_ABI" ]; then
178  echo "ERROR: Invalid abi '$TARGET_ABI'. Must be one of: $VALID_ABIS"
179  exit 1
180fi
181
182LIBCXX_LIBS=$(cd $LIBCXX_ROOT/.. && pwd)/libs/$TARGET_ABI
183for LIB in libc++_static.a libc++_shared.so; do
184  if [ ! -f "$LIBCXX_LIBS/$LIB" ]; then
185    echo "ERROR: Missing prebuilt library: $LIBCXX_LIBS/$LIB"
186    echo "Please run: build/tools/build-cxx-stl.sh --stl=libc++"
187    exit 1
188  fi
189done
190
191LIBCOMPILER_RT_LIBS=$(cd "$LIBCXX_ROOT"/../../../android/compiler-rt && pwd)/libs/$TARGET_ABI
192$READELF -d "$LIBCXX_LIBS/libc++_shared.so" | grep NEEDED | grep -q compiler_rt
193if [ $? = 0 ]; then
194  LIBCXX_NEEDS_COMPILER_RT=yes
195fi
196
197if [ "$WITH_COMPILER_RT" = "yes" -o "$LIBCXX_NEEDS_COMPILER_RT" = "yes" ]; then
198  for LIB in libcompiler_rt_static.a libcompiler_rt_shared.so; do
199    if [ ! -f "$LIBCOMPILER_RT_LIBS/$LIB" ]; then
200      echo "ERROR: Missing prebuilt library: $LIBCOMPILER_RT_LIBS/$LIB"
201      echo "Please run: build/tools/build-compiler-rt.sh --ndk-dir=$NDK \
202--src-dir=/tmp/ndk-$USER/src/llvm-3.3/compiler-rt --llvm-version=3.3"
203      exit 1
204    fi
205  done
206fi
207
208# Check or detect C++ toolchain.
209TOOLCHAIN_CFLAGS=
210TOOLCHAIN_LDFLAGS=
211if [ -z "$TOOLCHAIN_PREFIX" ]; then
212  # Compute
213  case $TARGET_ABI in
214    armeabi)
215      TOOLCHAIN_PREFIX=arm-linux-androideabi
216      TOOLCHAIN_CFLAGS="-mthumb"
217      TOOLCHAIN_LDFLAGS="-mthumb"
218      ;;
219    armeabi-v7a)
220      TOOLCHAIN_PREFIX=arm-linux-androideabi
221      TOOLCHAIN_CFLAGS="-march=armv7-a -mthumb -mfpu=vfpv3-d16"
222      TOOLCHAIN_LDFLAGS="-march=armv7-a -mthumb -Wl,--fix-cortex-a8"
223      ;;
224    armeabi-v7a-hard)
225      TOOLCHAIN_PREFIX=arm-linux-androideabi
226      TOOLCHAIN_CFLAGS="-march=armv7-a -mthumb -mfpu=vfpv3-d16 -mhard-float -D_NDK_MATH_NO_SOFTFP=1"
227      TOOLCHAIN_LDFLAGS="-march=armv7-a -mthumb -Wl,--fix-cortex-a8 -Wl,--no-warn-mismatch -lm_hard"
228      ;;
229    x86)
230      TOOLCHAIN_PREFIX=i686-linux-android
231      ;;
232    mips)
233      TOOLCHAIN_PREFIX=mipsel-linux-android
234      ;;
235    *)
236      echo "ERROR: Unknown ABI '$ABI'"
237      exit 1
238      ;;
239  esac
240  if [ -z "$CXX" ]; then
241      CXX=$TOOLCHAIN_PREFIX-g++
242  fi
243fi
244
245REAL_CXX=$(which "$CXX" 2>/dev/null)
246if [ -z "$REAL_CXX" ]; then
247  echo "ERROR: Missing C++ compiler: $CXX"
248  exit 1
249fi
250CC=$CXX
251
252if [ -z "$OPTIONS" ]
253then
254  # Note that some tests use assert() to check condition but -O2 defines assert() to nothing,
255  # unless we specify -UNDEBUG to bring assert() back.
256  OPTIONS="--std=c++11 -O2 -g -funwind-tables -UNDEBUG"
257  if [ "$CXX" != "${CXX%clang++*}" ] ; then
258    OPTIONS="$OPTIONS" # clang3.5 doesn't accept the following: -mllvm -arm-enable-ehabi-descriptors -mllvm -arm-enable-ehabi"
259  else
260    # Some libc++ implementation (eg. list) break TBAA!
261    OPTIONS="$OPTIONS -fno-strict-aliasing"
262  fi
263fi
264
265OPTIONS="$OPTIONS $TOOLCHAIN_CFLAGS $TOOLCHAIN_LDFLAGS"
266OPTIONS="$OPTIONS -I$LIBCXX_ROOT/test/support"
267# llvm-libc++/libcxx/test/lit.cfg line #278 defineds the following for testing only on Linux
268OPTIONS="$OPTIONS -D__STDC_FORMAT_MACROS -D__STDC_LIMIT_MACROS -D__STDC_CONSTANT_MACROS"
269
270if [ -z "$ADB" ]
271then
272  ADB=adb
273fi
274
275# Run a shell command through ADB, return its status.
276# Variable ERR contains output if $RET is non-zero
277adb_shell () {
278  # We need a temporary file to store the output of our command
279  local CMD_OUT RET
280  ERR=
281  CMD_OUT=$(mktemp /tmp/testit_android-cmdout-XXXXXX)
282  # Run the command, while storing the standard output to CMD_OUT
283  # and appending the exit code as the last line.
284  if [ "$VERBOSE" -gt 2 ]; then
285    echo "COMMAND: $ADB shell $@"
286  fi
287  $ADB shell "$@ ; echo \$?" | sed -e 's![[:cntrl:]]!!g' > $CMD_OUT 2>&1
288  # Get last line in log, which contains the exit code from the command
289  RET=$(sed -e '$!d' $CMD_OUT)
290  # Get output, which corresponds to everything except the last line
291  OUT=$(sed -e '$d' $CMD_OUT)
292  if [ "$RET" != "0" ]; then
293    ERR=$OUT
294  fi
295  if [ "$VERBOSE" -gt 2 ]; then
296    printf "%s" "$OUT"
297  fi
298  rm -f $CMD_OUT
299  return $RET
300}
301
302# Push a given file through ADB.
303# $1: File path
304adb_push () {
305  local FILE=$1
306  local FILE_BASENAME=$(basename "$FILE")
307  run2 $ADB push $FILE $TARGET_PATH/$FILE_BASENAME 2>/dev/null
308}
309
310# Run a given executable through ADB.
311# $1: Executable path
312adb_run () {
313  local EXECUTABLE=$1
314  local EXECUTABLE_BASENAME=$(basename "$EXECUTABLE")
315  run2 $ADB push $EXECUTABLE $TARGET_PATH/$EXECUTABLE_BASENAME 2>/dev/null
316  if [ "$?" != 0 ]; then
317    return 1;
318  fi
319  # Retry up to 10 times if fail is due to "Text file busy"
320  for i in 1 2 3 4 5 6 7 8 9 10; do
321    adb_shell "LD_LIBRARY_PATH=$TARGET_PATH; cd $TARGET_PATH; ./$EXECUTABLE_BASENAME"
322    if [ "$?" = "0" ]; then
323      return 0
324    fi
325    if ! $(echo $ERR | grep -iq "Text file busy"); then
326      if [ "$i" != "1" ]; then
327        # Dump error message to help diagnostics
328        echo "ERR=$ERR"
329      fi
330      break;
331    fi
332    echo "Text file busy.  Re-try $i"
333    sleep 1
334    run2 $ADB push $EXECUTABLE $TARGET_PATH/$EXECUTABLE_BASENAME 2>/dev/null
335    sleep 2  # try again
336  done
337  return 1
338}
339
340adb_shell "rm -rf $TARGET_PATH"
341adb_shell "mkdir -p $TARGET_PATH"
342
343if [ "$DO_STATIC" ]; then
344  # Statically link to ensure the executable can be run easily through ADB
345  # Note that in standalone toolchain libc++_static is renamed to libstdc++, and
346  # compiler adds -lstdc++ implicitly
347  if [ "$WITH_COMPILER_RT" = "yes" ]; then
348    LIBS="-nodefaultlibs -lstdc++ -latomic -ldl -lm -lc -lcompiler_rt_static"
349  else
350    LIBS="-latomic"
351  fi
352else
353  run2 $ADB push $LIBCXX_LIBS/libc++_shared.so $TARGET_PATH 2>/dev/null
354  if [ $? != 0 ]; then
355    echo "ERROR: Can't push shared libc++ to target device!"
356    exit 1
357  fi
358
359  if [ "$WITH_COMPILER_RT" = "yes" -o "$LIBCXX_NEEDS_COMPILER_RT" = "yes" ]; then
360    run2 $ADB push $LIBCOMPILER_RT_LIBS/libcompiler_rt_shared.so $TARGET_PATH 2>/dev/null
361    if [ $? != 0 ]; then
362      echo "ERROR: Can't push shared libcompiler_rt to target device!"
363      exit 1
364    fi
365  fi
366
367  if [ "$WITH_COMPILER_RT" = "yes" ]; then
368    LIBS="-nodefaultlibs -lc++_shared -latomic -ldl -lm -lc -lcompiler_rt_shared"
369  else
370    LIBS="-lc++_shared -latomic"
371  fi
372fi
373
374case $TRIPLE in
375  *-*-mingw* | *-*-cygwin* | *-*-win*)
376    TEST_EXE=test.exe
377    ;;
378  *)
379    TEST_EXE=a.out
380    ;;
381esac
382
383TEST_EXE=/tmp/testit_android-$USER-$$-$TEST_EXE
384
385FAIL=0
386PASS=0
387UNIMPLEMENTED=0
388IMPLEMENTED_FAIL=0
389IMPLEMENTED_PASS=0
390
391# Run tests in current directory, recursively
392#
393# Note that file path containing EQ are symlink to the existing tests whose path contain '=',
394# to workaround an issue in ndk-build which doesn't handle LOCAL_SRC_FILES with '='.
395# See tests/device/test-libc++-static-full/jni/Android.mk  We need to filter out path containing
396# EQ such that we don't run same tests twice
397#
398# An alternative is to do "find . -type f", but this doesn't work in NDK windows package
399# where zip turns symlink into physical file it points to.
400#
401# We also sort the test to make the test report comparable to previous test
402#
403
404afunc() {
405	fail=0
406	pass=0
407	if (ls *.fail.cpp > /dev/null 2>&1)
408	then
409		for FILE in $(ls *.fail.cpp | tr ' ' '\n' | grep -v EQ | sort); do
410			if run $CC $OPTIONS $HEADER_INCLUDE $SOURCE_LIB $FILE $LIBS -o $TEST_EXE > /dev/null 2>&1
411			then
412				rm $TEST_EXE
413				echo "$FILE should not compile"
414				fail=$(($fail+1))
415			else
416				pass=$(($pass+1))
417			fi
418		done
419	fi
420
421	if (ls *.pass.cpp > /dev/null 2>&1)
422	then
423		if (ls *.dat > /dev/null 2>&1)
424		then
425			adb_shell "rm -f $TARGET_PATH/*.dat"
426			for FILE in $(ls *.dat | tr ' ' '\n' | grep -v EQ | sort); do
427	                      if [ "$VERBOSE" -gt 1 ]; then
428	                          echo "Pushing data: " $FILE
429			      fi
430			      adb_push $FILE
431			      if [ $? != 0 ]; then
432				  echo "Failed to push file $FILE"
433	                      fi
434			done
435		fi
436		for FILE in $(ls *.pass.cpp | tr ' ' '\n' | grep -v EQ | sort); do
437                      if [ "$VERBOSE" -gt 1 ]; then
438                          echo "Running test: " $FILE
439                      fi
440                        COMMAND="( cd $(pwd) && $CC $OPTIONS $HEADER_INCLUDE $SOURCE_LIB $FILE $LIBS )"
441			if run $CC $OPTIONS $HEADER_INCLUDE $SOURCE_LIB $FILE $LIBS -o $TEST_EXE
442			then
443				if adb_run $TEST_EXE
444				then
445					rm $TEST_EXE
446					pass=$(($pass+1))
447				else
448					echo "`pwd`/$FILE failed at run time"
449					echo "Compile line was: $COMMAND # run-time"
450					fail=$(($fail+1))
451					rm $TEST_EXE
452				fi
453			else
454				echo "`pwd`/$FILE failed to compile"
455				echo "Compile line was: $COMMAND # compile-time"
456				fail=$(($fail+1))
457			fi
458		done
459	fi
460
461	if [ $fail -gt 0 ]
462	then
463		echo "failed $fail tests in `pwd`"
464		IMPLEMENTED_FAIL=$(($IMPLEMENTED_FAIL+1))
465	fi
466	if [ $pass -gt 0 ]
467	then
468		echo "passed $pass tests in `pwd`"
469		if [ $fail -eq 0 ]
470		then
471			IMPLEMENTED_PASS=$((IMPLEMENTED_PASS+1))
472		fi
473	fi
474	if [ $fail -eq 0 -a $pass -eq 0 ]
475	then
476		echo "not implemented:  `pwd`"
477		UNIMPLEMENTED=$(($UNIMPLEMENTED+1))
478	fi
479
480	FAIL=$(($FAIL+$fail))
481	PASS=$(($PASS+$pass))
482
483	for FILE in $(ls | tr ' ' '\n' | grep -v EQ | sort)
484	do
485		if [ -d "$FILE" ];
486		then
487			cd $FILE
488			afunc
489			cd ..
490		fi
491	done
492}
493
494afunc
495
496echo "****************************************************"
497echo "Results for `pwd`:"
498echo "using `$CC --version`"
499echo "with $OPTIONS $HEADER_INCLUDE $SOURCE_LIB"
500echo "----------------------------------------------------"
501echo "sections without tests   : $UNIMPLEMENTED"
502echo "sections with failures   : $IMPLEMENTED_FAIL"
503echo "sections without failures: $IMPLEMENTED_PASS"
504echo "                       +   ----"
505echo "total number of sections : $(($UNIMPLEMENTED+$IMPLEMENTED_FAIL+$IMPLEMENTED_PASS))"
506echo "----------------------------------------------------"
507echo "number of tests failed   : $FAIL"
508echo "number of tests passed   : $PASS"
509echo "                       +   ----"
510echo "total number of tests    : $(($FAIL+$PASS))"
511echo "****************************************************"
512
513exit $FAIL
514