| #include "rsCpuExecutable.h" | 
 | #include "rsCppUtils.h" | 
 |  | 
 | #include <fcntl.h> | 
 | #include <fstream> | 
 | #include <set> | 
 | #include <memory> | 
 |  | 
 | #include <sys/stat.h> | 
 |  | 
 | #ifdef RS_COMPATIBILITY_LIB | 
 | #include <stdio.h> | 
 | #else | 
 | #include "bcc/Config.h" | 
 | #endif | 
 |  | 
 | #include <unistd.h> | 
 | #include <dlfcn.h> | 
 | #include <android/dlext.h> | 
 | #include <sys/stat.h> | 
 |  | 
 | namespace android { | 
 | namespace renderscript { | 
 |  | 
 | namespace { | 
 |  | 
 | // Check if a path exists and attempt to create it if it doesn't. | 
 | [[maybe_unused]] | 
 | static bool ensureCacheDirExists(const char *path) { | 
 |     if (access(path, R_OK | W_OK | X_OK) == 0) { | 
 |         // Done if we can rwx the directory | 
 |         return true; | 
 |     } | 
 |     if (mkdir(path, 0700) == 0) { | 
 |         return true; | 
 |     } | 
 |     return false; | 
 | } | 
 |  | 
 | // Copy the file named \p srcFile to \p dstFile. | 
 | // Return 0 on success and -1 if anything wasn't copied. | 
 | [[maybe_unused]] | 
 | static int copyFile(const char *dstFile, const char *srcFile) { | 
 |     std::ifstream srcStream(srcFile); | 
 |     if (!srcStream) { | 
 |         ALOGE("Could not verify or read source file: %s", srcFile); | 
 |         return -1; | 
 |     } | 
 |     std::ofstream dstStream(dstFile); | 
 |     if (!dstStream) { | 
 |         ALOGE("Could not verify or write destination file: %s", dstFile); | 
 |         return -1; | 
 |     } | 
 |     dstStream << srcStream.rdbuf(); | 
 |     if (!dstStream) { | 
 |         ALOGE("Could not write destination file: %s", dstFile); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     srcStream.close(); | 
 |     dstStream.close(); | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | static std::string findSharedObjectName(const char *cacheDir, | 
 |                                         const char *resName, | 
 |                                         const bool reuse = true) { | 
 |     std::string scriptSOName(cacheDir); | 
 | #if defined(RS_COMPATIBILITY_LIB) && !defined(__LP64__) | 
 |     size_t cutPos = scriptSOName.rfind("cache"); | 
 |     if (cutPos != std::string::npos) { | 
 |         scriptSOName.erase(cutPos); | 
 |     } else { | 
 |         ALOGE("Found peculiar cacheDir (missing \"cache\"): %s", cacheDir); | 
 |     } | 
 |     scriptSOName.append("/lib/librs."); | 
 | #else | 
 |     scriptSOName.append("/librs."); | 
 | #endif // RS_COMPATIBILITY_LIB | 
 |     scriptSOName.append(resName); | 
 |     if (!reuse) { | 
 |         // If the generated shared library is not reused, e.g., with a debug | 
 |         // context or forced by a system property, multiple threads may read | 
 |         // and write the shared library at the same time. To avoid the race | 
 |         // on the generated shared library, delete it before finishing script | 
 |         // initialization. To avoid deleting a file generated by a regular | 
 |         // context, use a special suffix here. | 
 |         // Because the script initialization is guarded by a lock from the Java | 
 |         // API, it is safe to name this file with a consistent name and suffix | 
 |         // and delete it after loading. The same lock has also prevented write- | 
 |         // write races on the .so during script initialization even if reuse is | 
 |         // true. | 
 |         scriptSOName.append("#delete_after_load"); | 
 |     } | 
 |     scriptSOName.append(".so"); | 
 |  | 
 |     return scriptSOName; | 
 | } | 
 |  | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 | static bool isRunningInVndkNamespace() { | 
 |     static bool result = []() { | 
 |         Dl_info info; | 
 |         if (dladdr(reinterpret_cast<const void*>(&isRunningInVndkNamespace), &info) != 0) { | 
 |             std::string filename = std::string(info.dli_fname); | 
 |             return filename.find("/apex/com.android.vndk") != std::string::npos; | 
 |         } else { | 
 |             ALOGW("Can't determine whether this lib is running in vndk namespace or not. Assuming it is in vndk namespace."); | 
 |         } | 
 |         return true; | 
 |     }(); | 
 |     return result; | 
 | } | 
 | #endif | 
 |  | 
 | }  // anonymous namespace | 
 |  | 
 | const char* SharedLibraryUtils::LD_EXE_PATH = "/system/bin/ld.mc"; | 
 | const char* SharedLibraryUtils::RS_CACHE_DIR = "com.android.renderscript.cache"; | 
 |  | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 |  | 
 | bool SharedLibraryUtils::createSharedLibrary(const char *driverName, | 
 |                                              const char *cacheDir, | 
 |                                              const char *resName, | 
 |                                              const bool reuse, | 
 |                                              std::string *fullPath) { | 
 |     std::string sharedLibName = findSharedObjectName(cacheDir, resName, reuse); | 
 |     if (fullPath) { | 
 |         *fullPath = sharedLibName; | 
 |     } | 
 |     std::string objFileName = cacheDir; | 
 |     objFileName.append("/"); | 
 |     objFileName.append(resName); | 
 |     objFileName.append(".o"); | 
 |     // Should be something like "libRSDriver.so". | 
 |     std::string linkDriverName = driverName; | 
 |     // Remove ".so" and replace "lib" with "-l". | 
 |     // This will leave us with "-lRSDriver" instead. | 
 |     linkDriverName.erase(linkDriverName.length() - 3); | 
 |     linkDriverName.replace(0, 3, "-l"); | 
 |  | 
 |     static const std::string vndkLibCompilerRt = | 
 |         getVndkSysLibPath() + "/libcompiler_rt.so"; | 
 |     const char *compiler_rt = isRunningInVndkNamespace() ? | 
 |         vndkLibCompilerRt.c_str() : SYSLIBPATH "/libcompiler_rt.so"; | 
 |     const char *mTriple = "-mtriple=" DEFAULT_TARGET_TRIPLE_STRING; | 
 |     const char *libPath = "--library-path=" SYSLIBPATH; | 
 |     // vndk path is only added when RS framework is running in vndk namespace. | 
 |     // If we unconditionally add the vndk path to the library path, then RS | 
 |     // driver in the vndk-sp directory will always be used even for CPU fallback | 
 |     // case, where RS framework is loaded from the default namespace. | 
 |     static const std::string vndkLibPathString = | 
 |         "--library-path=" + getVndkSysLibPath(); | 
 |     const char *vndkLibPath = isRunningInVndkNamespace() ? | 
 |         vndkLibPathString.c_str() : ""; | 
 |     const char *vendorLibPath = "--library-path=" SYSLIBPATH_VENDOR; | 
 |  | 
 |     // The search path order should be vendor -> vndk -> system | 
 |     std::vector<const char *> args = { | 
 |         LD_EXE_PATH, | 
 |         "-shared", | 
 |         "-nostdlib", | 
 |         compiler_rt, mTriple, vendorLibPath, vndkLibPath, libPath, | 
 |         linkDriverName.c_str(), "-lm", "-lc", | 
 |         objFileName.c_str(), | 
 |         "-o", sharedLibName.c_str(), | 
 |         nullptr | 
 |     }; | 
 |  | 
 |     return rsuExecuteCommand(LD_EXE_PATH, args.size()-1, args.data()); | 
 |  | 
 | } | 
 |  | 
 | #endif  // RS_COMPATIBILITY_LIB | 
 |  | 
 | const char* RsdCpuScriptImpl::BCC_EXE_PATH = "/system/bin/bcc"; | 
 |  | 
 | void* SharedLibraryUtils::loadAndDeleteSharedLibrary(const char *fullPath) { | 
 |     void *loaded = dlopen(fullPath, RTLD_NOW | RTLD_LOCAL); | 
 |     if (loaded == nullptr) { | 
 |         ALOGE("Unable to open shared library (%s): %s", fullPath, dlerror()); | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     int r = unlink(fullPath); | 
 |     if (r != 0) { | 
 |         ALOGE("Could not unlink copy %s", fullPath); | 
 |         return nullptr; | 
 |     } | 
 |     return loaded; | 
 | } | 
 |  | 
 | void* SharedLibraryUtils::loadSharedLibrary(const char *cacheDir, | 
 |                                             const char *resName, | 
 |                                             const char *nativeLibDir, | 
 |                                             bool* alreadyLoaded) { | 
 |     void *loaded = nullptr; | 
 |  | 
 | #if defined(RS_COMPATIBILITY_LIB) && defined(__LP64__) | 
 |     std::string scriptSOName = findSharedObjectName(nativeLibDir, resName); | 
 | #else | 
 |     std::string scriptSOName = findSharedObjectName(cacheDir, resName); | 
 | #endif | 
 |  | 
 |     // We should check if we can load the library from the standard app | 
 |     // location for shared libraries first. | 
 |     loaded = loadSOHelper(scriptSOName.c_str(), cacheDir, resName, alreadyLoaded); | 
 |  | 
 |     if (loaded != nullptr) { | 
 |        return loaded; | 
 |     } | 
 |     ALOGE("Unable to open shared library (%s): %s", scriptSOName.c_str(), dlerror()); | 
 |  | 
 | #ifdef RS_COMPATIBILITY_LIB | 
 |     // Re-trying without absolute path. | 
 |     // For RS support lib, the shared object may not be extracted from the apk. | 
 |     // In order to access that, we need to load the library without specifying | 
 |     // the absolute path. | 
 |     std::string scriptSONameApk("librs."); | 
 |     scriptSONameApk.append(resName); | 
 |     scriptSONameApk.append(".so"); | 
 |     loaded = loadSOHelper(scriptSONameApk.c_str(), cacheDir, resName); | 
 |     if (loaded != nullptr) { | 
 |         return loaded; | 
 |     } | 
 |     ALOGE("Unable to open APK shared library (%s): %s", scriptSONameApk.c_str(), dlerror()); | 
 |  | 
 |     // One final attempt to find the library in "/system/lib". | 
 |     // We do this to allow bundled applications to use the compatibility | 
 |     // library fallback path. Those applications don't have a private | 
 |     // library path, so they need to install to the system directly. | 
 |     // Note that this is really just a testing path. | 
 |     std::string scriptSONameSystem("/system/lib/librs."); | 
 |     scriptSONameSystem.append(resName); | 
 |     scriptSONameSystem.append(".so"); | 
 |     loaded = loadSOHelper(scriptSONameSystem.c_str(), cacheDir, resName); | 
 |     if (loaded == nullptr) { | 
 |         ALOGE("Unable to open system shared library (%s): %s", | 
 |               scriptSONameSystem.c_str(), dlerror()); | 
 |     } | 
 | #endif | 
 |  | 
 |     return loaded; | 
 | } | 
 |  | 
 | std::string SharedLibraryUtils::getRandomString(size_t len) { | 
 |     char buf[len + 1]; | 
 |     for (size_t i = 0; i < len; i++) { | 
 |         uint32_t r = arc4random() & 0xffff; | 
 |         r %= 62; | 
 |         if (r < 26) { | 
 |             // lowercase | 
 |             buf[i] = 'a' + r; | 
 |         } else if (r < 52) { | 
 |             // uppercase | 
 |             buf[i] = 'A' + (r - 26); | 
 |         } else { | 
 |             // Use a number | 
 |             buf[i] = '0' + (r - 52); | 
 |         } | 
 |     } | 
 |     buf[len] = '\0'; | 
 |     return std::string(buf); | 
 | } | 
 |  | 
 | static void* loadAsCopy(const char *origName, std::string newName) { | 
 |     void *loaded = nullptr; | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 |     int fd = TEMP_FAILURE_RETRY(open(origName, O_RDONLY | O_CLOEXEC)); | 
 |     if (fd == -1) { | 
 |         ALOGE("Unable to open original file %s: %s", origName, strerror(errno)); | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     android_dlextinfo extinfo; | 
 |     memset(&extinfo, 0, sizeof(extinfo)); | 
 |     extinfo.flags = ANDROID_DLEXT_USE_LIBRARY_FD | ANDROID_DLEXT_FORCE_LOAD; | 
 |     extinfo.library_fd = fd; | 
 |  | 
 |     loaded = android_dlopen_ext(newName.c_str(), RTLD_NOW | RTLD_LOCAL, &extinfo); | 
 |     close(fd); | 
 | #else | 
 |     int r = copyFile(newName.c_str(), origName); | 
 |     if (r != 0) { | 
 |         ALOGE("Could not create copy %s -> %s", origName, newName.c_str()); | 
 |         return nullptr; | 
 |     } | 
 |     loaded = dlopen(newName.c_str(), RTLD_NOW | RTLD_LOCAL); | 
 |     r = unlink(newName.c_str()); | 
 |     if (r != 0) { | 
 |         ALOGE("Could not unlink copy %s", newName.c_str()); | 
 |     } | 
 | #endif  // RS_COMPATIBILITY_LIB | 
 |     return loaded; | 
 | } | 
 |  | 
 | void* SharedLibraryUtils::loadSOHelper(const char *origName, const char *cacheDir, | 
 |                                        const char *resName, bool *alreadyLoaded) { | 
 |     // Keep track of which .so libraries have been loaded. Once a library is | 
 |     // in the set (per-process granularity), we must instead make a copy of | 
 |     // the original shared object (randomly named .so file) and load that one | 
 |     // instead. If we don't do this, we end up aliasing global data between | 
 |     // the various Script instances (which are supposed to be completely | 
 |     // independent). | 
 |     static std::set<std::string> LoadedLibraries; | 
 |  | 
 |     void *loaded = nullptr; | 
 |  | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 |     // Skip everything if we don't even have the original library available. | 
 |     if (access(origName, F_OK) != 0) { | 
 |         return nullptr; | 
 |     } | 
 | #endif  // RS_COMPATIBILITY_LIB | 
 |  | 
 |     // Common path is that we have not loaded this Script/library before. | 
 |     if (LoadedLibraries.find(origName) == LoadedLibraries.end()) { | 
 |         if (alreadyLoaded != nullptr) { | 
 |             *alreadyLoaded = false; | 
 |         } | 
 |         loaded = dlopen(origName, RTLD_NOW | RTLD_LOCAL); | 
 |         if (loaded) { | 
 |             LoadedLibraries.insert(origName); | 
 |         } | 
 |         return loaded; | 
 |     } | 
 |  | 
 |     if (alreadyLoaded != nullptr) { | 
 |         *alreadyLoaded = true; | 
 |     } | 
 |  | 
 |     std::string newName(cacheDir); | 
 |  | 
 |     // Append RS_CACHE_DIR only if it is not found in cacheDir | 
 |     // In driver mode, RS_CACHE_DIR is already appended to cacheDir. | 
 |     if (newName.find(RS_CACHE_DIR) == std::string::npos) { | 
 |         newName.append("/"); | 
 |         newName.append(RS_CACHE_DIR); | 
 |         newName.append("/"); | 
 |     } | 
 |  | 
 |     if (!ensureCacheDirExists(newName.c_str())) { | 
 |         ALOGE("Could not verify or create cache dir: %s", cacheDir); | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     // Construct an appropriately randomized filename for the copy. | 
 |     newName.append("librs."); | 
 |     newName.append(resName); | 
 |     newName.append("#"); | 
 |     newName.append(getRandomString(6).c_str());  // 62^6 potential filename variants. | 
 |     newName.append(".so"); | 
 |  | 
 |     loaded = loadAsCopy(origName, newName); | 
 |  | 
 |     if (loaded) { | 
 |         LoadedLibraries.insert(newName.c_str()); | 
 |     } | 
 |  | 
 |     return loaded; | 
 | } | 
 |  | 
 | // MAXLINESTR must be compatible with operator '#' in C macro. | 
 | #define MAXLINESTR 499 | 
 | // MAXLINE must be (MAXLINESTR + 1), representing the size of a C string | 
 | // containing MAXLINESTR non-null chars plus a null. | 
 | #define MAXLINE (MAXLINESTR + 1) | 
 | #define MAKE_STR_HELPER(S) #S | 
 | #define MAKE_STR(S) MAKE_STR_HELPER(S) | 
 | #define EXPORT_VAR_STR "exportVarCount: " | 
 | #define EXPORT_FUNC_STR "exportFuncCount: " | 
 | #define EXPORT_FOREACH_STR "exportForEachCount: " | 
 | #define EXPORT_REDUCE_STR "exportReduceCount: " | 
 | #define OBJECT_SLOT_STR "objectSlotCount: " | 
 | #define PRAGMA_STR "pragmaCount: " | 
 | #define THREADABLE_STR "isThreadable: " | 
 | #define CHECKSUM_STR "buildChecksum: " | 
 | #define VERSIONINFO_STR "versionInfo: " | 
 |  | 
 | // Copy up to a newline or size chars from str -> s, updating str | 
 | // Returns s when successful and nullptr when '\0' is finally reached. | 
 | static char* strgets(char *s, int size, const char **ppstr) { | 
 |     if (!ppstr || !*ppstr || **ppstr == '\0' || size < 1) { | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     int i; | 
 |     for (i = 0; i < (size - 1); i++) { | 
 |         s[i] = **ppstr; | 
 |         (*ppstr)++; | 
 |         if (s[i] == '\0') { | 
 |             return s; | 
 |         } else if (s[i] == '\n') { | 
 |             s[i+1] = '\0'; | 
 |             return s; | 
 |         } | 
 |     } | 
 |  | 
 |     // size has been exceeded. | 
 |     s[i] = '\0'; | 
 |  | 
 |     return s; | 
 | } | 
 |  | 
 | // Creates a duplicate of a string. The new string is as small as possible, | 
 | // only including characters up to and including the first null-terminator; | 
 | // otherwise, the new string will be the same size as the input string. | 
 | // The code that calls duplicateString is responsible for the new string's | 
 | // lifetime, and is responsible for freeing it when it is no longer needed. | 
 | static char* duplicateString(const char *str, size_t length) { | 
 |     const size_t newLen = strnlen(str, length-1) + 1; | 
 |     char *newStr = new char[newLen]; | 
 |     strlcpy(newStr, str, newLen); | 
 |     return newStr; | 
 | } | 
 |  | 
 | ScriptExecutable* ScriptExecutable::createFromSharedObject( | 
 |     void* sharedObj, uint32_t expectedChecksum) { | 
 |     char line[MAXLINE]; | 
 |  | 
 |     size_t varCount = 0; | 
 |     size_t funcCount = 0; | 
 |     size_t forEachCount = 0; | 
 |     size_t reduceCount = 0; | 
 |     size_t objectSlotCount = 0; | 
 |     size_t pragmaCount = 0; | 
 |     bool isThreadable = true; | 
 |  | 
 |     void** fieldAddress = nullptr; | 
 |     bool* fieldIsObject = nullptr; | 
 |     char** fieldName = nullptr; | 
 |     InvokeFunc_t* invokeFunctions = nullptr; | 
 |     ForEachFunc_t* forEachFunctions = nullptr; | 
 |     uint32_t* forEachSignatures = nullptr; | 
 |     ReduceDescription* reduceDescriptions = nullptr; | 
 |     const char ** pragmaKeys = nullptr; | 
 |     const char ** pragmaValues = nullptr; | 
 |     uint32_t checksum = 0; | 
 |  | 
 |     const char *rsInfo = (const char *) dlsym(sharedObj, kRsInfo); | 
 |     int numEntries = 0; | 
 |     const int *rsGlobalEntries = (const int *) dlsym(sharedObj, kRsGlobalEntries); | 
 |     const char **rsGlobalNames = (const char **) dlsym(sharedObj, kRsGlobalNames); | 
 |     const void **rsGlobalAddresses = (const void **) dlsym(sharedObj, kRsGlobalAddresses); | 
 |     const size_t *rsGlobalSizes = (const size_t *) dlsym(sharedObj, kRsGlobalSizes); | 
 |     const uint32_t *rsGlobalProperties = (const uint32_t *) dlsym(sharedObj, kRsGlobalProperties); | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         return nullptr; | 
 |     } | 
 |     if (sscanf(line, EXPORT_VAR_STR "%zu", &varCount) != 1) { | 
 |         ALOGE("Invalid export var count!: %s", line); | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     fieldAddress = new void*[varCount]; | 
 |     if (fieldAddress == nullptr) { | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     fieldIsObject = new bool[varCount]; | 
 |     if (fieldIsObject == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     fieldName = new char*[varCount]; | 
 |     if (fieldName == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     for (size_t i = 0; i < varCount; ++i) { | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             goto error; | 
 |         } | 
 |         char *c = strrchr(line, '\n'); | 
 |         if (c) { | 
 |             *c = '\0'; | 
 |         } | 
 |         void* addr = dlsym(sharedObj, line); | 
 |         if (addr == nullptr) { | 
 |             ALOGE("Failed to find variable address for %s: %s", | 
 |                   line, dlerror()); | 
 |             // Not a critical error if we don't find a global variable. | 
 |         } | 
 |         fieldAddress[i] = addr; | 
 |         fieldIsObject[i] = false; | 
 |         fieldName[i] = duplicateString(line, sizeof(line)); | 
 |     } | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |     if (sscanf(line, EXPORT_FUNC_STR "%zu", &funcCount) != 1) { | 
 |         ALOGE("Invalid export func count!: %s", line); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     invokeFunctions = new InvokeFunc_t[funcCount]; | 
 |     if (invokeFunctions == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     for (size_t i = 0; i < funcCount; ++i) { | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             goto error; | 
 |         } | 
 |         char *c = strrchr(line, '\n'); | 
 |         if (c) { | 
 |             *c = '\0'; | 
 |         } | 
 |  | 
 |         invokeFunctions[i] = (InvokeFunc_t) dlsym(sharedObj, line); | 
 |         if (invokeFunctions[i] == nullptr) { | 
 |             ALOGE("Failed to get function address for %s(): %s", | 
 |                   line, dlerror()); | 
 |             goto error; | 
 |         } | 
 |     } | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |     if (sscanf(line, EXPORT_FOREACH_STR "%zu", &forEachCount) != 1) { | 
 |         ALOGE("Invalid export forEach count!: %s", line); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     forEachFunctions = new ForEachFunc_t[forEachCount]; | 
 |     if (forEachFunctions == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     forEachSignatures = new uint32_t[forEachCount]; | 
 |     if (forEachSignatures == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     for (size_t i = 0; i < forEachCount; ++i) { | 
 |         unsigned int tmpSig = 0; | 
 |         char tmpName[MAXLINE]; | 
 |  | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             goto error; | 
 |         } | 
 |         if (sscanf(line, "%u - %" MAKE_STR(MAXLINESTR) "s", | 
 |                    &tmpSig, tmpName) != 2) { | 
 |           ALOGE("Invalid export forEach!: %s", line); | 
 |           goto error; | 
 |         } | 
 |  | 
 |         // Lookup the expanded ForEach kernel. | 
 |         strncat(tmpName, ".expand", MAXLINESTR-strlen(tmpName)); | 
 |         forEachSignatures[i] = tmpSig; | 
 |         forEachFunctions[i] = | 
 |             (ForEachFunc_t) dlsym(sharedObj, tmpName); | 
 |         if (i != 0 && forEachFunctions[i] == nullptr && | 
 |             strcmp(tmpName, "root.expand")) { | 
 |             // Ignore missing root.expand functions. | 
 |             // root() is always specified at location 0. | 
 |             ALOGE("Failed to find forEach function address for %s(): %s", | 
 |                   tmpName, dlerror()); | 
 |             goto error; | 
 |         } | 
 |     } | 
 |  | 
 |     // Read general reduce kernels | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |     if (sscanf(line, EXPORT_REDUCE_STR "%zu", &reduceCount) != 1) { | 
 |         ALOGE("Invalid export reduce new count!: %s", line); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     reduceDescriptions = new ReduceDescription[reduceCount]; | 
 |     if (reduceDescriptions == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     for (size_t i = 0; i < reduceCount; ++i) { | 
 |         static const char kNoName[] = "."; | 
 |  | 
 |         unsigned int tmpSig = 0; | 
 |         size_t tmpSize = 0; | 
 |         char tmpNameReduce[MAXLINE]; | 
 |         char tmpNameInitializer[MAXLINE]; | 
 |         char tmpNameAccumulator[MAXLINE]; | 
 |         char tmpNameCombiner[MAXLINE]; | 
 |         char tmpNameOutConverter[MAXLINE]; | 
 |         char tmpNameHalter[MAXLINE]; | 
 |  | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             goto error; | 
 |         } | 
 | #define DELIMNAME " - %" MAKE_STR(MAXLINESTR) "s" | 
 |         if (sscanf(line, "%u - %zu" DELIMNAME DELIMNAME DELIMNAME DELIMNAME DELIMNAME DELIMNAME, | 
 |                    &tmpSig, &tmpSize, tmpNameReduce, tmpNameInitializer, tmpNameAccumulator, | 
 |                    tmpNameCombiner, tmpNameOutConverter, tmpNameHalter) != 8) { | 
 |             ALOGE("Invalid export reduce new!: %s", line); | 
 |             goto error; | 
 |         } | 
 | #undef DELIMNAME | 
 |  | 
 |         // For now, we expect | 
 |         // - Reduce and Accumulator names | 
 |         // - optional Initializer, Combiner, and OutConverter name | 
 |         // - no Halter name | 
 |         if (!strcmp(tmpNameReduce, kNoName) || | 
 |             !strcmp(tmpNameAccumulator, kNoName)) { | 
 |             ALOGE("Expected reduce and accumulator names!: %s", line); | 
 |             goto error; | 
 |         } | 
 |         if (strcmp(tmpNameHalter, kNoName)) { | 
 |             ALOGE("Did not expect halter name!: %s", line); | 
 |             goto error; | 
 |         } | 
 |  | 
 |         // The current implementation does not use the signature | 
 |         // or reduce name. | 
 |  | 
 |         reduceDescriptions[i].accumSize = tmpSize; | 
 |  | 
 |         // Process the (optional) initializer. | 
 |         if (strcmp(tmpNameInitializer, kNoName)) { | 
 |           // Lookup the original user-written initializer. | 
 |           if (!(reduceDescriptions[i].initFunc = | 
 |                 (ReduceInitializerFunc_t) dlsym(sharedObj, tmpNameInitializer))) { | 
 |             ALOGE("Failed to find initializer function address for %s(): %s", | 
 |                   tmpNameInitializer, dlerror()); | 
 |             goto error; | 
 |           } | 
 |         } else { | 
 |           reduceDescriptions[i].initFunc = nullptr; | 
 |         } | 
 |  | 
 |         // Lookup the expanded accumulator. | 
 |         strncat(tmpNameAccumulator, ".expand", MAXLINESTR-strlen(tmpNameAccumulator)); | 
 |         if (!(reduceDescriptions[i].accumFunc = | 
 |               (ReduceAccumulatorFunc_t) dlsym(sharedObj, tmpNameAccumulator))) { | 
 |             ALOGE("Failed to find accumulator function address for %s(): %s", | 
 |                   tmpNameAccumulator, dlerror()); | 
 |             goto error; | 
 |         } | 
 |  | 
 |         // Process the (optional) combiner. | 
 |         if (strcmp(tmpNameCombiner, kNoName)) { | 
 |           // Lookup the original user-written combiner. | 
 |           if (!(reduceDescriptions[i].combFunc = | 
 |                 (ReduceCombinerFunc_t) dlsym(sharedObj, tmpNameCombiner))) { | 
 |             ALOGE("Failed to find combiner function address for %s(): %s", | 
 |                   tmpNameCombiner, dlerror()); | 
 |             goto error; | 
 |           } | 
 |         } else { | 
 |           reduceDescriptions[i].combFunc = nullptr; | 
 |         } | 
 |  | 
 |         // Process the (optional) outconverter. | 
 |         if (strcmp(tmpNameOutConverter, kNoName)) { | 
 |           // Lookup the original user-written outconverter. | 
 |           if (!(reduceDescriptions[i].outFunc = | 
 |                 (ReduceOutConverterFunc_t) dlsym(sharedObj, tmpNameOutConverter))) { | 
 |             ALOGE("Failed to find outconverter function address for %s(): %s", | 
 |                   tmpNameOutConverter, dlerror()); | 
 |             goto error; | 
 |           } | 
 |         } else { | 
 |           reduceDescriptions[i].outFunc = nullptr; | 
 |         } | 
 |     } | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |     if (sscanf(line, OBJECT_SLOT_STR "%zu", &objectSlotCount) != 1) { | 
 |         ALOGE("Invalid object slot count!: %s", line); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     for (size_t i = 0; i < objectSlotCount; ++i) { | 
 |         uint32_t varNum = 0; | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             goto error; | 
 |         } | 
 |         if (sscanf(line, "%u", &varNum) != 1) { | 
 |             ALOGE("Invalid object slot!: %s", line); | 
 |             goto error; | 
 |         } | 
 |  | 
 |         if (varNum < varCount) { | 
 |             fieldIsObject[varNum] = true; | 
 |         } | 
 |     } | 
 |  | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 |     // Do not attempt to read pragmas or isThreadable flag in compat lib path. | 
 |     // Neither is applicable for compat lib | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     if (sscanf(line, PRAGMA_STR "%zu", &pragmaCount) != 1) { | 
 |         ALOGE("Invalid pragma count!: %s", line); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     pragmaKeys = new const char*[pragmaCount]; | 
 |     if (pragmaKeys == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     pragmaValues = new const char*[pragmaCount]; | 
 |     if (pragmaValues == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     bzero(pragmaKeys, sizeof(char*) * pragmaCount); | 
 |     bzero(pragmaValues, sizeof(char*) * pragmaCount); | 
 |  | 
 |     for (size_t i = 0; i < pragmaCount; ++i) { | 
 |         if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |             ALOGE("Unable to read pragma at index %zu!", i); | 
 |             goto error; | 
 |         } | 
 |         char key[MAXLINE]; | 
 |         char value[MAXLINE] = ""; // initialize in case value is empty | 
 |  | 
 |         // pragmas can just have a key and no value.  Only check to make sure | 
 |         // that the key is not empty | 
 |         if (sscanf(line, "%" MAKE_STR(MAXLINESTR) "s - %" MAKE_STR(MAXLINESTR) "s", | 
 |                    key, value) == 0 || | 
 |             strlen(key) == 0) | 
 |         { | 
 |             ALOGE("Invalid pragma value!: %s", line); | 
 |  | 
 |             goto error; | 
 |         } | 
 |  | 
 |         pragmaKeys[i] = duplicateString(key, sizeof(key)); | 
 |         pragmaValues[i] = duplicateString(value, sizeof(value)); | 
 |         //ALOGE("Pragma %zu: Key: '%s' Value: '%s'", i, pKey, pValue); | 
 |     } | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) == nullptr) { | 
 |         goto error; | 
 |     } | 
 |  | 
 |     char tmpFlag[4]; | 
 |     if (sscanf(line, THREADABLE_STR "%3s", tmpFlag) != 1) { | 
 |         ALOGE("Invalid threadable flag!: %s", line); | 
 |         goto error; | 
 |     } | 
 |     if (strcmp(tmpFlag, "yes") == 0) { | 
 |         isThreadable = true; | 
 |     } else if (strcmp(tmpFlag, "no") == 0) { | 
 |         isThreadable = false; | 
 |     } else { | 
 |         ALOGE("Invalid threadable flag!: %s", tmpFlag); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     if (strgets(line, MAXLINE, &rsInfo) != nullptr) { | 
 |         if (sscanf(line, CHECKSUM_STR "%08x", &checksum) != 1) { | 
 |             ALOGE("Invalid checksum flag!: %s", line); | 
 |             goto error; | 
 |         } | 
 |     } else { | 
 |         ALOGE("Missing checksum in shared obj file"); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     if (expectedChecksum != 0 && checksum != expectedChecksum) { | 
 |         ALOGE("Found invalid checksum.  Expected %08x, got %08x\n", | 
 |               expectedChecksum, checksum); | 
 |         goto error; | 
 |     } | 
 |  | 
 |     { | 
 |       // Parse the version info string, but ignore its contents as it's only | 
 |       // used by the debugger | 
 |       size_t nLines = 0; | 
 |       if (strgets(line, MAXLINE, &rsInfo) != nullptr) { | 
 |         if (sscanf(line, VERSIONINFO_STR "%zu", &nLines) != 1) { | 
 |           ALOGE("invalid versionInfo count"); | 
 |           goto error; | 
 |         } else { | 
 |           // skip the versionInfo packet as libRs doesn't use it | 
 |           while (nLines) { | 
 |             --nLines; | 
 |             if (strgets(line, MAXLINE, &rsInfo) == nullptr) | 
 |               goto error; | 
 |           } | 
 |         } | 
 |       } else { | 
 |         ALOGE(".rs.info is missing versionInfo section"); | 
 |       } | 
 |     } | 
 |  | 
 | #endif  // RS_COMPATIBILITY_LIB | 
 |  | 
 |     // Read in information about mutable global variables provided by bcc's | 
 |     // RSGlobalInfoPass | 
 |     if (rsGlobalEntries) { | 
 |         numEntries = *rsGlobalEntries; | 
 |         if (numEntries > 0) { | 
 |             rsAssert(rsGlobalNames); | 
 |             rsAssert(rsGlobalAddresses); | 
 |             rsAssert(rsGlobalSizes); | 
 |             rsAssert(rsGlobalProperties); | 
 |         } | 
 |     } | 
 |  | 
 |     return new ScriptExecutable( | 
 |         fieldAddress, fieldIsObject, fieldName, varCount, | 
 |         invokeFunctions, funcCount, | 
 |         forEachFunctions, forEachSignatures, forEachCount, | 
 |         reduceDescriptions, reduceCount, | 
 |         pragmaKeys, pragmaValues, pragmaCount, | 
 |         rsGlobalNames, rsGlobalAddresses, rsGlobalSizes, rsGlobalProperties, | 
 |         numEntries, isThreadable, checksum); | 
 |  | 
 | error: | 
 |  | 
 | #ifndef RS_COMPATIBILITY_LIB | 
 |  | 
 |     if (pragmaKeys) { | 
 |         for (size_t idx = 0; idx < pragmaCount; ++idx) { | 
 |             delete [] pragmaKeys[idx]; | 
 |         } | 
 |     } | 
 |  | 
 |     if (pragmaValues) { | 
 |         for (size_t idx = 0; idx < pragmaCount; ++idx) { | 
 |             delete [] pragmaValues[idx]; | 
 |         } | 
 |     } | 
 |  | 
 |     delete[] pragmaValues; | 
 |     delete[] pragmaKeys; | 
 | #endif  // RS_COMPATIBILITY_LIB | 
 |  | 
 |     delete[] reduceDescriptions; | 
 |  | 
 |     delete[] forEachSignatures; | 
 |     delete[] forEachFunctions; | 
 |  | 
 |     delete[] invokeFunctions; | 
 |  | 
 |     for (size_t i = 0; i < varCount; i++) { | 
 |         delete[] fieldName[i]; | 
 |     } | 
 |     delete[] fieldName; | 
 |     delete[] fieldIsObject; | 
 |     delete[] fieldAddress; | 
 |  | 
 |     return nullptr; | 
 | } | 
 |  | 
 | void* ScriptExecutable::getFieldAddress(const char* name) const { | 
 |     // TODO: improve this by using a hash map. | 
 |     for (size_t i = 0; i < mExportedVarCount; i++) { | 
 |         if (strcmp(name, mFieldName[i]) == 0) { | 
 |             return mFieldAddress[i]; | 
 |         } | 
 |     } | 
 |     return nullptr; | 
 | } | 
 |  | 
 | bool ScriptExecutable::dumpGlobalInfo() const { | 
 |     ALOGE("Globals: %p %p %p", mGlobalAddresses, mGlobalSizes, mGlobalNames); | 
 |     ALOGE("P   - Pointer"); | 
 |     ALOGE(" C  - Constant"); | 
 |     ALOGE("  S - Static"); | 
 |     for (int i = 0; i < mGlobalEntries; i++) { | 
 |         ALOGE("Global[%d]: %p %zu %s", i, mGlobalAddresses[i], mGlobalSizes[i], | 
 |               mGlobalNames[i]); | 
 |         uint32_t properties = mGlobalProperties[i]; | 
 |         ALOGE("%c%c%c Type: %u", | 
 |               isGlobalPointer(properties)  ? 'P' : ' ', | 
 |               isGlobalConstant(properties) ? 'C' : ' ', | 
 |               isGlobalStatic(properties)   ? 'S' : ' ', | 
 |               getGlobalRsType(properties)); | 
 |     } | 
 |     return true; | 
 | } | 
 |  | 
 | }  // namespace renderscript | 
 | }  // namespace android |