blob: 7de2c1cc8d50b29447fc59e5991ac45ea5530385 [file] [log] [blame]
Eric Christopher87abfc52010-03-05 22:25:30 +00001//===- BuildLibCalls.cpp - Utility builder for libcalls -------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements some functions that will create standard C libcalls.
11//
12//===----------------------------------------------------------------------===//
13
14#include "llvm/Transforms/Utils/BuildLibCalls.h"
Chandler Carruthed0881b2012-12-03 16:50:05 +000015#include "llvm/ADT/SmallString.h"
Ahmed Bougachab0624a22016-04-27 19:04:40 +000016#include "llvm/ADT/Statistic.h"
Weiming Zhao45d4cb92015-11-24 18:57:06 +000017#include "llvm/Analysis/TargetLibraryInfo.h"
Chandler Carruth9fb823b2013-01-02 11:36:10 +000018#include "llvm/IR/Constants.h"
19#include "llvm/IR/DataLayout.h"
20#include "llvm/IR/Function.h"
21#include "llvm/IR/IRBuilder.h"
22#include "llvm/IR/Intrinsics.h"
23#include "llvm/IR/LLVMContext.h"
24#include "llvm/IR/Module.h"
25#include "llvm/IR/Type.h"
Eric Christopher87abfc52010-03-05 22:25:30 +000026
27using namespace llvm;
28
Ahmed Bougachab0624a22016-04-27 19:04:40 +000029#define DEBUG_TYPE "build-libcalls"
30
31//- Infer Attributes ---------------------------------------------------------//
32
33STATISTIC(NumReadNone, "Number of functions inferred as readnone");
34STATISTIC(NumReadOnly, "Number of functions inferred as readonly");
35STATISTIC(NumArgMemOnly, "Number of functions inferred as argmemonly");
36STATISTIC(NumNoUnwind, "Number of functions inferred as nounwind");
37STATISTIC(NumNoCapture, "Number of arguments inferred as nocapture");
38STATISTIC(NumReadOnlyArg, "Number of arguments inferred as readonly");
39STATISTIC(NumNoAlias, "Number of function returns inferred as noalias");
40STATISTIC(NumNonNull, "Number of function returns inferred as nonnull returns");
41
42static bool setDoesNotAccessMemory(Function &F) {
43 if (F.doesNotAccessMemory())
44 return false;
45 F.setDoesNotAccessMemory();
46 ++NumReadNone;
47 return true;
48}
49
50static bool setOnlyReadsMemory(Function &F) {
51 if (F.onlyReadsMemory())
52 return false;
53 F.setOnlyReadsMemory();
54 ++NumReadOnly;
55 return true;
56}
57
58static bool setOnlyAccessesArgMemory(Function &F) {
59 if (F.onlyAccessesArgMemory())
60 return false;
Reid Klecknera0b45f42017-05-03 18:17:31 +000061 F.setOnlyAccessesArgMemory();
Ahmed Bougachab0624a22016-04-27 19:04:40 +000062 ++NumArgMemOnly;
63 return true;
64}
65
66static bool setDoesNotThrow(Function &F) {
67 if (F.doesNotThrow())
68 return false;
69 F.setDoesNotThrow();
70 ++NumNoUnwind;
71 return true;
72}
73
Reid Klecknera0b45f42017-05-03 18:17:31 +000074static bool setRetDoesNotAlias(Function &F) {
75 if (F.hasAttribute(AttributeList::ReturnIndex, Attribute::NoAlias))
Ahmed Bougachab0624a22016-04-27 19:04:40 +000076 return false;
Reid Klecknera0b45f42017-05-03 18:17:31 +000077 F.addAttribute(AttributeList::ReturnIndex, Attribute::NoAlias);
Ahmed Bougachab0624a22016-04-27 19:04:40 +000078 ++NumNoAlias;
79 return true;
80}
81
Reid Klecknera0b45f42017-05-03 18:17:31 +000082static bool setDoesNotCapture(Function &F, unsigned ArgNo) {
83 if (F.hasParamAttribute(ArgNo, Attribute::NoCapture))
Ahmed Bougachab0624a22016-04-27 19:04:40 +000084 return false;
Reid Klecknera0b45f42017-05-03 18:17:31 +000085 F.addParamAttr(ArgNo, Attribute::NoCapture);
86 ++NumNoCapture;
87 return true;
88}
89
90static bool setOnlyReadsMemory(Function &F, unsigned ArgNo) {
91 if (F.hasParamAttribute(ArgNo, Attribute::ReadOnly))
92 return false;
93 F.addParamAttr(ArgNo, Attribute::ReadOnly);
94 ++NumReadOnlyArg;
95 return true;
96}
97
98static bool setRetNonNull(Function &F) {
99 assert(F.getReturnType()->isPointerTy() &&
100 "nonnull applies only to pointers");
101 if (F.hasAttribute(AttributeList::ReturnIndex, Attribute::NonNull))
102 return false;
103 F.addAttribute(AttributeList::ReturnIndex, Attribute::NonNull);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000104 ++NumNonNull;
105 return true;
106}
107
Sriraman Tallam182f2df2018-04-10 23:32:36 +0000108static bool setNonLazyBind(Function &F) {
109 if (F.hasFnAttribute(Attribute::NonLazyBind))
110 return false;
111 F.addFnAttr(Attribute::NonLazyBind);
112 return true;
113}
114
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000115bool llvm::inferLibFuncAttributes(Function &F, const TargetLibraryInfo &TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000116 LibFunc TheLibFunc;
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000117 if (!(TLI.getLibFunc(F, TheLibFunc) && TLI.has(TheLibFunc)))
118 return false;
119
120 bool Changed = false;
Sriraman Tallam182f2df2018-04-10 23:32:36 +0000121
122 if (F.getParent() != nullptr && F.getParent()->getRtLibUseGOT())
123 Changed |= setNonLazyBind(F);
124
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000125 switch (TheLibFunc) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000126 case LibFunc_strlen:
Matthias Braun60b40b82017-05-05 20:25:50 +0000127 case LibFunc_wcslen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000128 Changed |= setOnlyReadsMemory(F);
129 Changed |= setDoesNotThrow(F);
Xin Tong9d2a5b12017-06-18 03:10:26 +0000130 Changed |= setOnlyAccessesArgMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000131 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000132 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000133 case LibFunc_strchr:
134 case LibFunc_strrchr:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000135 Changed |= setOnlyReadsMemory(F);
136 Changed |= setDoesNotThrow(F);
137 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000138 case LibFunc_strtol:
139 case LibFunc_strtod:
140 case LibFunc_strtof:
141 case LibFunc_strtoul:
142 case LibFunc_strtoll:
143 case LibFunc_strtold:
144 case LibFunc_strtoull:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000145 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000146 Changed |= setDoesNotCapture(F, 1);
147 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000148 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000149 case LibFunc_strcpy:
150 case LibFunc_stpcpy:
151 case LibFunc_strcat:
152 case LibFunc_strncat:
153 case LibFunc_strncpy:
154 case LibFunc_stpncpy:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000155 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000156 Changed |= setDoesNotCapture(F, 1);
157 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000158 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000159 case LibFunc_strxfrm:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000160 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000161 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000162 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000163 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000164 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000165 case LibFunc_strcmp: // 0,1
166 case LibFunc_strspn: // 0,1
167 case LibFunc_strncmp: // 0,1
168 case LibFunc_strcspn: // 0,1
169 case LibFunc_strcoll: // 0,1
170 case LibFunc_strcasecmp: // 0,1
171 case LibFunc_strncasecmp: //
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000172 Changed |= setOnlyReadsMemory(F);
173 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000174 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000175 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000176 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000177 case LibFunc_strstr:
178 case LibFunc_strpbrk:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000179 Changed |= setOnlyReadsMemory(F);
180 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000181 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000182 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000183 case LibFunc_strtok:
184 case LibFunc_strtok_r:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000185 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000186 Changed |= setDoesNotCapture(F, 1);
187 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000188 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000189 case LibFunc_scanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000190 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000191 Changed |= setDoesNotCapture(F, 0);
192 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000193 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000194 case LibFunc_setbuf:
195 case LibFunc_setvbuf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000196 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000197 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000198 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000199 case LibFunc_strdup:
200 case LibFunc_strndup:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000201 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000202 Changed |= setRetDoesNotAlias(F);
203 Changed |= setDoesNotCapture(F, 0);
204 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000205 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000206 case LibFunc_stat:
207 case LibFunc_statvfs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000208 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000209 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000210 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000211 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000212 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000213 case LibFunc_sscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000214 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000215 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000216 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000217 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000218 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000219 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000220 case LibFunc_sprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000221 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000222 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000223 Changed |= setDoesNotCapture(F, 1);
224 Changed |= setOnlyReadsMemory(F, 1);
225 return Changed;
Reid Klecknera0b45f42017-05-03 18:17:31 +0000226 case LibFunc_snprintf:
227 Changed |= setDoesNotThrow(F);
228 Changed |= setDoesNotCapture(F, 0);
229 Changed |= setDoesNotCapture(F, 2);
230 Changed |= setOnlyReadsMemory(F, 2);
231 return Changed;
232 case LibFunc_setitimer:
233 Changed |= setDoesNotThrow(F);
234 Changed |= setDoesNotCapture(F, 1);
235 Changed |= setDoesNotCapture(F, 2);
236 Changed |= setOnlyReadsMemory(F, 1);
237 return Changed;
238 case LibFunc_system:
239 // May throw; "system" is a valid pthread cancellation point.
240 Changed |= setDoesNotCapture(F, 0);
241 Changed |= setOnlyReadsMemory(F, 0);
242 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000243 case LibFunc_malloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000244 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000245 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000246 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000247 case LibFunc_memcmp:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000248 Changed |= setOnlyReadsMemory(F);
249 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000250 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000251 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000252 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000253 case LibFunc_memchr:
254 case LibFunc_memrchr:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000255 Changed |= setOnlyReadsMemory(F);
256 Changed |= setDoesNotThrow(F);
257 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000258 case LibFunc_modf:
259 case LibFunc_modff:
260 case LibFunc_modfl:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000261 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000262 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000263 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000264 case LibFunc_memcpy:
265 case LibFunc_mempcpy:
266 case LibFunc_memccpy:
267 case LibFunc_memmove:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000268 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000269 Changed |= setDoesNotCapture(F, 1);
270 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000271 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000272 case LibFunc_memcpy_chk:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000273 Changed |= setDoesNotThrow(F);
274 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000275 case LibFunc_memalign:
Reid Klecknera0b45f42017-05-03 18:17:31 +0000276 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000277 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000278 case LibFunc_mkdir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000279 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000280 Changed |= setDoesNotCapture(F, 0);
281 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000282 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000283 case LibFunc_mktime:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000284 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000285 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000286 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000287 case LibFunc_realloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000288 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000289 Changed |= setRetDoesNotAlias(F);
290 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000291 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000292 case LibFunc_read:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000293 // May throw; "read" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000294 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000295 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000296 case LibFunc_rewind:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000297 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000298 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000299 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000300 case LibFunc_rmdir:
301 case LibFunc_remove:
302 case LibFunc_realpath:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000303 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000304 Changed |= setDoesNotCapture(F, 0);
305 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000306 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000307 case LibFunc_rename:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000308 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000309 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000310 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000311 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000312 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000313 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000314 case LibFunc_readlink:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000315 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000316 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000317 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000318 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000319 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000320 case LibFunc_write:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000321 // May throw; "write" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000322 Changed |= setDoesNotCapture(F, 1);
323 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000324 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000325 case LibFunc_bcopy:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000326 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000327 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000328 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000329 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000330 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000331 case LibFunc_bcmp:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000332 Changed |= setDoesNotThrow(F);
333 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000334 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000335 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000336 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000337 case LibFunc_bzero:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000338 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000339 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000340 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000341 case LibFunc_calloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000342 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000343 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000344 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000345 case LibFunc_chmod:
346 case LibFunc_chown:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000347 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000348 Changed |= setDoesNotCapture(F, 0);
349 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000350 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000351 case LibFunc_ctermid:
352 case LibFunc_clearerr:
353 case LibFunc_closedir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000354 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000355 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000356 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000357 case LibFunc_atoi:
358 case LibFunc_atol:
359 case LibFunc_atof:
360 case LibFunc_atoll:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000361 Changed |= setDoesNotThrow(F);
362 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000363 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000364 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000365 case LibFunc_access:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000366 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000367 Changed |= setDoesNotCapture(F, 0);
368 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000369 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000370 case LibFunc_fopen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000371 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000372 Changed |= setRetDoesNotAlias(F);
373 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000374 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000375 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000376 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000377 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000378 case LibFunc_fdopen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000379 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000380 Changed |= setRetDoesNotAlias(F);
381 Changed |= setDoesNotCapture(F, 1);
382 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000383 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000384 case LibFunc_feof:
385 case LibFunc_free:
386 case LibFunc_fseek:
387 case LibFunc_ftell:
388 case LibFunc_fgetc:
389 case LibFunc_fseeko:
390 case LibFunc_ftello:
391 case LibFunc_fileno:
392 case LibFunc_fflush:
393 case LibFunc_fclose:
394 case LibFunc_fsetpos:
395 case LibFunc_flockfile:
396 case LibFunc_funlockfile:
397 case LibFunc_ftrylockfile:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000398 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000399 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000400 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000401 case LibFunc_ferror:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000402 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000403 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000404 Changed |= setOnlyReadsMemory(F);
405 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000406 case LibFunc_fputc:
407 case LibFunc_fstat:
408 case LibFunc_frexp:
409 case LibFunc_frexpf:
410 case LibFunc_frexpl:
411 case LibFunc_fstatvfs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000412 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000413 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000414 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000415 case LibFunc_fgets:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000416 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000417 Changed |= setDoesNotCapture(F, 2);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000418 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000419 case LibFunc_fread:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000420 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000421 Changed |= setDoesNotCapture(F, 0);
422 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000423 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000424 case LibFunc_fwrite:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000425 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000426 Changed |= setDoesNotCapture(F, 0);
427 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000428 // FIXME: readonly #1?
429 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000430 case LibFunc_fputs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000431 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000432 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000433 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000434 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000435 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000436 case LibFunc_fscanf:
437 case LibFunc_fprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000438 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000439 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000440 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000441 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000442 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000443 case LibFunc_fgetpos:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000444 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000445 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000446 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000447 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000448 case LibFunc_getc:
449 case LibFunc_getlogin_r:
450 case LibFunc_getc_unlocked:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000451 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000452 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000453 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000454 case LibFunc_getenv:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000455 Changed |= setDoesNotThrow(F);
456 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000457 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000458 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000459 case LibFunc_gets:
460 case LibFunc_getchar:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000461 Changed |= setDoesNotThrow(F);
462 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000463 case LibFunc_getitimer:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000464 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000465 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000466 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000467 case LibFunc_getpwnam:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000468 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000469 Changed |= setDoesNotCapture(F, 0);
470 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000471 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000472 case LibFunc_ungetc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000473 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000474 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000475 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000476 case LibFunc_uname:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000477 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000478 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000479 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000480 case LibFunc_unlink:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000481 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000482 Changed |= setDoesNotCapture(F, 0);
483 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000484 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000485 case LibFunc_unsetenv:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000486 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000487 Changed |= setDoesNotCapture(F, 0);
488 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000489 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000490 case LibFunc_utime:
491 case LibFunc_utimes:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000492 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000493 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000494 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000495 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000496 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000497 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000498 case LibFunc_putc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000499 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000500 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000501 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000502 case LibFunc_puts:
503 case LibFunc_printf:
504 case LibFunc_perror:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000505 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000506 Changed |= setDoesNotCapture(F, 0);
507 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000508 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000509 case LibFunc_pread:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000510 // May throw; "pread" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000511 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000512 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000513 case LibFunc_pwrite:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000514 // May throw; "pwrite" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000515 Changed |= setDoesNotCapture(F, 1);
516 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000517 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000518 case LibFunc_putchar:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000519 Changed |= setDoesNotThrow(F);
520 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000521 case LibFunc_popen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000522 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000523 Changed |= setRetDoesNotAlias(F);
524 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000525 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000526 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000527 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000528 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000529 case LibFunc_pclose:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000530 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000531 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000532 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000533 case LibFunc_vscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000534 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000535 Changed |= setDoesNotCapture(F, 0);
536 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000537 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000538 case LibFunc_vsscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000539 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000540 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000541 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000542 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000543 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000544 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000545 case LibFunc_vfscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000546 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000547 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000548 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000549 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000550 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000551 case LibFunc_valloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000552 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000553 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000554 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000555 case LibFunc_vprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000556 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000557 Changed |= setDoesNotCapture(F, 0);
558 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000559 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000560 case LibFunc_vfprintf:
561 case LibFunc_vsprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000562 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000563 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000564 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000565 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000566 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000567 case LibFunc_vsnprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000568 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000569 Changed |= setDoesNotCapture(F, 0);
570 Changed |= setDoesNotCapture(F, 2);
571 Changed |= setOnlyReadsMemory(F, 2);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000572 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000573 case LibFunc_open:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000574 // May throw; "open" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000575 Changed |= setDoesNotCapture(F, 0);
576 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000577 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000578 case LibFunc_opendir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000579 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000580 Changed |= setRetDoesNotAlias(F);
581 Changed |= setDoesNotCapture(F, 0);
582 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000583 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000584 case LibFunc_tmpfile:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000585 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000586 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000587 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000588 case LibFunc_times:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000589 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000590 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000591 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000592 case LibFunc_htonl:
593 case LibFunc_htons:
594 case LibFunc_ntohl:
595 case LibFunc_ntohs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000596 Changed |= setDoesNotThrow(F);
597 Changed |= setDoesNotAccessMemory(F);
598 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000599 case LibFunc_lstat:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000600 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000601 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000602 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000603 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000604 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000605 case LibFunc_lchown:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000606 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000607 Changed |= setDoesNotCapture(F, 0);
608 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000609 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000610 case LibFunc_qsort:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000611 // May throw; places call through function pointer.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000612 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000613 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000614 case LibFunc_dunder_strdup:
615 case LibFunc_dunder_strndup:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000616 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000617 Changed |= setRetDoesNotAlias(F);
618 Changed |= setDoesNotCapture(F, 0);
619 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000620 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000621 case LibFunc_dunder_strtok_r:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000622 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000623 Changed |= setDoesNotCapture(F, 1);
624 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000625 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000626 case LibFunc_under_IO_getc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000627 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000628 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000629 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000630 case LibFunc_under_IO_putc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000631 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000632 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000633 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000634 case LibFunc_dunder_isoc99_scanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000635 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000636 Changed |= setDoesNotCapture(F, 0);
637 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000638 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000639 case LibFunc_stat64:
640 case LibFunc_lstat64:
641 case LibFunc_statvfs64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000642 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000643 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000644 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000645 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000646 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000647 case LibFunc_dunder_isoc99_sscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000648 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000649 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000650 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000651 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000652 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000653 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000654 case LibFunc_fopen64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000655 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000656 Changed |= setRetDoesNotAlias(F);
657 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000658 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000659 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000660 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000661 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000662 case LibFunc_fseeko64:
663 case LibFunc_ftello64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000664 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000665 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000666 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000667 case LibFunc_tmpfile64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000668 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000669 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000670 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000671 case LibFunc_fstat64:
672 case LibFunc_fstatvfs64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000673 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000674 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000675 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000676 case LibFunc_open64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000677 // May throw; "open" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000678 Changed |= setDoesNotCapture(F, 0);
679 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000680 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000681 case LibFunc_gettimeofday:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000682 // Currently some platforms have the restrict keyword on the arguments to
683 // gettimeofday. To be conservative, do not add noalias to gettimeofday's
684 // arguments.
685 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000686 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000687 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000688 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000689 case LibFunc_Znwj: // new(unsigned int)
690 case LibFunc_Znwm: // new(unsigned long)
691 case LibFunc_Znaj: // new[](unsigned int)
692 case LibFunc_Znam: // new[](unsigned long)
693 case LibFunc_msvc_new_int: // new(unsigned int)
694 case LibFunc_msvc_new_longlong: // new(unsigned long long)
695 case LibFunc_msvc_new_array_int: // new[](unsigned int)
696 case LibFunc_msvc_new_array_longlong: // new[](unsigned long long)
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000697 // Operator new always returns a nonnull noalias pointer
Reid Klecknera0b45f42017-05-03 18:17:31 +0000698 Changed |= setRetNonNull(F);
699 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000700 return Changed;
701 //TODO: add LibFunc entries for:
David L. Jonesd21529f2017-01-23 23:16:46 +0000702 //case LibFunc_memset_pattern4:
703 //case LibFunc_memset_pattern8:
704 case LibFunc_memset_pattern16:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000705 Changed |= setOnlyAccessesArgMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000706 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougacha44c19872016-04-27 19:04:43 +0000707 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000708 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000709 return Changed;
710 // int __nvvm_reflect(const char *)
David L. Jonesd21529f2017-01-23 23:16:46 +0000711 case LibFunc_nvvm_reflect:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000712 Changed |= setDoesNotAccessMemory(F);
713 Changed |= setDoesNotThrow(F);
714 return Changed;
715
716 default:
717 // FIXME: It'd be really nice to cover all the library functions we're
718 // aware of here.
719 return false;
720 }
721}
722
Dmitry Venikove5fbf592018-01-11 06:33:00 +0000723bool llvm::hasUnaryFloatFn(const TargetLibraryInfo *TLI, Type *Ty,
724 LibFunc DoubleFn, LibFunc FloatFn,
725 LibFunc LongDoubleFn) {
726 switch (Ty->getTypeID()) {
727 case Type::FloatTyID:
728 return TLI->has(FloatFn);
729 case Type::DoubleTyID:
730 return TLI->has(DoubleFn);
731 default:
732 return TLI->has(LongDoubleFn);
733 }
734}
735
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000736//- Emit LibCalls ------------------------------------------------------------//
737
Sanjay Pateld3112a52016-01-19 19:46:10 +0000738Value *llvm::castToCStr(Value *V, IRBuilder<> &B) {
Matt Arsenaultbe558882014-04-23 20:58:57 +0000739 unsigned AS = V->getType()->getPointerAddressSpace();
740 return B.CreateBitCast(V, B.getInt8PtrTy(AS), "cstr");
Eric Christopher87abfc52010-03-05 22:25:30 +0000741}
742
Sanjay Pateld3112a52016-01-19 19:46:10 +0000743Value *llvm::emitStrLen(Value *Ptr, IRBuilder<> &B, const DataLayout &DL,
Nuno Lopes89702e92012-07-25 16:46:31 +0000744 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000745 if (!TLI->has(LibFunc_strlen))
Craig Topperf40110f2014-04-25 05:29:35 +0000746 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000747
Sanjay Pateld4af2972016-01-19 19:58:49 +0000748 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000749 LLVMContext &Context = B.GetInsertBlock()->getContext();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000750 Constant *StrLen = M->getOrInsertFunction("strlen", DL.getIntPtrType(Context),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000751 B.getInt8PtrTy());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000752 inferLibFuncAttributes(*M->getFunction("strlen"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000753 CallInst *CI = B.CreateCall(StrLen, castToCStr(Ptr, B), "strlen");
Evan Chengad6efbf2014-03-12 18:09:37 +0000754 if (const Function *F = dyn_cast<Function>(StrLen->stripPointerCasts()))
755 CI->setCallingConv(F->getCallingConv());
756
Eric Christopher87abfc52010-03-05 22:25:30 +0000757 return CI;
758}
759
Sanjay Pateld3112a52016-01-19 19:46:10 +0000760Value *llvm::emitStrChr(Value *Ptr, char C, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000761 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000762 if (!TLI->has(LibFunc_strchr))
Craig Topperf40110f2014-04-25 05:29:35 +0000763 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000764
Sanjay Pateld4af2972016-01-19 19:58:49 +0000765 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000766 Type *I8Ptr = B.getInt8PtrTy();
767 Type *I32Ty = B.getInt32Ty();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000768 Constant *StrChr =
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000769 M->getOrInsertFunction("strchr", I8Ptr, I8Ptr, I32Ty);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000770 inferLibFuncAttributes(*M->getFunction("strchr"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000771 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000772 StrChr, {castToCStr(Ptr, B), ConstantInt::get(I32Ty, C)}, "strchr");
Evan Chengad6efbf2014-03-12 18:09:37 +0000773 if (const Function *F = dyn_cast<Function>(StrChr->stripPointerCasts()))
774 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000775 return CI;
776}
777
Sanjay Pateld3112a52016-01-19 19:46:10 +0000778Value *llvm::emitStrNCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000779 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000780 if (!TLI->has(LibFunc_strncmp))
Craig Topperf40110f2014-04-25 05:29:35 +0000781 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000782
Sanjay Pateld4af2972016-01-19 19:58:49 +0000783 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000784 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000785 Value *StrNCmp = M->getOrInsertFunction("strncmp", B.getInt32Ty(),
786 B.getInt8PtrTy(), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000787 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000788 inferLibFuncAttributes(*M->getFunction("strncmp"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000789 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000790 StrNCmp, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, "strncmp");
Evan Chengad6efbf2014-03-12 18:09:37 +0000791
792 if (const Function *F = dyn_cast<Function>(StrNCmp->stripPointerCasts()))
793 CI->setCallingConv(F->getCallingConv());
794
Benjamin Kramer11188602010-06-15 21:34:25 +0000795 return CI;
796}
797
Sanjay Pateld3112a52016-01-19 19:46:10 +0000798Value *llvm::emitStrCpy(Value *Dst, Value *Src, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000799 const TargetLibraryInfo *TLI, StringRef Name) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000800 if (!TLI->has(LibFunc_strcpy))
Craig Topperf40110f2014-04-25 05:29:35 +0000801 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000802
Sanjay Pateld4af2972016-01-19 19:58:49 +0000803 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000804 Type *I8Ptr = B.getInt8PtrTy();
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000805 Value *StrCpy = M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000806 inferLibFuncAttributes(*M->getFunction(Name), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000807 CallInst *CI =
Sanjay Pateld3112a52016-01-19 19:46:10 +0000808 B.CreateCall(StrCpy, {castToCStr(Dst, B), castToCStr(Src, B)}, Name);
Evan Chengad6efbf2014-03-12 18:09:37 +0000809 if (const Function *F = dyn_cast<Function>(StrCpy->stripPointerCasts()))
810 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000811 return CI;
812}
813
Sanjay Pateld3112a52016-01-19 19:46:10 +0000814Value *llvm::emitStrNCpy(Value *Dst, Value *Src, Value *Len, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000815 const TargetLibraryInfo *TLI, StringRef Name) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000816 if (!TLI->has(LibFunc_strncpy))
Craig Topperf40110f2014-04-25 05:29:35 +0000817 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000818
Sanjay Pateld4af2972016-01-19 19:58:49 +0000819 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000820 Type *I8Ptr = B.getInt8PtrTy();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000821 Value *StrNCpy = M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr,
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000822 Len->getType());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000823 inferLibFuncAttributes(*M->getFunction(Name), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000824 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000825 StrNCpy, {castToCStr(Dst, B), castToCStr(Src, B), Len}, "strncpy");
Evan Chengad6efbf2014-03-12 18:09:37 +0000826 if (const Function *F = dyn_cast<Function>(StrNCpy->stripPointerCasts()))
827 CI->setCallingConv(F->getCallingConv());
Eric Christopher43dc11c2010-03-11 01:25:07 +0000828 return CI;
829}
830
Sanjay Pateld3112a52016-01-19 19:46:10 +0000831Value *llvm::emitMemCpyChk(Value *Dst, Value *Src, Value *Len, Value *ObjSize,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000832 IRBuilder<> &B, const DataLayout &DL,
Nuno Lopes89702e92012-07-25 16:46:31 +0000833 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000834 if (!TLI->has(LibFunc_memcpy_chk))
Craig Topperf40110f2014-04-25 05:29:35 +0000835 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000836
Sanjay Pateld4af2972016-01-19 19:58:49 +0000837 Module *M = B.GetInsertBlock()->getModule();
Reid Klecknerb5180542017-03-21 16:57:19 +0000838 AttributeList AS;
839 AS = AttributeList::get(M->getContext(), AttributeList::FunctionIndex,
840 Attribute::NoUnwind);
Chandler Carruth7ec50852012-11-01 08:07:29 +0000841 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000842 Value *MemCpy = M->getOrInsertFunction(
Reid Klecknerb5180542017-03-21 16:57:19 +0000843 "__memcpy_chk", AttributeList::get(M->getContext(), AS), B.getInt8PtrTy(),
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000844 B.getInt8PtrTy(), B.getInt8PtrTy(), DL.getIntPtrType(Context),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000845 DL.getIntPtrType(Context));
Sanjay Pateld3112a52016-01-19 19:46:10 +0000846 Dst = castToCStr(Dst, B);
847 Src = castToCStr(Src, B);
David Blaikieff6409d2015-05-18 22:13:54 +0000848 CallInst *CI = B.CreateCall(MemCpy, {Dst, Src, Len, ObjSize});
Evan Chengad6efbf2014-03-12 18:09:37 +0000849 if (const Function *F = dyn_cast<Function>(MemCpy->stripPointerCasts()))
850 CI->setCallingConv(F->getCallingConv());
Evan Chengd9e82232010-03-23 15:48:04 +0000851 return CI;
852}
853
Sanjay Pateld3112a52016-01-19 19:46:10 +0000854Value *llvm::emitMemChr(Value *Ptr, Value *Val, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000855 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000856 if (!TLI->has(LibFunc_memchr))
Craig Topperf40110f2014-04-25 05:29:35 +0000857 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000858
Sanjay Pateld4af2972016-01-19 19:58:49 +0000859 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000860 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000861 Value *MemChr = M->getOrInsertFunction("memchr", B.getInt8PtrTy(),
862 B.getInt8PtrTy(), B.getInt32Ty(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000863 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000864 inferLibFuncAttributes(*M->getFunction("memchr"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000865 CallInst *CI = B.CreateCall(MemChr, {castToCStr(Ptr, B), Val, Len}, "memchr");
Evan Chengad6efbf2014-03-12 18:09:37 +0000866
867 if (const Function *F = dyn_cast<Function>(MemChr->stripPointerCasts()))
868 CI->setCallingConv(F->getCallingConv());
869
Eric Christopher87abfc52010-03-05 22:25:30 +0000870 return CI;
871}
872
Sanjay Pateld3112a52016-01-19 19:46:10 +0000873Value *llvm::emitMemCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000874 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000875 if (!TLI->has(LibFunc_memcmp))
Craig Topperf40110f2014-04-25 05:29:35 +0000876 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000877
Sanjay Pateld4af2972016-01-19 19:58:49 +0000878 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000879 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000880 Value *MemCmp = M->getOrInsertFunction("memcmp", B.getInt32Ty(),
881 B.getInt8PtrTy(), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000882 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000883 inferLibFuncAttributes(*M->getFunction("memcmp"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000884 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000885 MemCmp, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, "memcmp");
Evan Chengad6efbf2014-03-12 18:09:37 +0000886
887 if (const Function *F = dyn_cast<Function>(MemCmp->stripPointerCasts()))
888 CI->setCallingConv(F->getCallingConv());
889
Eric Christopher87abfc52010-03-05 22:25:30 +0000890 return CI;
891}
892
Yi Jiang6ab044e2013-12-16 22:42:40 +0000893/// Append a suffix to the function name according to the type of 'Op'.
Sanjay Pateld3112a52016-01-19 19:46:10 +0000894static void appendTypeSuffix(Value *Op, StringRef &Name,
Sanjay Patelb50325e2016-01-19 19:17:47 +0000895 SmallString<20> &NameBuffer) {
Yi Jiang6ab044e2013-12-16 22:42:40 +0000896 if (!Op->getType()->isDoubleTy()) {
897 NameBuffer += Name;
898
899 if (Op->getType()->isFloatTy())
900 NameBuffer += 'f';
901 else
902 NameBuffer += 'l';
903
904 Name = NameBuffer;
905 }
Yi Jiang6ab044e2013-12-16 22:42:40 +0000906}
907
Sanjay Pateld3112a52016-01-19 19:46:10 +0000908Value *llvm::emitUnaryFloatFnCall(Value *Op, StringRef Name, IRBuilder<> &B,
Reid Klecknerb5180542017-03-21 16:57:19 +0000909 const AttributeList &Attrs) {
Benjamin Kramerb106bcc2011-11-15 19:12:09 +0000910 SmallString<20> NameBuffer;
Sanjay Pateld3112a52016-01-19 19:46:10 +0000911 appendTypeSuffix(Op, Name, NameBuffer);
Eric Christopher87abfc52010-03-05 22:25:30 +0000912
Sanjay Pateld4af2972016-01-19 19:58:49 +0000913 Module *M = B.GetInsertBlock()->getModule();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000914 Value *Callee = M->getOrInsertFunction(Name, Op->getType(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000915 Op->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +0000916 CallInst *CI = B.CreateCall(Callee, Op, Name);
Matt Arsenault6a288c12017-05-03 02:26:10 +0000917
918 // The incoming attribute set may have come from a speculatable intrinsic, but
919 // is being replaced with a library call which is not allowed to be
920 // speculatable.
921 CI->setAttributes(Attrs.removeAttribute(B.getContext(),
922 AttributeList::FunctionIndex,
923 Attribute::Speculatable));
Evan Chengad6efbf2014-03-12 18:09:37 +0000924 if (const Function *F = dyn_cast<Function>(Callee->stripPointerCasts()))
925 CI->setCallingConv(F->getCallingConv());
926
Eric Christopher87abfc52010-03-05 22:25:30 +0000927 return CI;
928}
929
Sanjay Pateld3112a52016-01-19 19:46:10 +0000930Value *llvm::emitBinaryFloatFnCall(Value *Op1, Value *Op2, StringRef Name,
Reid Klecknerb5180542017-03-21 16:57:19 +0000931 IRBuilder<> &B, const AttributeList &Attrs) {
Yi Jiang6ab044e2013-12-16 22:42:40 +0000932 SmallString<20> NameBuffer;
Sanjay Pateld3112a52016-01-19 19:46:10 +0000933 appendTypeSuffix(Op1, Name, NameBuffer);
Yi Jiang6ab044e2013-12-16 22:42:40 +0000934
Sanjay Pateld4af2972016-01-19 19:58:49 +0000935 Module *M = B.GetInsertBlock()->getModule();
Sanjay Patelb50325e2016-01-19 19:17:47 +0000936 Value *Callee = M->getOrInsertFunction(Name, Op1->getType(), Op1->getType(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000937 Op2->getType());
David Blaikieff6409d2015-05-18 22:13:54 +0000938 CallInst *CI = B.CreateCall(Callee, {Op1, Op2}, Name);
Yi Jiang6ab044e2013-12-16 22:42:40 +0000939 CI->setAttributes(Attrs);
Evan Chengad6efbf2014-03-12 18:09:37 +0000940 if (const Function *F = dyn_cast<Function>(Callee->stripPointerCasts()))
941 CI->setCallingConv(F->getCallingConv());
942
Yi Jiang6ab044e2013-12-16 22:42:40 +0000943 return CI;
944}
945
Sanjay Pateld3112a52016-01-19 19:46:10 +0000946Value *llvm::emitPutChar(Value *Char, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000947 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000948 if (!TLI->has(LibFunc_putchar))
Craig Topperf40110f2014-04-25 05:29:35 +0000949 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000950
Sanjay Pateld4af2972016-01-19 19:58:49 +0000951 Module *M = B.GetInsertBlock()->getModule();
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000952 Value *PutChar = M->getOrInsertFunction("putchar", B.getInt32Ty(), B.getInt32Ty());
Craig Topper0f5063c2017-03-17 23:48:02 +0000953 inferLibFuncAttributes(*M->getFunction("putchar"), *TLI);
Eric Christopher87abfc52010-03-05 22:25:30 +0000954 CallInst *CI = B.CreateCall(PutChar,
955 B.CreateIntCast(Char,
956 B.getInt32Ty(),
957 /*isSigned*/true,
958 "chari"),
959 "putchar");
Evan Chengad6efbf2014-03-12 18:09:37 +0000960
961 if (const Function *F = dyn_cast<Function>(PutChar->stripPointerCasts()))
962 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000963 return CI;
964}
965
Sanjay Pateld3112a52016-01-19 19:46:10 +0000966Value *llvm::emitPutS(Value *Str, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000967 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000968 if (!TLI->has(LibFunc_puts))
Craig Topperf40110f2014-04-25 05:29:35 +0000969 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000970
Sanjay Pateld4af2972016-01-19 19:58:49 +0000971 Module *M = B.GetInsertBlock()->getModule();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000972 Value *PutS =
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000973 M->getOrInsertFunction("puts", B.getInt32Ty(), B.getInt8PtrTy());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000974 inferLibFuncAttributes(*M->getFunction("puts"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000975 CallInst *CI = B.CreateCall(PutS, castToCStr(Str, B), "puts");
Evan Chengad6efbf2014-03-12 18:09:37 +0000976 if (const Function *F = dyn_cast<Function>(PutS->stripPointerCasts()))
977 CI->setCallingConv(F->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +0000978 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +0000979}
980
Sanjay Pateld3112a52016-01-19 19:46:10 +0000981Value *llvm::emitFPutC(Value *Char, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000982 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000983 if (!TLI->has(LibFunc_fputc))
Craig Topperf40110f2014-04-25 05:29:35 +0000984 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000985
Sanjay Pateld4af2972016-01-19 19:58:49 +0000986 Module *M = B.GetInsertBlock()->getModule();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000987 Constant *F = M->getOrInsertFunction("fputc", B.getInt32Ty(), B.getInt32Ty(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000988 File->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +0000989 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000990 inferLibFuncAttributes(*M->getFunction("fputc"), *TLI);
Eric Christopher87abfc52010-03-05 22:25:30 +0000991 Char = B.CreateIntCast(Char, B.getInt32Ty(), /*isSigned*/true,
992 "chari");
David Blaikieff6409d2015-05-18 22:13:54 +0000993 CallInst *CI = B.CreateCall(F, {Char, File}, "fputc");
Evan Chengad6efbf2014-03-12 18:09:37 +0000994
995 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
996 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +0000997 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +0000998}
999
Sanjay Pateld3112a52016-01-19 19:46:10 +00001000Value *llvm::emitFPutS(Value *Str, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +00001001 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +00001002 if (!TLI->has(LibFunc_fputs))
Craig Topperf40110f2014-04-25 05:29:35 +00001003 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +00001004
Sanjay Pateld4af2972016-01-19 19:58:49 +00001005 Module *M = B.GetInsertBlock()->getModule();
David L. Jonesd21529f2017-01-23 23:16:46 +00001006 StringRef FPutsName = TLI->getName(LibFunc_fputs);
Mehdi Aminidb11fdf2017-04-06 20:23:57 +00001007 Constant *F = M->getOrInsertFunction(
Serge Guelton59a2d7b2017-04-11 15:01:18 +00001008 FPutsName, B.getInt32Ty(), B.getInt8PtrTy(), File->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +00001009 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +00001010 inferLibFuncAttributes(*M->getFunction(FPutsName), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +00001011 CallInst *CI = B.CreateCall(F, {castToCStr(Str, B), File}, "fputs");
Evan Chengad6efbf2014-03-12 18:09:37 +00001012
1013 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
1014 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +00001015 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +00001016}
1017
Sanjay Pateld3112a52016-01-19 19:46:10 +00001018Value *llvm::emitFWrite(Value *Ptr, Value *Size, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +00001019 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +00001020 if (!TLI->has(LibFunc_fwrite))
Craig Topperf40110f2014-04-25 05:29:35 +00001021 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +00001022
Sanjay Pateld4af2972016-01-19 19:58:49 +00001023 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +00001024 LLVMContext &Context = B.GetInsertBlock()->getContext();
David L. Jonesd21529f2017-01-23 23:16:46 +00001025 StringRef FWriteName = TLI->getName(LibFunc_fwrite);
Ahmed Bougachab0624a22016-04-27 19:04:40 +00001026 Constant *F = M->getOrInsertFunction(
1027 FWriteName, DL.getIntPtrType(Context), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +00001028 DL.getIntPtrType(Context), DL.getIntPtrType(Context), File->getType());
1029
Eric Christopher87abfc52010-03-05 22:25:30 +00001030 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +00001031 inferLibFuncAttributes(*M->getFunction(FWriteName), *TLI);
Mehdi Aminia28d91d2015-03-10 02:37:25 +00001032 CallInst *CI =
Sanjay Pateld3112a52016-01-19 19:46:10 +00001033 B.CreateCall(F, {castToCStr(Ptr, B), Size,
David Blaikieff6409d2015-05-18 22:13:54 +00001034 ConstantInt::get(DL.getIntPtrType(Context), 1), File});
Evan Chengad6efbf2014-03-12 18:09:37 +00001035
1036 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
1037 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +00001038 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +00001039}
Sanjay Patelb2ab3f22018-04-18 14:21:31 +00001040
1041Value *llvm::emitMalloc(Value *Num, IRBuilder<> &B, const DataLayout &DL,
1042 const TargetLibraryInfo *TLI) {
1043 if (!TLI->has(LibFunc_malloc))
1044 return nullptr;
1045
1046 Module *M = B.GetInsertBlock()->getModule();
1047 LLVMContext &Context = B.GetInsertBlock()->getContext();
1048 Value *Malloc = M->getOrInsertFunction("malloc", B.getInt8PtrTy(),
1049 DL.getIntPtrType(Context));
1050 inferLibFuncAttributes(*M->getFunction("malloc"), *TLI);
1051 CallInst *CI = B.CreateCall(Malloc, Num, "malloc");
1052
1053 if (const Function *F = dyn_cast<Function>(Malloc->stripPointerCasts()))
1054 CI->setCallingConv(F->getCallingConv());
1055
1056 return CI;
1057}
1058
1059Value *llvm::emitCalloc(Value *Num, Value *Size, const AttributeList &Attrs,
1060 IRBuilder<> &B, const TargetLibraryInfo &TLI) {
1061 if (!TLI.has(LibFunc_calloc))
1062 return nullptr;
1063
1064 Module *M = B.GetInsertBlock()->getModule();
1065 const DataLayout &DL = M->getDataLayout();
1066 IntegerType *PtrType = DL.getIntPtrType((B.GetInsertBlock()->getContext()));
1067 Value *Calloc = M->getOrInsertFunction("calloc", Attrs, B.getInt8PtrTy(),
1068 PtrType, PtrType);
1069 inferLibFuncAttributes(*M->getFunction("calloc"), TLI);
1070 CallInst *CI = B.CreateCall(Calloc, {Num, Size}, "calloc");
1071
1072 if (const auto *F = dyn_cast<Function>(Calloc->stripPointerCasts()))
1073 CI->setCallingConv(F->getCallingConv());
1074
1075 return CI;
1076}