2020-11-12 17:39:19 +01:00
#!/bin/bash -e
# Copyright 2020 Google Inc. All rights reserved.
#
# 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.
# Generates NDK API txt file used by Mainline modules. NDK APIs would have value
# "UND" in Ndx column and have suffix "@LIB_NAME" in Name column.
# For example, current line llvm-readelf output is:
# 1: 00000000 0 FUNC GLOBAL DEFAULT UND dlopen@LIBC
# After the parse function below "dlopen" would be write to the output file.
2022-11-03 11:38:12 +01:00
2020-11-12 17:39:19 +01:00
printHelp( ) {
echo "**************************** Usage Instructions ****************************"
echo "This script is used to generate the Mainline modules used-by NDK symbols."
echo ""
echo "To run this script use: ./ndk_usedby_module.sh \$BINARY_IMAGE_DIRECTORY \$BINARY_LLVM_PATH \$OUTPUT_FILE_PATH"
echo "For example: If all the module image files that you would like to run is under directory '/myModule' and output write to /myModule.txt then the command would be:"
echo "./ndk_usedby_module.sh /myModule \$BINARY_LLVM_PATH /myModule.txt"
}
parseReadelfOutput( ) {
2022-11-03 11:38:12 +01:00
local readelfOutput = $1 ; shift
local ndkApisOutput = $1 ; shift
2020-11-12 17:39:19 +01:00
while IFS = read -r line
do
if [ [ $line = *FUNC*GLOBAL*UND*@* ] ] ;
then
2022-11-03 11:38:12 +01:00
echo " $line " | sed -r 's/.*UND (.*@.*)/\1/g' >> " ${ ndkApisOutput } "
2020-11-12 17:39:19 +01:00
fi
2022-11-03 11:38:12 +01:00
done < " ${ readelfOutput } "
echo "" >> " ${ ndkApisOutput } "
2020-11-12 17:39:19 +01:00
}
unzipJarAndApk( ) {
2022-11-03 11:38:12 +01:00
local dir = " $1 " ; shift
local tmpUnzippedDir = " $1 " ; shift
mkdir -p " ${ tmpUnzippedDir } "
find " $dir " -name "*.jar" -exec unzip -o { } -d " ${ tmpUnzippedDir } " \;
find " $dir " -name "*.apk" -exec unzip -o { } -d " ${ tmpUnzippedDir } " \;
find " ${ tmpUnzippedDir } " -name "*.MF" -exec rm { } \;
2020-11-12 17:39:19 +01:00
}
lookForExecFile( ) {
2022-11-03 11:38:12 +01:00
local dir = " $1 " ; shift
local readelf = " $1 " ; shift
local tmpOutput = " $1 " ; shift
find -L " $dir " -type f -name "*.so" -exec " ${ readelf } " --dyn-symbols { } >> " ${ tmpOutput } " \;
find -L " $dir " -type f -perm /111 ! -name "*.so" -exec " ${ readelf } " --dyn-symbols { } >> " ${ tmpOutput } " \;
2020-11-12 17:39:19 +01:00
}
if [ [ " $1 " = = "help" ] ]
then
printHelp
elif [ [ " $# " -ne 3 ] ]
then
echo "Wrong argument length. Expecting 3 argument representing image file directory, llvm-readelf tool path, output path."
else
2022-11-03 11:38:12 +01:00
imageDir = " $1 " ; shift
readelf = " $1 " ; shift
outputFile = " $1 " ; shift
tmpReadelfOutput = $( mktemp /tmp/temporary-file.XXXXXXXX)
tmpUnzippedDir = $( mktemp -d /tmp/temporary-dir.XXXXXXXX)
trap 'rm -rf -- "${tmpReadelfOutput}" "${tmpUnzippedDir}"' EXIT
# If there are any jars or apks, unzip them to surface native files.
unzipJarAndApk " ${ imageDir } " " ${ tmpUnzippedDir } "
# Analyze the unzipped files.
lookForExecFile " ${ tmpUnzippedDir } " " ${ readelf } " " ${ tmpReadelfOutput } "
# Analyze the apex image staging dir itself.
lookForExecFile " ${ imageDir } " " ${ readelf } " " ${ tmpReadelfOutput } "
[ [ -e " ${ outputFile } " ] ] && rm " ${ outputFile } "
parseReadelfOutput " ${ tmpReadelfOutput } " " ${ outputFile } "
fi