blob: 1956697ccb8bf48f555600d8867429aa36065e47 [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
108bool llvm::inferLibFuncAttributes(Function &F, const TargetLibraryInfo &TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000109 LibFunc TheLibFunc;
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000110 if (!(TLI.getLibFunc(F, TheLibFunc) && TLI.has(TheLibFunc)))
111 return false;
112
113 bool Changed = false;
114 switch (TheLibFunc) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000115 case LibFunc_strlen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000116 Changed |= setOnlyReadsMemory(F);
117 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000118 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000119 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000120 case LibFunc_strchr:
121 case LibFunc_strrchr:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000122 Changed |= setOnlyReadsMemory(F);
123 Changed |= setDoesNotThrow(F);
124 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000125 case LibFunc_strtol:
126 case LibFunc_strtod:
127 case LibFunc_strtof:
128 case LibFunc_strtoul:
129 case LibFunc_strtoll:
130 case LibFunc_strtold:
131 case LibFunc_strtoull:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000132 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000133 Changed |= setDoesNotCapture(F, 1);
134 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000135 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000136 case LibFunc_strcpy:
137 case LibFunc_stpcpy:
138 case LibFunc_strcat:
139 case LibFunc_strncat:
140 case LibFunc_strncpy:
141 case LibFunc_stpncpy:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000142 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000143 Changed |= setDoesNotCapture(F, 1);
144 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000145 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000146 case LibFunc_strxfrm:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000147 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000148 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000149 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000150 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000151 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000152 case LibFunc_strcmp: // 0,1
153 case LibFunc_strspn: // 0,1
154 case LibFunc_strncmp: // 0,1
155 case LibFunc_strcspn: // 0,1
156 case LibFunc_strcoll: // 0,1
157 case LibFunc_strcasecmp: // 0,1
158 case LibFunc_strncasecmp: //
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000159 Changed |= setOnlyReadsMemory(F);
160 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);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000163 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000164 case LibFunc_strstr:
165 case LibFunc_strpbrk:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000166 Changed |= setOnlyReadsMemory(F);
167 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000168 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000169 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000170 case LibFunc_strtok:
171 case LibFunc_strtok_r:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000172 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000173 Changed |= setDoesNotCapture(F, 1);
174 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000175 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000176 case LibFunc_scanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000177 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000178 Changed |= setDoesNotCapture(F, 0);
179 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000180 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000181 case LibFunc_setbuf:
182 case LibFunc_setvbuf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000183 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000184 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000185 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000186 case LibFunc_strdup:
187 case LibFunc_strndup:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000188 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000189 Changed |= setRetDoesNotAlias(F);
190 Changed |= setDoesNotCapture(F, 0);
191 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000192 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000193 case LibFunc_stat:
194 case LibFunc_statvfs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000195 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000196 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000197 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000198 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000199 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000200 case LibFunc_sscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000201 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000202 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000203 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000204 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000205 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000206 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000207 case LibFunc_sprintf:
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);
211 Changed |= setOnlyReadsMemory(F, 1);
212 return Changed;
Reid Klecknera0b45f42017-05-03 18:17:31 +0000213 case LibFunc_snprintf:
214 Changed |= setDoesNotThrow(F);
215 Changed |= setDoesNotCapture(F, 0);
216 Changed |= setDoesNotCapture(F, 2);
217 Changed |= setOnlyReadsMemory(F, 2);
218 return Changed;
219 case LibFunc_setitimer:
220 Changed |= setDoesNotThrow(F);
221 Changed |= setDoesNotCapture(F, 1);
222 Changed |= setDoesNotCapture(F, 2);
223 Changed |= setOnlyReadsMemory(F, 1);
224 return Changed;
225 case LibFunc_system:
226 // May throw; "system" is a valid pthread cancellation point.
227 Changed |= setDoesNotCapture(F, 0);
228 Changed |= setOnlyReadsMemory(F, 0);
229 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000230 case LibFunc_malloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000231 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000232 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000233 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000234 case LibFunc_memcmp:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000235 Changed |= setOnlyReadsMemory(F);
236 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000237 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000238 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000239 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000240 case LibFunc_memchr:
241 case LibFunc_memrchr:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000242 Changed |= setOnlyReadsMemory(F);
243 Changed |= setDoesNotThrow(F);
244 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000245 case LibFunc_modf:
246 case LibFunc_modff:
247 case LibFunc_modfl:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000248 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000249 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000250 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000251 case LibFunc_memcpy:
252 case LibFunc_mempcpy:
253 case LibFunc_memccpy:
254 case LibFunc_memmove:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000255 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000256 Changed |= setDoesNotCapture(F, 1);
257 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000258 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000259 case LibFunc_memcpy_chk:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000260 Changed |= setDoesNotThrow(F);
261 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000262 case LibFunc_memalign:
Reid Klecknera0b45f42017-05-03 18:17:31 +0000263 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000264 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000265 case LibFunc_mkdir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000266 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000267 Changed |= setDoesNotCapture(F, 0);
268 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000269 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000270 case LibFunc_mktime:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000271 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000272 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000273 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000274 case LibFunc_realloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000275 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000276 Changed |= setRetDoesNotAlias(F);
277 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000278 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000279 case LibFunc_read:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000280 // May throw; "read" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000281 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000282 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000283 case LibFunc_rewind:
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_rmdir:
288 case LibFunc_remove:
289 case LibFunc_realpath:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000290 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000291 Changed |= setDoesNotCapture(F, 0);
292 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000293 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000294 case LibFunc_rename:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000295 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000296 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000297 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000298 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000299 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000300 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000301 case LibFunc_readlink:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000302 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000303 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000304 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000305 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_write:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000308 // May throw; "write" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000309 Changed |= setDoesNotCapture(F, 1);
310 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000311 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000312 case LibFunc_bcopy:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000313 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000314 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000315 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000316 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000317 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000318 case LibFunc_bcmp:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000319 Changed |= setDoesNotThrow(F);
320 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000321 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000322 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000323 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000324 case LibFunc_bzero:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000325 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000326 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000327 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000328 case LibFunc_calloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000329 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000330 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000331 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000332 case LibFunc_chmod:
333 case LibFunc_chown:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000334 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000335 Changed |= setDoesNotCapture(F, 0);
336 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000337 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000338 case LibFunc_ctermid:
339 case LibFunc_clearerr:
340 case LibFunc_closedir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000341 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000342 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000343 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000344 case LibFunc_atoi:
345 case LibFunc_atol:
346 case LibFunc_atof:
347 case LibFunc_atoll:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000348 Changed |= setDoesNotThrow(F);
349 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000350 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000351 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000352 case LibFunc_access:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000353 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000354 Changed |= setDoesNotCapture(F, 0);
355 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000356 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000357 case LibFunc_fopen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000358 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000359 Changed |= setRetDoesNotAlias(F);
360 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000361 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000362 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000363 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000364 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000365 case LibFunc_fdopen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000366 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000367 Changed |= setRetDoesNotAlias(F);
368 Changed |= setDoesNotCapture(F, 1);
369 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000370 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000371 case LibFunc_feof:
372 case LibFunc_free:
373 case LibFunc_fseek:
374 case LibFunc_ftell:
375 case LibFunc_fgetc:
376 case LibFunc_fseeko:
377 case LibFunc_ftello:
378 case LibFunc_fileno:
379 case LibFunc_fflush:
380 case LibFunc_fclose:
381 case LibFunc_fsetpos:
382 case LibFunc_flockfile:
383 case LibFunc_funlockfile:
384 case LibFunc_ftrylockfile:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000385 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000386 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000387 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000388 case LibFunc_ferror:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000389 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000390 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000391 Changed |= setOnlyReadsMemory(F);
392 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000393 case LibFunc_fputc:
394 case LibFunc_fstat:
395 case LibFunc_frexp:
396 case LibFunc_frexpf:
397 case LibFunc_frexpl:
398 case LibFunc_fstatvfs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000399 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000400 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000401 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000402 case LibFunc_fgets:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000403 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000404 Changed |= setDoesNotCapture(F, 2);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000405 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000406 case LibFunc_fread:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000407 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000408 Changed |= setDoesNotCapture(F, 0);
409 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000410 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000411 case LibFunc_fwrite:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000412 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000413 Changed |= setDoesNotCapture(F, 0);
414 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000415 // FIXME: readonly #1?
416 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000417 case LibFunc_fputs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000418 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000419 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000420 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000421 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000422 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000423 case LibFunc_fscanf:
424 case LibFunc_fprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000425 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000426 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000427 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000428 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000429 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000430 case LibFunc_fgetpos:
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);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000434 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000435 case LibFunc_getc:
436 case LibFunc_getlogin_r:
437 case LibFunc_getc_unlocked:
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 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000441 case LibFunc_getenv:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000442 Changed |= setDoesNotThrow(F);
443 Changed |= setOnlyReadsMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000444 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000445 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000446 case LibFunc_gets:
447 case LibFunc_getchar:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000448 Changed |= setDoesNotThrow(F);
449 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000450 case LibFunc_getitimer:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000451 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000452 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000453 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000454 case LibFunc_getpwnam:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000455 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000456 Changed |= setDoesNotCapture(F, 0);
457 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000458 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000459 case LibFunc_ungetc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000460 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000461 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000462 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000463 case LibFunc_uname:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000464 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000465 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000466 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000467 case LibFunc_unlink:
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_unsetenv:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000473 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000474 Changed |= setDoesNotCapture(F, 0);
475 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000476 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000477 case LibFunc_utime:
478 case LibFunc_utimes:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000479 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000480 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000481 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000482 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000483 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000484 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000485 case LibFunc_putc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000486 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000487 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000488 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000489 case LibFunc_puts:
490 case LibFunc_printf:
491 case LibFunc_perror:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000492 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000493 Changed |= setDoesNotCapture(F, 0);
494 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000495 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000496 case LibFunc_pread:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000497 // May throw; "pread" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000498 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000499 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000500 case LibFunc_pwrite:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000501 // May throw; "pwrite" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000502 Changed |= setDoesNotCapture(F, 1);
503 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000504 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000505 case LibFunc_putchar:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000506 Changed |= setDoesNotThrow(F);
507 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000508 case LibFunc_popen:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000509 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000510 Changed |= setRetDoesNotAlias(F);
511 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000512 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000513 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000514 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000515 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000516 case LibFunc_pclose:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000517 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000518 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000519 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000520 case LibFunc_vscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000521 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000522 Changed |= setDoesNotCapture(F, 0);
523 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000524 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000525 case LibFunc_vsscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000526 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000527 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000528 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000529 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000530 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000531 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000532 case LibFunc_vfscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000533 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000534 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000535 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000536 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000537 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000538 case LibFunc_valloc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000539 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000540 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000541 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000542 case LibFunc_vprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000543 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000544 Changed |= setDoesNotCapture(F, 0);
545 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000546 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000547 case LibFunc_vfprintf:
548 case LibFunc_vsprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000549 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000550 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000551 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000552 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000553 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000554 case LibFunc_vsnprintf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000555 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000556 Changed |= setDoesNotCapture(F, 0);
557 Changed |= setDoesNotCapture(F, 2);
558 Changed |= setOnlyReadsMemory(F, 2);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000559 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000560 case LibFunc_open:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000561 // May throw; "open" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000562 Changed |= setDoesNotCapture(F, 0);
563 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000564 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000565 case LibFunc_opendir:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000566 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000567 Changed |= setRetDoesNotAlias(F);
568 Changed |= setDoesNotCapture(F, 0);
569 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000570 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000571 case LibFunc_tmpfile:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000572 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000573 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000574 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000575 case LibFunc_times:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000576 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000577 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000578 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000579 case LibFunc_htonl:
580 case LibFunc_htons:
581 case LibFunc_ntohl:
582 case LibFunc_ntohs:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000583 Changed |= setDoesNotThrow(F);
584 Changed |= setDoesNotAccessMemory(F);
585 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000586 case LibFunc_lstat:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000587 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000588 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000589 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000590 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000591 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000592 case LibFunc_lchown:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000593 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000594 Changed |= setDoesNotCapture(F, 0);
595 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000596 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000597 case LibFunc_qsort:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000598 // May throw; places call through function pointer.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000599 Changed |= setDoesNotCapture(F, 3);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000600 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000601 case LibFunc_dunder_strdup:
602 case LibFunc_dunder_strndup:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000603 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000604 Changed |= setRetDoesNotAlias(F);
605 Changed |= setDoesNotCapture(F, 0);
606 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000607 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000608 case LibFunc_dunder_strtok_r:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000609 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000610 Changed |= setDoesNotCapture(F, 1);
611 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000612 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000613 case LibFunc_under_IO_getc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000614 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000615 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000616 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000617 case LibFunc_under_IO_putc:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000618 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000619 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000620 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000621 case LibFunc_dunder_isoc99_scanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000622 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000623 Changed |= setDoesNotCapture(F, 0);
624 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000625 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000626 case LibFunc_stat64:
627 case LibFunc_lstat64:
628 case LibFunc_statvfs64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000629 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000630 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000631 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000632 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000633 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000634 case LibFunc_dunder_isoc99_sscanf:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000635 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000636 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000637 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000638 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000639 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000640 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000641 case LibFunc_fopen64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000642 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000643 Changed |= setRetDoesNotAlias(F);
644 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000645 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000646 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000647 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000648 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000649 case LibFunc_fseeko64:
650 case LibFunc_ftello64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000651 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000652 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000653 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000654 case LibFunc_tmpfile64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000655 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000656 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000657 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000658 case LibFunc_fstat64:
659 case LibFunc_fstatvfs64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000660 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000661 Changed |= setDoesNotCapture(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000662 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000663 case LibFunc_open64:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000664 // May throw; "open" is a valid pthread cancellation point.
Reid Klecknera0b45f42017-05-03 18:17:31 +0000665 Changed |= setDoesNotCapture(F, 0);
666 Changed |= setOnlyReadsMemory(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000667 return Changed;
David L. Jonesd21529f2017-01-23 23:16:46 +0000668 case LibFunc_gettimeofday:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000669 // Currently some platforms have the restrict keyword on the arguments to
670 // gettimeofday. To be conservative, do not add noalias to gettimeofday's
671 // arguments.
672 Changed |= setDoesNotThrow(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000673 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougachab0624a22016-04-27 19:04:40 +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_Znwj: // new(unsigned int)
677 case LibFunc_Znwm: // new(unsigned long)
678 case LibFunc_Znaj: // new[](unsigned int)
679 case LibFunc_Znam: // new[](unsigned long)
680 case LibFunc_msvc_new_int: // new(unsigned int)
681 case LibFunc_msvc_new_longlong: // new(unsigned long long)
682 case LibFunc_msvc_new_array_int: // new[](unsigned int)
683 case LibFunc_msvc_new_array_longlong: // new[](unsigned long long)
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000684 // Operator new always returns a nonnull noalias pointer
Reid Klecknera0b45f42017-05-03 18:17:31 +0000685 Changed |= setRetNonNull(F);
686 Changed |= setRetDoesNotAlias(F);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000687 return Changed;
688 //TODO: add LibFunc entries for:
David L. Jonesd21529f2017-01-23 23:16:46 +0000689 //case LibFunc_memset_pattern4:
690 //case LibFunc_memset_pattern8:
691 case LibFunc_memset_pattern16:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000692 Changed |= setOnlyAccessesArgMemory(F);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000693 Changed |= setDoesNotCapture(F, 0);
Ahmed Bougacha44c19872016-04-27 19:04:43 +0000694 Changed |= setDoesNotCapture(F, 1);
Reid Klecknera0b45f42017-05-03 18:17:31 +0000695 Changed |= setOnlyReadsMemory(F, 1);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000696 return Changed;
697 // int __nvvm_reflect(const char *)
David L. Jonesd21529f2017-01-23 23:16:46 +0000698 case LibFunc_nvvm_reflect:
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000699 Changed |= setDoesNotAccessMemory(F);
700 Changed |= setDoesNotThrow(F);
701 return Changed;
702
703 default:
704 // FIXME: It'd be really nice to cover all the library functions we're
705 // aware of here.
706 return false;
707 }
708}
709
710//- Emit LibCalls ------------------------------------------------------------//
711
Sanjay Pateld3112a52016-01-19 19:46:10 +0000712Value *llvm::castToCStr(Value *V, IRBuilder<> &B) {
Matt Arsenaultbe558882014-04-23 20:58:57 +0000713 unsigned AS = V->getType()->getPointerAddressSpace();
714 return B.CreateBitCast(V, B.getInt8PtrTy(AS), "cstr");
Eric Christopher87abfc52010-03-05 22:25:30 +0000715}
716
Sanjay Pateld3112a52016-01-19 19:46:10 +0000717Value *llvm::emitStrLen(Value *Ptr, IRBuilder<> &B, const DataLayout &DL,
Nuno Lopes89702e92012-07-25 16:46:31 +0000718 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000719 if (!TLI->has(LibFunc_strlen))
Craig Topperf40110f2014-04-25 05:29:35 +0000720 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000721
Sanjay Pateld4af2972016-01-19 19:58:49 +0000722 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000723 LLVMContext &Context = B.GetInsertBlock()->getContext();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000724 Constant *StrLen = M->getOrInsertFunction("strlen", DL.getIntPtrType(Context),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000725 B.getInt8PtrTy());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000726 inferLibFuncAttributes(*M->getFunction("strlen"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000727 CallInst *CI = B.CreateCall(StrLen, castToCStr(Ptr, B), "strlen");
Evan Chengad6efbf2014-03-12 18:09:37 +0000728 if (const Function *F = dyn_cast<Function>(StrLen->stripPointerCasts()))
729 CI->setCallingConv(F->getCallingConv());
730
Eric Christopher87abfc52010-03-05 22:25:30 +0000731 return CI;
732}
733
Sanjay Pateld3112a52016-01-19 19:46:10 +0000734Value *llvm::emitStrChr(Value *Ptr, char C, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000735 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000736 if (!TLI->has(LibFunc_strchr))
Craig Topperf40110f2014-04-25 05:29:35 +0000737 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000738
Sanjay Pateld4af2972016-01-19 19:58:49 +0000739 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000740 Type *I8Ptr = B.getInt8PtrTy();
741 Type *I32Ty = B.getInt32Ty();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000742 Constant *StrChr =
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000743 M->getOrInsertFunction("strchr", I8Ptr, I8Ptr, I32Ty);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000744 inferLibFuncAttributes(*M->getFunction("strchr"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000745 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000746 StrChr, {castToCStr(Ptr, B), ConstantInt::get(I32Ty, C)}, "strchr");
Evan Chengad6efbf2014-03-12 18:09:37 +0000747 if (const Function *F = dyn_cast<Function>(StrChr->stripPointerCasts()))
748 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000749 return CI;
750}
751
Sanjay Pateld3112a52016-01-19 19:46:10 +0000752Value *llvm::emitStrNCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000753 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000754 if (!TLI->has(LibFunc_strncmp))
Craig Topperf40110f2014-04-25 05:29:35 +0000755 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000756
Sanjay Pateld4af2972016-01-19 19:58:49 +0000757 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000758 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000759 Value *StrNCmp = M->getOrInsertFunction("strncmp", B.getInt32Ty(),
760 B.getInt8PtrTy(), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000761 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000762 inferLibFuncAttributes(*M->getFunction("strncmp"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000763 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000764 StrNCmp, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, "strncmp");
Evan Chengad6efbf2014-03-12 18:09:37 +0000765
766 if (const Function *F = dyn_cast<Function>(StrNCmp->stripPointerCasts()))
767 CI->setCallingConv(F->getCallingConv());
768
Benjamin Kramer11188602010-06-15 21:34:25 +0000769 return CI;
770}
771
Sanjay Pateld3112a52016-01-19 19:46:10 +0000772Value *llvm::emitStrCpy(Value *Dst, Value *Src, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000773 const TargetLibraryInfo *TLI, StringRef Name) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000774 if (!TLI->has(LibFunc_strcpy))
Craig Topperf40110f2014-04-25 05:29:35 +0000775 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000776
Sanjay Pateld4af2972016-01-19 19:58:49 +0000777 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000778 Type *I8Ptr = B.getInt8PtrTy();
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000779 Value *StrCpy = M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr);
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000780 inferLibFuncAttributes(*M->getFunction(Name), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000781 CallInst *CI =
Sanjay Pateld3112a52016-01-19 19:46:10 +0000782 B.CreateCall(StrCpy, {castToCStr(Dst, B), castToCStr(Src, B)}, Name);
Evan Chengad6efbf2014-03-12 18:09:37 +0000783 if (const Function *F = dyn_cast<Function>(StrCpy->stripPointerCasts()))
784 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000785 return CI;
786}
787
Sanjay Pateld3112a52016-01-19 19:46:10 +0000788Value *llvm::emitStrNCpy(Value *Dst, Value *Src, Value *Len, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000789 const TargetLibraryInfo *TLI, StringRef Name) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000790 if (!TLI->has(LibFunc_strncpy))
Craig Topperf40110f2014-04-25 05:29:35 +0000791 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000792
Sanjay Pateld4af2972016-01-19 19:58:49 +0000793 Module *M = B.GetInsertBlock()->getModule();
Chris Lattner229907c2011-07-18 04:54:35 +0000794 Type *I8Ptr = B.getInt8PtrTy();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000795 Value *StrNCpy = M->getOrInsertFunction(Name, I8Ptr, I8Ptr, I8Ptr,
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000796 Len->getType());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000797 inferLibFuncAttributes(*M->getFunction(Name), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000798 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000799 StrNCpy, {castToCStr(Dst, B), castToCStr(Src, B), Len}, "strncpy");
Evan Chengad6efbf2014-03-12 18:09:37 +0000800 if (const Function *F = dyn_cast<Function>(StrNCpy->stripPointerCasts()))
801 CI->setCallingConv(F->getCallingConv());
Eric Christopher43dc11c2010-03-11 01:25:07 +0000802 return CI;
803}
804
Sanjay Pateld3112a52016-01-19 19:46:10 +0000805Value *llvm::emitMemCpyChk(Value *Dst, Value *Src, Value *Len, Value *ObjSize,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000806 IRBuilder<> &B, const DataLayout &DL,
Nuno Lopes89702e92012-07-25 16:46:31 +0000807 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000808 if (!TLI->has(LibFunc_memcpy_chk))
Craig Topperf40110f2014-04-25 05:29:35 +0000809 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000810
Sanjay Pateld4af2972016-01-19 19:58:49 +0000811 Module *M = B.GetInsertBlock()->getModule();
Reid Klecknerb5180542017-03-21 16:57:19 +0000812 AttributeList AS;
813 AS = AttributeList::get(M->getContext(), AttributeList::FunctionIndex,
814 Attribute::NoUnwind);
Chandler Carruth7ec50852012-11-01 08:07:29 +0000815 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000816 Value *MemCpy = M->getOrInsertFunction(
Reid Klecknerb5180542017-03-21 16:57:19 +0000817 "__memcpy_chk", AttributeList::get(M->getContext(), AS), B.getInt8PtrTy(),
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000818 B.getInt8PtrTy(), B.getInt8PtrTy(), DL.getIntPtrType(Context),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000819 DL.getIntPtrType(Context));
Sanjay Pateld3112a52016-01-19 19:46:10 +0000820 Dst = castToCStr(Dst, B);
821 Src = castToCStr(Src, B);
David Blaikieff6409d2015-05-18 22:13:54 +0000822 CallInst *CI = B.CreateCall(MemCpy, {Dst, Src, Len, ObjSize});
Evan Chengad6efbf2014-03-12 18:09:37 +0000823 if (const Function *F = dyn_cast<Function>(MemCpy->stripPointerCasts()))
824 CI->setCallingConv(F->getCallingConv());
Evan Chengd9e82232010-03-23 15:48:04 +0000825 return CI;
826}
827
Sanjay Pateld3112a52016-01-19 19:46:10 +0000828Value *llvm::emitMemChr(Value *Ptr, Value *Val, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000829 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000830 if (!TLI->has(LibFunc_memchr))
Craig Topperf40110f2014-04-25 05:29:35 +0000831 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000832
Sanjay Pateld4af2972016-01-19 19:58:49 +0000833 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000834 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000835 Value *MemChr = M->getOrInsertFunction("memchr", B.getInt8PtrTy(),
836 B.getInt8PtrTy(), B.getInt32Ty(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000837 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000838 inferLibFuncAttributes(*M->getFunction("memchr"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000839 CallInst *CI = B.CreateCall(MemChr, {castToCStr(Ptr, B), Val, Len}, "memchr");
Evan Chengad6efbf2014-03-12 18:09:37 +0000840
841 if (const Function *F = dyn_cast<Function>(MemChr->stripPointerCasts()))
842 CI->setCallingConv(F->getCallingConv());
843
Eric Christopher87abfc52010-03-05 22:25:30 +0000844 return CI;
845}
846
Sanjay Pateld3112a52016-01-19 19:46:10 +0000847Value *llvm::emitMemCmp(Value *Ptr1, Value *Ptr2, Value *Len, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000848 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000849 if (!TLI->has(LibFunc_memcmp))
Craig Topperf40110f2014-04-25 05:29:35 +0000850 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000851
Sanjay Pateld4af2972016-01-19 19:58:49 +0000852 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000853 LLVMContext &Context = B.GetInsertBlock()->getContext();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000854 Value *MemCmp = M->getOrInsertFunction("memcmp", B.getInt32Ty(),
855 B.getInt8PtrTy(), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000856 DL.getIntPtrType(Context));
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000857 inferLibFuncAttributes(*M->getFunction("memcmp"), *TLI);
David Blaikieff6409d2015-05-18 22:13:54 +0000858 CallInst *CI = B.CreateCall(
Sanjay Pateld3112a52016-01-19 19:46:10 +0000859 MemCmp, {castToCStr(Ptr1, B), castToCStr(Ptr2, B), Len}, "memcmp");
Evan Chengad6efbf2014-03-12 18:09:37 +0000860
861 if (const Function *F = dyn_cast<Function>(MemCmp->stripPointerCasts()))
862 CI->setCallingConv(F->getCallingConv());
863
Eric Christopher87abfc52010-03-05 22:25:30 +0000864 return CI;
865}
866
Yi Jiang6ab044e2013-12-16 22:42:40 +0000867/// Append a suffix to the function name according to the type of 'Op'.
Sanjay Pateld3112a52016-01-19 19:46:10 +0000868static void appendTypeSuffix(Value *Op, StringRef &Name,
Sanjay Patelb50325e2016-01-19 19:17:47 +0000869 SmallString<20> &NameBuffer) {
Yi Jiang6ab044e2013-12-16 22:42:40 +0000870 if (!Op->getType()->isDoubleTy()) {
871 NameBuffer += Name;
872
873 if (Op->getType()->isFloatTy())
874 NameBuffer += 'f';
875 else
876 NameBuffer += 'l';
877
878 Name = NameBuffer;
879 }
Yi Jiang6ab044e2013-12-16 22:42:40 +0000880}
881
Sanjay Pateld3112a52016-01-19 19:46:10 +0000882Value *llvm::emitUnaryFloatFnCall(Value *Op, StringRef Name, IRBuilder<> &B,
Reid Klecknerb5180542017-03-21 16:57:19 +0000883 const AttributeList &Attrs) {
Benjamin Kramerb106bcc2011-11-15 19:12:09 +0000884 SmallString<20> NameBuffer;
Sanjay Pateld3112a52016-01-19 19:46:10 +0000885 appendTypeSuffix(Op, Name, NameBuffer);
Eric Christopher87abfc52010-03-05 22:25:30 +0000886
Sanjay Pateld4af2972016-01-19 19:58:49 +0000887 Module *M = B.GetInsertBlock()->getModule();
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000888 Value *Callee = M->getOrInsertFunction(Name, Op->getType(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000889 Op->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +0000890 CallInst *CI = B.CreateCall(Callee, Op, Name);
Matt Arsenault6a288c12017-05-03 02:26:10 +0000891
892 // The incoming attribute set may have come from a speculatable intrinsic, but
893 // is being replaced with a library call which is not allowed to be
894 // speculatable.
895 CI->setAttributes(Attrs.removeAttribute(B.getContext(),
896 AttributeList::FunctionIndex,
897 Attribute::Speculatable));
Evan Chengad6efbf2014-03-12 18:09:37 +0000898 if (const Function *F = dyn_cast<Function>(Callee->stripPointerCasts()))
899 CI->setCallingConv(F->getCallingConv());
900
Eric Christopher87abfc52010-03-05 22:25:30 +0000901 return CI;
902}
903
Sanjay Pateld3112a52016-01-19 19:46:10 +0000904Value *llvm::emitBinaryFloatFnCall(Value *Op1, Value *Op2, StringRef Name,
Reid Klecknerb5180542017-03-21 16:57:19 +0000905 IRBuilder<> &B, const AttributeList &Attrs) {
Yi Jiang6ab044e2013-12-16 22:42:40 +0000906 SmallString<20> NameBuffer;
Sanjay Pateld3112a52016-01-19 19:46:10 +0000907 appendTypeSuffix(Op1, Name, NameBuffer);
Yi Jiang6ab044e2013-12-16 22:42:40 +0000908
Sanjay Pateld4af2972016-01-19 19:58:49 +0000909 Module *M = B.GetInsertBlock()->getModule();
Sanjay Patelb50325e2016-01-19 19:17:47 +0000910 Value *Callee = M->getOrInsertFunction(Name, Op1->getType(), Op1->getType(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000911 Op2->getType());
David Blaikieff6409d2015-05-18 22:13:54 +0000912 CallInst *CI = B.CreateCall(Callee, {Op1, Op2}, Name);
Yi Jiang6ab044e2013-12-16 22:42:40 +0000913 CI->setAttributes(Attrs);
Evan Chengad6efbf2014-03-12 18:09:37 +0000914 if (const Function *F = dyn_cast<Function>(Callee->stripPointerCasts()))
915 CI->setCallingConv(F->getCallingConv());
916
Yi Jiang6ab044e2013-12-16 22:42:40 +0000917 return CI;
918}
919
Sanjay Pateld3112a52016-01-19 19:46:10 +0000920Value *llvm::emitPutChar(Value *Char, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000921 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000922 if (!TLI->has(LibFunc_putchar))
Craig Topperf40110f2014-04-25 05:29:35 +0000923 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000924
Sanjay Pateld4af2972016-01-19 19:58:49 +0000925 Module *M = B.GetInsertBlock()->getModule();
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000926 Value *PutChar = M->getOrInsertFunction("putchar", B.getInt32Ty(), B.getInt32Ty());
Craig Topper0f5063c2017-03-17 23:48:02 +0000927 inferLibFuncAttributes(*M->getFunction("putchar"), *TLI);
Eric Christopher87abfc52010-03-05 22:25:30 +0000928 CallInst *CI = B.CreateCall(PutChar,
929 B.CreateIntCast(Char,
930 B.getInt32Ty(),
931 /*isSigned*/true,
932 "chari"),
933 "putchar");
Evan Chengad6efbf2014-03-12 18:09:37 +0000934
935 if (const Function *F = dyn_cast<Function>(PutChar->stripPointerCasts()))
936 CI->setCallingConv(F->getCallingConv());
Eric Christopher87abfc52010-03-05 22:25:30 +0000937 return CI;
938}
939
Sanjay Pateld3112a52016-01-19 19:46:10 +0000940Value *llvm::emitPutS(Value *Str, IRBuilder<> &B,
Nuno Lopes89702e92012-07-25 16:46:31 +0000941 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000942 if (!TLI->has(LibFunc_puts))
Craig Topperf40110f2014-04-25 05:29:35 +0000943 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000944
Sanjay Pateld4af2972016-01-19 19:58:49 +0000945 Module *M = B.GetInsertBlock()->getModule();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000946 Value *PutS =
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000947 M->getOrInsertFunction("puts", B.getInt32Ty(), B.getInt8PtrTy());
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000948 inferLibFuncAttributes(*M->getFunction("puts"), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000949 CallInst *CI = B.CreateCall(PutS, castToCStr(Str, B), "puts");
Evan Chengad6efbf2014-03-12 18:09:37 +0000950 if (const Function *F = dyn_cast<Function>(PutS->stripPointerCasts()))
951 CI->setCallingConv(F->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +0000952 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +0000953}
954
Sanjay Pateld3112a52016-01-19 19:46:10 +0000955Value *llvm::emitFPutC(Value *Char, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000956 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000957 if (!TLI->has(LibFunc_fputc))
Craig Topperf40110f2014-04-25 05:29:35 +0000958 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000959
Sanjay Pateld4af2972016-01-19 19:58:49 +0000960 Module *M = B.GetInsertBlock()->getModule();
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000961 Constant *F = M->getOrInsertFunction("fputc", B.getInt32Ty(), B.getInt32Ty(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000962 File->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +0000963 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000964 inferLibFuncAttributes(*M->getFunction("fputc"), *TLI);
Eric Christopher87abfc52010-03-05 22:25:30 +0000965 Char = B.CreateIntCast(Char, B.getInt32Ty(), /*isSigned*/true,
966 "chari");
David Blaikieff6409d2015-05-18 22:13:54 +0000967 CallInst *CI = B.CreateCall(F, {Char, File}, "fputc");
Evan Chengad6efbf2014-03-12 18:09:37 +0000968
969 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
970 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +0000971 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +0000972}
973
Sanjay Pateld3112a52016-01-19 19:46:10 +0000974Value *llvm::emitFPutS(Value *Str, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000975 const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000976 if (!TLI->has(LibFunc_fputs))
Craig Topperf40110f2014-04-25 05:29:35 +0000977 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000978
Sanjay Pateld4af2972016-01-19 19:58:49 +0000979 Module *M = B.GetInsertBlock()->getModule();
David L. Jonesd21529f2017-01-23 23:16:46 +0000980 StringRef FPutsName = TLI->getName(LibFunc_fputs);
Mehdi Aminidb11fdf2017-04-06 20:23:57 +0000981 Constant *F = M->getOrInsertFunction(
Serge Guelton59a2d7b2017-04-11 15:01:18 +0000982 FPutsName, B.getInt32Ty(), B.getInt8PtrTy(), File->getType());
Eric Christopher87abfc52010-03-05 22:25:30 +0000983 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +0000984 inferLibFuncAttributes(*M->getFunction(FPutsName), *TLI);
Sanjay Pateld3112a52016-01-19 19:46:10 +0000985 CallInst *CI = B.CreateCall(F, {castToCStr(Str, B), File}, "fputs");
Evan Chengad6efbf2014-03-12 18:09:37 +0000986
987 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
988 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +0000989 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +0000990}
991
Sanjay Pateld3112a52016-01-19 19:46:10 +0000992Value *llvm::emitFWrite(Value *Ptr, Value *Size, Value *File, IRBuilder<> &B,
Mehdi Aminia28d91d2015-03-10 02:37:25 +0000993 const DataLayout &DL, const TargetLibraryInfo *TLI) {
David L. Jonesd21529f2017-01-23 23:16:46 +0000994 if (!TLI->has(LibFunc_fwrite))
Craig Topperf40110f2014-04-25 05:29:35 +0000995 return nullptr;
Nuno Lopes89702e92012-07-25 16:46:31 +0000996
Sanjay Pateld4af2972016-01-19 19:58:49 +0000997 Module *M = B.GetInsertBlock()->getModule();
Chandler Carruth7ec50852012-11-01 08:07:29 +0000998 LLVMContext &Context = B.GetInsertBlock()->getContext();
David L. Jonesd21529f2017-01-23 23:16:46 +0000999 StringRef FWriteName = TLI->getName(LibFunc_fwrite);
Ahmed Bougachab0624a22016-04-27 19:04:40 +00001000 Constant *F = M->getOrInsertFunction(
1001 FWriteName, DL.getIntPtrType(Context), B.getInt8PtrTy(),
Serge Guelton59a2d7b2017-04-11 15:01:18 +00001002 DL.getIntPtrType(Context), DL.getIntPtrType(Context), File->getType());
1003
Eric Christopher87abfc52010-03-05 22:25:30 +00001004 if (File->getType()->isPointerTy())
Ahmed Bougachab0624a22016-04-27 19:04:40 +00001005 inferLibFuncAttributes(*M->getFunction(FWriteName), *TLI);
Mehdi Aminia28d91d2015-03-10 02:37:25 +00001006 CallInst *CI =
Sanjay Pateld3112a52016-01-19 19:46:10 +00001007 B.CreateCall(F, {castToCStr(Ptr, B), Size,
David Blaikieff6409d2015-05-18 22:13:54 +00001008 ConstantInt::get(DL.getIntPtrType(Context), 1), File});
Evan Chengad6efbf2014-03-12 18:09:37 +00001009
1010 if (const Function *Fn = dyn_cast<Function>(F->stripPointerCasts()))
1011 CI->setCallingConv(Fn->getCallingConv());
Nuno Lopes89702e92012-07-25 16:46:31 +00001012 return CI;
Eric Christopher87abfc52010-03-05 22:25:30 +00001013}