| #!/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. |
| |
| 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() { |
| local readelfOutput=$1; shift |
| local ndkApisOutput=$1; shift |
| while IFS= read -r line |
| do |
| if [[ $line = *FUNC*GLOBAL*UND*@* ]] ; |
| then |
| echo "$line" | sed -r 's/.*UND (.*@.*)/\1/g' >> "${ndkApisOutput}" |
| fi |
| done < "${readelfOutput}" |
| echo "" >> "${ndkApisOutput}" |
| } |
| |
| unzipJarAndApk() { |
| 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 {} \; |
| } |
| |
| lookForExecFile() { |
| 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}" \; |
| } |
| |
| 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 |
| 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 |