|  | //===- BuildLibCalls.cpp - Utility builder for libcalls -------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file implements some functions that will create standard C libcalls. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "llvm/Transforms/Utils/BuildLibCalls.h" | 
|  | #include "llvm/ADT/SmallString.h" | 
|  | #include "llvm/ADT/Statistic.h" | 
|  | #include "llvm/Analysis/TargetLibraryInfo.h" | 
|  | #include "llvm/IR/Constants.h" | 
|  | #include "llvm/IR/DataLayout.h" | 
|  | #include "llvm/IR/Function.h" | 
|  | #include "llvm/IR/IRBuilder.h" | 
|  | #include "llvm/IR/Intrinsics.h" | 
|  | #include "llvm/IR/LLVMContext.h" | 
|  | #include "llvm/IR/Module.h" | 
|  | #include "llvm/IR/Type.h" | 
|  |  | 
|  | using namespace llvm; | 
|  |  | 
|  | #define DEBUG_TYPE "build-libcalls" | 
|  |  | 
|  | //- Infer Attributes ---------------------------------------------------------// | 
|  |  | 
|  | STATISTIC(NumReadNone, "Number of functions inferred as readnone"); | 
|  | STATISTIC(NumReadOnly, "Number of functions inferred as readonly"); | 
|  | STATISTIC(NumArgMemOnly, "Number of functions inferred as argmemonly"); | 
|  | STATISTIC(NumNoUnwind, "Number of functions inferred as nounwind"); | 
|  | STATISTIC(NumNoCapture, "Number of arguments inferred as nocapture"); | 
|  | STATISTIC(NumReadOnlyArg, "Number of arguments inferred as readonly"); | 
|  | STATISTIC(NumNoAlias, "Number of function returns inferred as noalias"); | 
|  | STATISTIC(NumNonNull, "Number of function returns inferred as nonnull returns"); | 
|  | STATISTIC(NumReturnedArg, "Number of arguments inferred as returned"); | 
|  |  | 
|  | static bool setDoesNotAccessMemory(Function &F) { | 
|  | if (F.doesNotAccessMemory()) | 
|  | return false; | 
|  | F.setDoesNotAccessMemory(); | 
|  | ++NumReadNone; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setOnlyReadsMemory(Function &F) { | 
|  | if (F.onlyReadsMemory()) | 
|  | return false; | 
|  | F.setOnlyReadsMemory(); | 
|  | ++NumReadOnly; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setOnlyAccessesArgMemory(Function &F) { | 
|  | if (F.onlyAccessesArgMemory()) | 
|  | return false; | 
|  | F.setOnlyAccessesArgMemory(); | 
|  | ++NumArgMemOnly; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setDoesNotThrow(Function &F) { | 
|  | if (F.doesNotThrow()) | 
|  | return false; | 
|  | F.setDoesNotThrow(); | 
|  | ++NumNoUnwind; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setRetDoesNotAlias(Function &F) { | 
|  | if (F.hasAttribute(AttributeList::ReturnIndex, Attribute::NoAlias)) | 
|  | return false; | 
|  | F.addAttribute(AttributeList::ReturnIndex, Attribute::NoAlias); | 
|  | ++NumNoAlias; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setDoesNotCapture(Function &F, unsigned ArgNo) { | 
|  | if (F.hasParamAttribute(ArgNo, Attribute::NoCapture)) | 
|  | return false; | 
|  | F.addParamAttr(ArgNo, Attribute::NoCapture); | 
|  | ++NumNoCapture; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setOnlyReadsMemory(Function &F, unsigned ArgNo) { | 
|  | if (F.hasParamAttribute(ArgNo, Attribute::ReadOnly)) | 
|  | return false; | 
|  | F.addParamAttr(ArgNo, Attribute::ReadOnly); | 
|  | ++NumReadOnlyArg; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setRetNonNull(Function &F) { | 
|  | assert(F.getReturnType()->isPointerTy() && | 
|  | "nonnull applies only to pointers"); | 
|  | if (F.hasAttribute(AttributeList::ReturnIndex, Attribute::NonNull)) | 
|  | return false; | 
|  | F.addAttribute(AttributeList::ReturnIndex, Attribute::NonNull); | 
|  | ++NumNonNull; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setReturnedArg(Function &F, unsigned ArgNo) { | 
|  | if (F.hasParamAttribute(ArgNo, Attribute::Returned)) | 
|  | return false; | 
|  | F.addParamAttr(ArgNo, Attribute::Returned); | 
|  | ++NumReturnedArg; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static bool setNonLazyBind(Function &F) { | 
|  | if (F.hasFnAttribute(Attribute::NonLazyBind)) | 
|  | return false; | 
|  | F.addFnAttr(Attribute::NonLazyBind); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool llvm::inferLibFuncAttributes(Module *M, StringRef Name, | 
|  | const TargetLibraryInfo &TLI) { | 
|  | Function *F = M->getFunction(Name); | 
|  | if (!F) | 
|  | return false; | 
|  | return inferLibFuncAttributes(*F, TLI); | 
|  | } | 
|  |  | 
|  | bool llvm::inferLibFuncAttributes(Function &F, const TargetLibraryInfo &TLI) { | 
|  | LibFunc TheLibFunc; | 
|  | if (!(TLI.getLibFunc(F, TheLibFunc) && TLI.has(TheLibFunc))) | 
|  | return false; | 
|  |  | 
|  | bool Changed = false; | 
|  |  | 
|  | if (F.getParent() != nullptr && F.getParent()->getRtLibUseGOT()) | 
|  | Changed |= setNonLazyBind(F); | 
|  |  | 
|  | switch (TheLibFunc) { | 
|  | case LibFunc_strlen: | 
|  | case LibFunc_wcslen: | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setOnlyAccessesArgMemory(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_strchr: | 
|  | case LibFunc_strrchr: | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  | case LibFunc_strtol: | 
|  | case LibFunc_strtod: | 
|  | case LibFunc_strtof: | 
|  | case LibFunc_strtoul: | 
|  | case LibFunc_strtoll: | 
|  | case LibFunc_strtold: | 
|  | case LibFunc_strtoull: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_strcpy: | 
|  | case LibFunc_strncpy: | 
|  | case LibFunc_strcat: | 
|  | case LibFunc_strncat: | 
|  | Changed |= setReturnedArg(F, 0); | 
|  | LLVM_FALLTHROUGH; | 
|  | case LibFunc_stpcpy: | 
|  | case LibFunc_stpncpy: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_strxfrm: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_strcmp:      // 0,1 | 
|  | case LibFunc_strspn:      // 0,1 | 
|  | case LibFunc_strncmp:     // 0,1 | 
|  | case LibFunc_strcspn:     // 0,1 | 
|  | case LibFunc_strcoll:     // 0,1 | 
|  | case LibFunc_strcasecmp:  // 0,1 | 
|  | case LibFunc_strncasecmp: // | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_strstr: | 
|  | case LibFunc_strpbrk: | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_strtok: | 
|  | case LibFunc_strtok_r: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_scanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_setbuf: | 
|  | case LibFunc_setvbuf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_strdup: | 
|  | case LibFunc_strndup: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_stat: | 
|  | case LibFunc_statvfs: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_sscanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_sprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_snprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 2); | 
|  | Changed |= setOnlyReadsMemory(F, 2); | 
|  | return Changed; | 
|  | case LibFunc_setitimer: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setDoesNotCapture(F, 2); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_system: | 
|  | // May throw; "system" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_malloc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_memcmp: | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_memchr: | 
|  | case LibFunc_memrchr: | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  | case LibFunc_modf: | 
|  | case LibFunc_modff: | 
|  | case LibFunc_modfl: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_memcpy: | 
|  | case LibFunc_memmove: | 
|  | Changed |= setReturnedArg(F, 0); | 
|  | LLVM_FALLTHROUGH; | 
|  | case LibFunc_mempcpy: | 
|  | case LibFunc_memccpy: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_memcpy_chk: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  | case LibFunc_memalign: | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_mkdir: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_mktime: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_realloc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_read: | 
|  | // May throw; "read" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_rewind: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_rmdir: | 
|  | case LibFunc_remove: | 
|  | case LibFunc_realpath: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_rename: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_readlink: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_write: | 
|  | // May throw; "write" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_bcopy: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_bcmp: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_bzero: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_calloc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_chmod: | 
|  | case LibFunc_chown: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_ctermid: | 
|  | case LibFunc_clearerr: | 
|  | case LibFunc_closedir: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_atoi: | 
|  | case LibFunc_atol: | 
|  | case LibFunc_atof: | 
|  | case LibFunc_atoll: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_access: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_fopen: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_fdopen: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_feof: | 
|  | case LibFunc_free: | 
|  | case LibFunc_fseek: | 
|  | case LibFunc_ftell: | 
|  | case LibFunc_fgetc: | 
|  | case LibFunc_fgetc_unlocked: | 
|  | case LibFunc_fseeko: | 
|  | case LibFunc_ftello: | 
|  | case LibFunc_fileno: | 
|  | case LibFunc_fflush: | 
|  | case LibFunc_fclose: | 
|  | case LibFunc_fsetpos: | 
|  | case LibFunc_flockfile: | 
|  | case LibFunc_funlockfile: | 
|  | case LibFunc_ftrylockfile: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_ferror: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | return Changed; | 
|  | case LibFunc_fputc: | 
|  | case LibFunc_fputc_unlocked: | 
|  | case LibFunc_fstat: | 
|  | case LibFunc_frexp: | 
|  | case LibFunc_frexpf: | 
|  | case LibFunc_frexpl: | 
|  | case LibFunc_fstatvfs: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_fgets: | 
|  | case LibFunc_fgets_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 2); | 
|  | return Changed; | 
|  | case LibFunc_fread: | 
|  | case LibFunc_fread_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 3); | 
|  | return Changed; | 
|  | case LibFunc_fwrite: | 
|  | case LibFunc_fwrite_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 3); | 
|  | // FIXME: readonly #1? | 
|  | return Changed; | 
|  | case LibFunc_fputs: | 
|  | case LibFunc_fputs_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_fscanf: | 
|  | case LibFunc_fprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_fgetpos: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_getc: | 
|  | case LibFunc_getlogin_r: | 
|  | case LibFunc_getc_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_getenv: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setOnlyReadsMemory(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_gets: | 
|  | case LibFunc_getchar: | 
|  | case LibFunc_getchar_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  | case LibFunc_getitimer: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_getpwnam: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_ungetc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_uname: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_unlink: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_unsetenv: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_utime: | 
|  | case LibFunc_utimes: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_putc: | 
|  | case LibFunc_putc_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_puts: | 
|  | case LibFunc_printf: | 
|  | case LibFunc_perror: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_pread: | 
|  | // May throw; "pread" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_pwrite: | 
|  | // May throw; "pwrite" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_putchar: | 
|  | case LibFunc_putchar_unlocked: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  | case LibFunc_popen: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_pclose: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_vscanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_vsscanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_vfscanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_valloc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_vprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_vfprintf: | 
|  | case LibFunc_vsprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_vsnprintf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 2); | 
|  | Changed |= setOnlyReadsMemory(F, 2); | 
|  | return Changed; | 
|  | case LibFunc_open: | 
|  | // May throw; "open" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_opendir: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_tmpfile: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_times: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_htonl: | 
|  | case LibFunc_htons: | 
|  | case LibFunc_ntohl: | 
|  | case LibFunc_ntohs: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotAccessMemory(F); | 
|  | return Changed; | 
|  | case LibFunc_lstat: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_lchown: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_qsort: | 
|  | // May throw; places call through function pointer. | 
|  | Changed |= setDoesNotCapture(F, 3); | 
|  | return Changed; | 
|  | case LibFunc_dunder_strdup: | 
|  | case LibFunc_dunder_strndup: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_dunder_strtok_r: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_under_IO_getc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_under_IO_putc: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_dunder_isoc99_scanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_stat64: | 
|  | case LibFunc_lstat64: | 
|  | case LibFunc_statvfs64: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_dunder_isoc99_sscanf: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_fopen64: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_fseeko64: | 
|  | case LibFunc_ftello64: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_tmpfile64: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | case LibFunc_fstat64: | 
|  | case LibFunc_fstatvfs64: | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_open64: | 
|  | // May throw; "open" is a valid pthread cancellation point. | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setOnlyReadsMemory(F, 0); | 
|  | return Changed; | 
|  | case LibFunc_gettimeofday: | 
|  | // Currently some platforms have the restrict keyword on the arguments to | 
|  | // gettimeofday. To be conservative, do not add noalias to gettimeofday's | 
|  | // arguments. | 
|  | Changed |= setDoesNotThrow(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | return Changed; | 
|  | case LibFunc_Znwj: // new(unsigned int) | 
|  | case LibFunc_Znwm: // new(unsigned long) | 
|  | case LibFunc_Znaj: // new[](unsigned int) | 
|  | case LibFunc_Znam: // new[](unsigned long) | 
|  | case LibFunc_msvc_new_int: // new(unsigned int) | 
|  | case LibFunc_msvc_new_longlong: // new(unsigned long long) | 
|  | case LibFunc_msvc_new_array_int: // new[](unsigned int) | 
|  | case LibFunc_msvc_new_array_longlong: // new[](unsigned long long) | 
|  | // Operator new always returns a nonnull noalias pointer | 
|  | Changed |= setRetNonNull(F); | 
|  | Changed |= setRetDoesNotAlias(F); | 
|  | return Changed; | 
|  | // TODO: add LibFunc entries for: | 
|  | // case LibFunc_memset_pattern4: | 
|  | // case LibFunc_memset_pattern8: | 
|  | case LibFunc_memset_pattern16: | 
|  | Changed |= setOnlyAccessesArgMemory(F); | 
|  | Changed |= setDoesNotCapture(F, 0); | 
|  | Changed |= setDoesNotCapture(F, 1); | 
|  | Changed |= setOnlyReadsMemory(F, 1); | 
|  | return Changed; | 
|  | // int __nvvm_reflect(const char *) | 
|  | case LibFunc_nvvm_reflect: | 
|  | Changed |= setDoesNotAccessMemory(F); | 
|  | Changed |= setDoesNotThrow(F); | 
|  | return Changed; | 
|  |  | 
|  | default: | 
|  | // FIXME: It'd be really nice to cover all the library functions we're | 
|  | // aware of here. | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool llvm::hasUnaryFloatFn(const TargetLibraryInfo *TLI, Type *Ty, | 
|  | LibFunc DoubleFn, LibFunc FloatFn, | 
|  | LibFunc LongDoubleFn) { | 
|  | switch (Ty->getTypeID()) { | 
|  | case Type::HalfTyID: | 
|  | return false; | 
|  | case Type::FloatTyID: | 
|  | return TLI->has(FloatFn); | 
|  | case Type::DoubleTyID: | 
|  | return TLI->has(DoubleFn); | 
|  | default: | 
|  | return TLI->has(LongDoubleFn); | 
|  | } | 
|  | } | 
|  |  | 
|  | StringRef llvm::getUnaryFloatFn(const TargetLibraryInfo *TLI, Type *Ty, | 
|  | LibFunc DoubleFn, LibFunc FloatFn, | 
|  | LibFunc LongDoubleFn) { | 
|  | assert(hasUnaryFloatFn(TLI, Ty, DoubleFn, FloatFn, LongDoubleFn) && | 
|  | "Cannot get name for unavailable function!"); | 
|  |  | 
|  | switch (Ty->getTypeID()) { | 
|  | case Type::HalfTyID: | 
|  | llvm_unreachable("No name for HalfTy!"); | 
|  | case Type::FloatTyID: | 
|  | return TLI->getName(FloatFn); | 
|  | case Type::DoubleTyID: | 
|  | return TLI->getName(DoubleFn); | 
|  | default: | 
|  | return TLI->getName(LongDoubleFn); | 
|  | } | 
|  | } | 
|  |  | 
|  | //- Emit LibCalls ------------------------------------------------------------// | 
|  |  | 
|  | Value *llvm::castToCStr(Value *V, IRBuilder<> &B) { | 
|  | unsigned AS = V->getType()->getPointerAddressSpace(); | 
|  | return B.CreateBitCast(V, B.getInt8PtrTy(AS), "cstr"); | 
|  | } | 
|  |  | 
|  | Value *llvm::emitStrLen(Value *Ptr, IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_strlen)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef StrlenName = TLI->getName(LibFunc_strlen); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee StrLen = M->getOrInsertFunction( | 
|  | StrlenName, DL.getIntPtrType(Context), B.getInt8PtrTy()); | 
|  | inferLibFuncAttributes(M, StrlenName, *TLI); | 
|  | CallInst *CI = B.CreateCall(StrLen, castToCStr(Ptr, B), StrlenName); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(StrLen.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitStrChr(Value *Ptr, char C, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_strchr)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef StrChrName = TLI->getName(LibFunc_strchr); | 
|  | Type *I8Ptr = B.getInt8PtrTy(); | 
|  | Type *I32Ty = B.getInt32Ty(); | 
|  | FunctionCallee StrChr = | 
|  | M->getOrInsertFunction(StrChrName, I8Ptr, I8Ptr, I32Ty); | 
|  | inferLibFuncAttributes(M, StrChrName, *TLI); | 
|  | CallInst *CI = B.CreateCall( | 
|  | StrChr, {castToCStr(Ptr, B), ConstantInt::get(I32Ty, C)}, StrChrName); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(StrChr.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitStrNCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B, | 
|  | const DataLayout &DL, const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_strncmp)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef StrNCmpName = TLI->getName(LibFunc_strncmp); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee StrNCmp = | 
|  | M->getOrInsertFunction(StrNCmpName, B.getInt32Ty(), B.getInt8PtrTy(), | 
|  | B.getInt8PtrTy(), DL.getIntPtrType(Context)); | 
|  | inferLibFuncAttributes(M, StrNCmpName, *TLI); | 
|  | CallInst *CI = B.CreateCall( | 
|  | StrNCmp, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, StrNCmpName); | 
|  |  | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(StrNCmp.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitStrCpy(Value *Dst, Value *Src, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI, StringRef Name) { | 
|  | if (!TLI->has(LibFunc_strcpy)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | Type *I8Ptr = B.getInt8PtrTy(); | 
|  | FunctionCallee StrCpy = M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr); | 
|  | inferLibFuncAttributes(M, Name, *TLI); | 
|  | CallInst *CI = | 
|  | B.CreateCall(StrCpy, {castToCStr(Dst, B), castToCStr(Src, B)}, Name); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(StrCpy.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitStrNCpy(Value *Dst, Value *Src, Value *Len, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI, StringRef Name) { | 
|  | if (!TLI->has(LibFunc_strncpy)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | Type *I8Ptr = B.getInt8PtrTy(); | 
|  | FunctionCallee StrNCpy = | 
|  | M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr, Len->getType()); | 
|  | inferLibFuncAttributes(M, Name, *TLI); | 
|  | CallInst *CI = B.CreateCall( | 
|  | StrNCpy, {castToCStr(Dst, B), castToCStr(Src, B), Len}, Name); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(StrNCpy.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitMemCpyChk(Value *Dst, Value *Src, Value *Len, Value *ObjSize, | 
|  | IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_memcpy_chk)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | AttributeList AS; | 
|  | AS = AttributeList::get(M->getContext(), AttributeList::FunctionIndex, | 
|  | Attribute::NoUnwind); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee MemCpy = M->getOrInsertFunction( | 
|  | "__memcpy_chk", AttributeList::get(M->getContext(), AS), B.getInt8PtrTy(), | 
|  | B.getInt8PtrTy(), B.getInt8PtrTy(), DL.getIntPtrType(Context), | 
|  | DL.getIntPtrType(Context)); | 
|  | Dst = castToCStr(Dst, B); | 
|  | Src = castToCStr(Src, B); | 
|  | CallInst *CI = B.CreateCall(MemCpy, {Dst, Src, Len, ObjSize}); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(MemCpy.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitMemChr(Value *Ptr, Value *Val, Value *Len, IRBuilder<> &B, | 
|  | const DataLayout &DL, const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_memchr)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef MemChrName = TLI->getName(LibFunc_memchr); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee MemChr = | 
|  | M->getOrInsertFunction(MemChrName, B.getInt8PtrTy(), B.getInt8PtrTy(), | 
|  | B.getInt32Ty(), DL.getIntPtrType(Context)); | 
|  | inferLibFuncAttributes(M, MemChrName, *TLI); | 
|  | CallInst *CI = B.CreateCall(MemChr, {castToCStr(Ptr, B), Val, Len}, MemChrName); | 
|  |  | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(MemChr.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | // Common code for memcmp() and bcmp(), which have the exact same properties, | 
|  | // just a slight difference in semantics. | 
|  | static Value *emitMemCmpOrBcmp(llvm::LibFunc libfunc, Value *Ptr1, Value *Ptr2, | 
|  | Value *Len, IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(libfunc)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef CmpFnName = TLI->getName(libfunc); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee CmpFn = | 
|  | M->getOrInsertFunction(CmpFnName, B.getInt32Ty(), B.getInt8PtrTy(), | 
|  | B.getInt8PtrTy(), DL.getIntPtrType(Context)); | 
|  | inferLibFuncAttributes(M, CmpFnName, *TLI); | 
|  | CallInst *CI = B.CreateCall( | 
|  | CmpFn, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, CmpFnName); | 
|  |  | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(CmpFn.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitMemCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B, | 
|  | const DataLayout &DL, const TargetLibraryInfo *TLI) { | 
|  | return emitMemCmpOrBcmp(LibFunc_memcmp, Ptr1, Ptr2, Len, B, DL, TLI); | 
|  | } | 
|  |  | 
|  | Value *llvm::emitBCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B, | 
|  | const DataLayout &DL, const TargetLibraryInfo *TLI) { | 
|  | return emitMemCmpOrBcmp(LibFunc_bcmp, Ptr1, Ptr2, Len, B, DL, TLI); | 
|  | } | 
|  |  | 
|  | /// Append a suffix to the function name according to the type of 'Op'. | 
|  | static void appendTypeSuffix(Value *Op, StringRef &Name, | 
|  | SmallString<20> &NameBuffer) { | 
|  | if (!Op->getType()->isDoubleTy()) { | 
|  | NameBuffer += Name; | 
|  |  | 
|  | if (Op->getType()->isFloatTy()) | 
|  | NameBuffer += 'f'; | 
|  | else | 
|  | NameBuffer += 'l'; | 
|  |  | 
|  | Name = NameBuffer; | 
|  | } | 
|  | } | 
|  |  | 
|  | static Value *emitUnaryFloatFnCallHelper(Value *Op, StringRef Name, | 
|  | IRBuilder<> &B, | 
|  | const AttributeList &Attrs) { | 
|  | assert((Name != "") && "Must specify Name to emitUnaryFloatFnCall"); | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | FunctionCallee Callee = | 
|  | M->getOrInsertFunction(Name, Op->getType(), Op->getType()); | 
|  | CallInst *CI = B.CreateCall(Callee, Op, Name); | 
|  |  | 
|  | // The incoming attribute set may have come from a speculatable intrinsic, but | 
|  | // is being replaced with a library call which is not allowed to be | 
|  | // speculatable. | 
|  | CI->setAttributes(Attrs.removeAttribute(B.getContext(), | 
|  | AttributeList::FunctionIndex, | 
|  | Attribute::Speculatable)); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(Callee.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitUnaryFloatFnCall(Value *Op, StringRef Name, IRBuilder<> &B, | 
|  | const AttributeList &Attrs) { | 
|  | SmallString<20> NameBuffer; | 
|  | appendTypeSuffix(Op, Name, NameBuffer); | 
|  |  | 
|  | return emitUnaryFloatFnCallHelper(Op, Name, B, Attrs); | 
|  | } | 
|  |  | 
|  | Value *llvm::emitUnaryFloatFnCall(Value *Op, const TargetLibraryInfo *TLI, | 
|  | LibFunc DoubleFn, LibFunc FloatFn, | 
|  | LibFunc LongDoubleFn, IRBuilder<> &B, | 
|  | const AttributeList &Attrs) { | 
|  | // Get the name of the function according to TLI. | 
|  | StringRef Name = getUnaryFloatFn(TLI, Op->getType(), | 
|  | DoubleFn, FloatFn, LongDoubleFn); | 
|  |  | 
|  | return emitUnaryFloatFnCallHelper(Op, Name, B, Attrs); | 
|  | } | 
|  |  | 
|  | Value *llvm::emitBinaryFloatFnCall(Value *Op1, Value *Op2, StringRef Name, | 
|  | IRBuilder<> &B, const AttributeList &Attrs) { | 
|  | assert((Name != "") && "Must specify Name to emitBinaryFloatFnCall"); | 
|  |  | 
|  | SmallString<20> NameBuffer; | 
|  | appendTypeSuffix(Op1, Name, NameBuffer); | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | FunctionCallee Callee = M->getOrInsertFunction( | 
|  | Name, Op1->getType(), Op1->getType(), Op2->getType()); | 
|  | CallInst *CI = B.CreateCall(Callee, {Op1, Op2}, Name); | 
|  | CI->setAttributes(Attrs); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(Callee.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitPutChar(Value *Char, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_putchar)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef PutCharName = TLI->getName(LibFunc_putchar); | 
|  | FunctionCallee PutChar = | 
|  | M->getOrInsertFunction(PutCharName, B.getInt32Ty(), B.getInt32Ty()); | 
|  | inferLibFuncAttributes(M, PutCharName, *TLI); | 
|  | CallInst *CI = B.CreateCall(PutChar, | 
|  | B.CreateIntCast(Char, | 
|  | B.getInt32Ty(), | 
|  | /*isSigned*/true, | 
|  | "chari"), | 
|  | PutCharName); | 
|  |  | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(PutChar.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitPutS(Value *Str, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_puts)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef PutsName = TLI->getName(LibFunc_puts); | 
|  | FunctionCallee PutS = | 
|  | M->getOrInsertFunction(PutsName, B.getInt32Ty(), B.getInt8PtrTy()); | 
|  | inferLibFuncAttributes(M, PutsName, *TLI); | 
|  | CallInst *CI = B.CreateCall(PutS, castToCStr(Str, B), PutsName); | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(PutS.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFPutC(Value *Char, Value *File, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fputc)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FPutcName = TLI->getName(LibFunc_fputc); | 
|  | FunctionCallee F = M->getOrInsertFunction(FPutcName, B.getInt32Ty(), | 
|  | B.getInt32Ty(), File->getType()); | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FPutcName, *TLI); | 
|  | Char = B.CreateIntCast(Char, B.getInt32Ty(), /*isSigned*/true, | 
|  | "chari"); | 
|  | CallInst *CI = B.CreateCall(F, {Char, File}, FPutcName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFPutCUnlocked(Value *Char, Value *File, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fputc_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FPutcUnlockedName = TLI->getName(LibFunc_fputc_unlocked); | 
|  | FunctionCallee F = M->getOrInsertFunction(FPutcUnlockedName, B.getInt32Ty(), | 
|  | B.getInt32Ty(), File->getType()); | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FPutcUnlockedName, *TLI); | 
|  | Char = B.CreateIntCast(Char, B.getInt32Ty(), /*isSigned*/ true, "chari"); | 
|  | CallInst *CI = B.CreateCall(F, {Char, File}, FPutcUnlockedName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFPutS(Value *Str, Value *File, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fputs)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FPutsName = TLI->getName(LibFunc_fputs); | 
|  | FunctionCallee F = M->getOrInsertFunction(FPutsName, B.getInt32Ty(), | 
|  | B.getInt8PtrTy(), File->getType()); | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FPutsName, *TLI); | 
|  | CallInst *CI = B.CreateCall(F, {castToCStr(Str, B), File}, FPutsName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFPutSUnlocked(Value *Str, Value *File, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fputs_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FPutsUnlockedName = TLI->getName(LibFunc_fputs_unlocked); | 
|  | FunctionCallee F = M->getOrInsertFunction(FPutsUnlockedName, B.getInt32Ty(), | 
|  | B.getInt8PtrTy(), File->getType()); | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FPutsUnlockedName, *TLI); | 
|  | CallInst *CI = B.CreateCall(F, {castToCStr(Str, B), File}, FPutsUnlockedName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFWrite(Value *Ptr, Value *Size, Value *File, IRBuilder<> &B, | 
|  | const DataLayout &DL, const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fwrite)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | StringRef FWriteName = TLI->getName(LibFunc_fwrite); | 
|  | FunctionCallee F = M->getOrInsertFunction( | 
|  | FWriteName, DL.getIntPtrType(Context), B.getInt8PtrTy(), | 
|  | DL.getIntPtrType(Context), DL.getIntPtrType(Context), File->getType()); | 
|  |  | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FWriteName, *TLI); | 
|  | CallInst *CI = | 
|  | B.CreateCall(F, {castToCStr(Ptr, B), Size, | 
|  | ConstantInt::get(DL.getIntPtrType(Context), 1), File}); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitMalloc(Value *Num, IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_malloc)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef MallocName = TLI->getName(LibFunc_malloc); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | FunctionCallee Malloc = M->getOrInsertFunction(MallocName, B.getInt8PtrTy(), | 
|  | DL.getIntPtrType(Context)); | 
|  | inferLibFuncAttributes(M, MallocName, *TLI); | 
|  | CallInst *CI = B.CreateCall(Malloc, Num, MallocName); | 
|  |  | 
|  | if (const Function *F = | 
|  | dyn_cast<Function>(Malloc.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitCalloc(Value *Num, Value *Size, const AttributeList &Attrs, | 
|  | IRBuilder<> &B, const TargetLibraryInfo &TLI) { | 
|  | if (!TLI.has(LibFunc_calloc)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef CallocName = TLI.getName(LibFunc_calloc); | 
|  | const DataLayout &DL = M->getDataLayout(); | 
|  | IntegerType *PtrType = DL.getIntPtrType((B.GetInsertBlock()->getContext())); | 
|  | FunctionCallee Calloc = M->getOrInsertFunction( | 
|  | CallocName, Attrs, B.getInt8PtrTy(), PtrType, PtrType); | 
|  | inferLibFuncAttributes(M, CallocName, TLI); | 
|  | CallInst *CI = B.CreateCall(Calloc, {Num, Size}, CallocName); | 
|  |  | 
|  | if (const auto *F = | 
|  | dyn_cast<Function>(Calloc.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(F->getCallingConv()); | 
|  |  | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFWriteUnlocked(Value *Ptr, Value *Size, Value *N, Value *File, | 
|  | IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fwrite_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | StringRef FWriteUnlockedName = TLI->getName(LibFunc_fwrite_unlocked); | 
|  | FunctionCallee F = M->getOrInsertFunction( | 
|  | FWriteUnlockedName, DL.getIntPtrType(Context), B.getInt8PtrTy(), | 
|  | DL.getIntPtrType(Context), DL.getIntPtrType(Context), File->getType()); | 
|  |  | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FWriteUnlockedName, *TLI); | 
|  | CallInst *CI = B.CreateCall(F, {castToCStr(Ptr, B), Size, N, File}); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFGetCUnlocked(Value *File, IRBuilder<> &B, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fgetc_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FGetCUnlockedName = TLI->getName(LibFunc_fgetc_unlocked); | 
|  | FunctionCallee F = M->getOrInsertFunction(FGetCUnlockedName, B.getInt32Ty(), | 
|  | File->getType()); | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FGetCUnlockedName, *TLI); | 
|  | CallInst *CI = B.CreateCall(F, File, FGetCUnlockedName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFGetSUnlocked(Value *Str, Value *Size, Value *File, | 
|  | IRBuilder<> &B, const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fgets_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | StringRef FGetSUnlockedName = TLI->getName(LibFunc_fgets_unlocked); | 
|  | FunctionCallee F = | 
|  | M->getOrInsertFunction(FGetSUnlockedName, B.getInt8PtrTy(), | 
|  | B.getInt8PtrTy(), B.getInt32Ty(), File->getType()); | 
|  | inferLibFuncAttributes(M, FGetSUnlockedName, *TLI); | 
|  | CallInst *CI = | 
|  | B.CreateCall(F, {castToCStr(Str, B), Size, File}, FGetSUnlockedName); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } | 
|  |  | 
|  | Value *llvm::emitFReadUnlocked(Value *Ptr, Value *Size, Value *N, Value *File, | 
|  | IRBuilder<> &B, const DataLayout &DL, | 
|  | const TargetLibraryInfo *TLI) { | 
|  | if (!TLI->has(LibFunc_fread_unlocked)) | 
|  | return nullptr; | 
|  |  | 
|  | Module *M = B.GetInsertBlock()->getModule(); | 
|  | LLVMContext &Context = B.GetInsertBlock()->getContext(); | 
|  | StringRef FReadUnlockedName = TLI->getName(LibFunc_fread_unlocked); | 
|  | FunctionCallee F = M->getOrInsertFunction( | 
|  | FReadUnlockedName, DL.getIntPtrType(Context), B.getInt8PtrTy(), | 
|  | DL.getIntPtrType(Context), DL.getIntPtrType(Context), File->getType()); | 
|  |  | 
|  | if (File->getType()->isPointerTy()) | 
|  | inferLibFuncAttributes(M, FReadUnlockedName, *TLI); | 
|  | CallInst *CI = B.CreateCall(F, {castToCStr(Ptr, B), Size, N, File}); | 
|  |  | 
|  | if (const Function *Fn = | 
|  | dyn_cast<Function>(F.getCallee()->stripPointerCasts())) | 
|  | CI->setCallingConv(Fn->getCallingConv()); | 
|  | return CI; | 
|  | } |