blob: d07199cb3360cb4aa4269d79bbf6331f2b4626ce [file] [log] [blame]
Chandler Carruth74b6a772013-01-07 15:35:46 +00001//===- llvm/unittest/IR/Metadata.cpp - Metadata unit tests ----------------===//
Nick Lewycky49f89192009-04-04 07:22:01 +00002//
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
Duncan P. N. Exon Smith845755c42015-01-13 00:46:34 +000010#include "llvm/ADT/STLExtras.h"
Chandler Carruth9fb823b2013-01-02 11:36:10 +000011#include "llvm/IR/Metadata.h"
12#include "llvm/IR/Constants.h"
13#include "llvm/IR/Instructions.h"
14#include "llvm/IR/LLVMContext.h"
15#include "llvm/IR/Module.h"
16#include "llvm/IR/Type.h"
Chandler Carruth130cec22012-12-04 10:23:08 +000017#include "llvm/Support/raw_ostream.h"
Chandler Carruth130cec22012-12-04 10:23:08 +000018#include "gtest/gtest.h"
Nick Lewycky49f89192009-04-04 07:22:01 +000019using namespace llvm;
20
21namespace {
22
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000023class MetadataTest : public testing::Test {
24protected:
25 LLVMContext Context;
Duncan P. N. Exon Smithfee167f2014-12-16 07:09:37 +000026 MDNode *getNode() { return MDNode::get(Context, None); }
27 MDNode *getNode(Metadata *MD) { return MDNode::get(Context, MD); }
28 MDNode *getNode(Metadata *MD1, Metadata *MD2) {
29 Metadata *MDs[] = {MD1, MD2};
30 return MDNode::get(Context, MDs);
31 }
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000032};
33typedef MetadataTest MDStringTest;
Owen Anderson23587322009-07-31 21:38:10 +000034
Nick Lewycky49f89192009-04-04 07:22:01 +000035// Test that construction of MDString with different value produces different
36// MDString objects, even with the same string pointer and nulls in the string.
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000037TEST_F(MDStringTest, CreateDifferent) {
Nick Lewycky49f89192009-04-04 07:22:01 +000038 char x[3] = { 'f', 0, 'A' };
Owen Anderson23587322009-07-31 21:38:10 +000039 MDString *s1 = MDString::get(Context, StringRef(&x[0], 3));
Nick Lewycky49f89192009-04-04 07:22:01 +000040 x[2] = 'B';
Owen Anderson23587322009-07-31 21:38:10 +000041 MDString *s2 = MDString::get(Context, StringRef(&x[0], 3));
Nick Lewycky49f89192009-04-04 07:22:01 +000042 EXPECT_NE(s1, s2);
43}
44
45// Test that creation of MDStrings with the same string contents produces the
46// same MDString object, even with different pointers.
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000047TEST_F(MDStringTest, CreateSame) {
Nick Lewycky49f89192009-04-04 07:22:01 +000048 char x[4] = { 'a', 'b', 'c', 'X' };
49 char y[4] = { 'a', 'b', 'c', 'Y' };
50
Owen Anderson23587322009-07-31 21:38:10 +000051 MDString *s1 = MDString::get(Context, StringRef(&x[0], 3));
52 MDString *s2 = MDString::get(Context, StringRef(&y[0], 3));
Nick Lewycky49f89192009-04-04 07:22:01 +000053 EXPECT_EQ(s1, s2);
54}
55
56// Test that MDString prints out the string we fed it.
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000057TEST_F(MDStringTest, PrintingSimple) {
Nick Lewycky49f89192009-04-04 07:22:01 +000058 char *str = new char[13];
59 strncpy(str, "testing 1 2 3", 13);
Owen Anderson23587322009-07-31 21:38:10 +000060 MDString *s = MDString::get(Context, StringRef(str, 13));
Nick Lewycky49f89192009-04-04 07:22:01 +000061 strncpy(str, "aaaaaaaaaaaaa", 13);
62 delete[] str;
63
Chris Lattnerbe354a62009-08-23 04:47:35 +000064 std::string Str;
65 raw_string_ostream oss(Str);
Nick Lewycky49f89192009-04-04 07:22:01 +000066 s->print(oss);
Duncan P. N. Exon Smithbb7d2fb2014-12-16 07:40:31 +000067 EXPECT_STREQ("!\"testing 1 2 3\"", oss.str().c_str());
Nick Lewycky49f89192009-04-04 07:22:01 +000068}
69
70// Test printing of MDString with non-printable characters.
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000071TEST_F(MDStringTest, PrintingComplex) {
Jeffrey Yasskin065c3572011-08-30 20:53:29 +000072 char str[5] = {0, '\n', '"', '\\', (char)-1};
Owen Anderson23587322009-07-31 21:38:10 +000073 MDString *s = MDString::get(Context, StringRef(str+0, 5));
Chris Lattnerbe354a62009-08-23 04:47:35 +000074 std::string Str;
75 raw_string_ostream oss(Str);
Nick Lewycky49f89192009-04-04 07:22:01 +000076 s->print(oss);
Duncan P. N. Exon Smithbb7d2fb2014-12-16 07:40:31 +000077 EXPECT_STREQ("!\"\\00\\0A\\22\\5C\\FF\"", oss.str().c_str());
Nick Lewycky49f89192009-04-04 07:22:01 +000078}
79
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000080typedef MetadataTest MDNodeTest;
81
Nick Lewycky49f89192009-04-04 07:22:01 +000082// Test the two constructors, and containing other Constants.
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +000083TEST_F(MDNodeTest, Simple) {
Nick Lewycky49f89192009-04-04 07:22:01 +000084 char x[3] = { 'a', 'b', 'c' };
85 char y[3] = { '1', '2', '3' };
86
Owen Anderson23587322009-07-31 21:38:10 +000087 MDString *s1 = MDString::get(Context, StringRef(&x[0], 3));
88 MDString *s2 = MDString::get(Context, StringRef(&y[0], 3));
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +000089 ConstantAsMetadata *CI = ConstantAsMetadata::get(
90 ConstantInt::get(getGlobalContext(), APInt(8, 0)));
Nick Lewycky49f89192009-04-04 07:22:01 +000091
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +000092 std::vector<Metadata *> V;
Nick Lewycky49f89192009-04-04 07:22:01 +000093 V.push_back(s1);
94 V.push_back(CI);
95 V.push_back(s2);
96
Jay Foad5514afe2011-04-21 19:59:31 +000097 MDNode *n1 = MDNode::get(Context, V);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +000098 Metadata *const c1 = n1;
Jay Foad5514afe2011-04-21 19:59:31 +000099 MDNode *n2 = MDNode::get(Context, c1);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000100 Metadata *const c2 = n2;
Jay Foad5514afe2011-04-21 19:59:31 +0000101 MDNode *n3 = MDNode::get(Context, V);
Duncan Sands26a80f32012-03-31 08:20:11 +0000102 MDNode *n4 = MDNode::getIfExists(Context, V);
103 MDNode *n5 = MDNode::getIfExists(Context, c1);
104 MDNode *n6 = MDNode::getIfExists(Context, c2);
Nick Lewycky49f89192009-04-04 07:22:01 +0000105 EXPECT_NE(n1, n2);
Devang Patelf7188322009-09-03 01:39:20 +0000106 EXPECT_EQ(n1, n3);
Duncan Sands26a80f32012-03-31 08:20:11 +0000107 EXPECT_EQ(n4, n1);
108 EXPECT_EQ(n5, n2);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000109 EXPECT_EQ(n6, (Metadata *)nullptr);
Nick Lewycky49f89192009-04-04 07:22:01 +0000110
Chris Lattner9b493022009-12-31 01:22:29 +0000111 EXPECT_EQ(3u, n1->getNumOperands());
112 EXPECT_EQ(s1, n1->getOperand(0));
113 EXPECT_EQ(CI, n1->getOperand(1));
114 EXPECT_EQ(s2, n1->getOperand(2));
Nick Lewycky49f89192009-04-04 07:22:01 +0000115
Chris Lattner9b493022009-12-31 01:22:29 +0000116 EXPECT_EQ(1u, n2->getNumOperands());
117 EXPECT_EQ(n1, n2->getOperand(0));
Nick Lewycky49f89192009-04-04 07:22:01 +0000118}
Nick Lewyckyb8f9b7a2009-05-10 20:57:05 +0000119
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +0000120TEST_F(MDNodeTest, Delete) {
Owen Anderson55f1c092009-08-13 21:58:54 +0000121 Constant *C = ConstantInt::get(Type::getInt32Ty(getGlobalContext()), 1);
122 Instruction *I = new BitCastInst(C, Type::getInt32Ty(getGlobalContext()));
Nick Lewyckyb8f9b7a2009-05-10 20:57:05 +0000123
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000124 Metadata *const V = LocalAsMetadata::get(I);
Jay Foad5514afe2011-04-21 19:59:31 +0000125 MDNode *n = MDNode::get(Context, V);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000126 TrackingMDRef wvh(n);
Nick Lewyckyb8f9b7a2009-05-10 20:57:05 +0000127
128 EXPECT_EQ(n, wvh);
129
130 delete I;
Nick Lewyckyb8f9b7a2009-05-10 20:57:05 +0000131}
Devang Patel0924b332009-07-30 00:03:41 +0000132
Duncan P. N. Exon Smithb565b102015-01-12 20:19:54 +0000133TEST_F(MDNodeTest, DeleteMDNodeFwdDecl) {
134 delete MDNode::getTemporary(Context, None);
135}
136
Duncan P. N. Exon Smithac8ee282014-12-07 19:52:06 +0000137TEST_F(MDNodeTest, SelfReference) {
Duncan P. N. Exon Smith8c662732014-12-16 07:45:05 +0000138 // !0 = !{!0}
139 // !1 = !{!0}
Duncan P. N. Exon Smithac8ee282014-12-07 19:52:06 +0000140 {
141 MDNode *Temp = MDNode::getTemporary(Context, None);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000142 Metadata *Args[] = {Temp};
Duncan P. N. Exon Smithac8ee282014-12-07 19:52:06 +0000143 MDNode *Self = MDNode::get(Context, Args);
144 Self->replaceOperandWith(0, Self);
145 MDNode::deleteTemporary(Temp);
146 ASSERT_EQ(Self, Self->getOperand(0));
147
148 // Self-references should be distinct, so MDNode::get() should grab a
149 // uniqued node that references Self, not Self.
150 Args[0] = Self;
151 MDNode *Ref1 = MDNode::get(Context, Args);
152 MDNode *Ref2 = MDNode::get(Context, Args);
153 EXPECT_NE(Self, Ref1);
154 EXPECT_EQ(Ref1, Ref2);
155 }
156
Duncan P. N. Exon Smith8c662732014-12-16 07:45:05 +0000157 // !0 = !{!0, !{}}
158 // !1 = !{!0, !{}}
Duncan P. N. Exon Smithac8ee282014-12-07 19:52:06 +0000159 {
160 MDNode *Temp = MDNode::getTemporary(Context, None);
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000161 Metadata *Args[] = {Temp, MDNode::get(Context, None)};
Duncan P. N. Exon Smithac8ee282014-12-07 19:52:06 +0000162 MDNode *Self = MDNode::get(Context, Args);
163 Self->replaceOperandWith(0, Self);
164 MDNode::deleteTemporary(Temp);
165 ASSERT_EQ(Self, Self->getOperand(0));
166
167 // Self-references should be distinct, so MDNode::get() should grab a
168 // uniqued node that references Self, not Self itself.
169 Args[0] = Self;
170 MDNode *Ref1 = MDNode::get(Context, Args);
171 MDNode *Ref2 = MDNode::get(Context, Args);
172 EXPECT_NE(Self, Ref1);
173 EXPECT_EQ(Ref1, Ref2);
174 }
175}
176
Duncan P. N. Exon Smithfee167f2014-12-16 07:09:37 +0000177TEST_F(MDNodeTest, Print) {
178 Constant *C = ConstantInt::get(Type::getInt32Ty(Context), 7);
179 MDString *S = MDString::get(Context, "foo");
180 MDNode *N0 = getNode();
181 MDNode *N1 = getNode(N0);
182 MDNode *N2 = getNode(N0, N1);
183
184 Metadata *Args[] = {ConstantAsMetadata::get(C), S, nullptr, N0, N1, N2};
185 MDNode *N = MDNode::get(Context, Args);
186
187 std::string Expected;
188 {
189 raw_string_ostream OS(Expected);
Duncan P. N. Exon Smithbb7d2fb2014-12-16 07:40:31 +0000190 OS << "!{";
Duncan P. N. Exon Smithfee167f2014-12-16 07:09:37 +0000191 C->printAsOperand(OS);
192 OS << ", ";
Duncan P. N. Exon Smithbb7d2fb2014-12-16 07:40:31 +0000193 S->printAsOperand(OS);
Duncan P. N. Exon Smithfee167f2014-12-16 07:09:37 +0000194 OS << ", null";
195 MDNode *Nodes[] = {N0, N1, N2};
196 for (auto *Node : Nodes)
197 OS << ", <" << (void *)Node << ">";
198 OS << "}\n";
199 }
200
201 std::string Actual;
202 {
203 raw_string_ostream OS(Actual);
204 N->print(OS);
205 }
206
207 EXPECT_EQ(Expected, Actual);
208}
209
Duncan P. N. Exon Smithbcd960a2015-01-05 23:31:54 +0000210TEST_F(MDNodeTest, NullOperand) {
211 // metadata !{}
212 MDNode *Empty = MDNode::get(Context, None);
213
214 // metadata !{metadata !{}}
215 Metadata *Ops[] = {Empty};
216 MDNode *N = MDNode::get(Context, Ops);
217 ASSERT_EQ(Empty, N->getOperand(0));
218
219 // metadata !{metadata !{}} => metadata !{null}
220 N->replaceOperandWith(0, nullptr);
221 ASSERT_EQ(nullptr, N->getOperand(0));
222
223 // metadata !{null}
224 Ops[0] = nullptr;
225 MDNode *NullOp = MDNode::get(Context, Ops);
226 ASSERT_EQ(nullptr, NullOp->getOperand(0));
227 EXPECT_EQ(N, NullOp);
228}
229
Duncan P. N. Exon Smith136ea3f2015-01-07 21:35:38 +0000230TEST_F(MDNodeTest, DistinctOnUniquingCollision) {
231 // !{}
232 MDNode *Empty = MDNode::get(Context, None);
233 ASSERT_TRUE(Empty->isResolved());
234 EXPECT_FALSE(Empty->isDistinct());
235
236 // !{!{}}
237 Metadata *Wrapped1Ops[] = {Empty};
238 MDNode *Wrapped1 = MDNode::get(Context, Wrapped1Ops);
239 ASSERT_EQ(Empty, Wrapped1->getOperand(0));
240 ASSERT_TRUE(Wrapped1->isResolved());
241 EXPECT_FALSE(Wrapped1->isDistinct());
242
243 // !{!{!{}}}
244 Metadata *Wrapped2Ops[] = {Wrapped1};
245 MDNode *Wrapped2 = MDNode::get(Context, Wrapped2Ops);
246 ASSERT_EQ(Wrapped1, Wrapped2->getOperand(0));
247 ASSERT_TRUE(Wrapped2->isResolved());
248 EXPECT_FALSE(Wrapped2->isDistinct());
249
250 // !{!{!{}}} => !{!{}}
251 Wrapped2->replaceOperandWith(0, Empty);
252 ASSERT_EQ(Empty, Wrapped2->getOperand(0));
253 EXPECT_TRUE(Wrapped2->isDistinct());
254 EXPECT_FALSE(Wrapped1->isDistinct());
255}
256
Duncan P. N. Exon Smith5e5b8502015-01-07 22:24:46 +0000257TEST_F(MDNodeTest, getDistinct) {
258 // !{}
259 MDNode *Empty = MDNode::get(Context, None);
260 ASSERT_TRUE(Empty->isResolved());
261 ASSERT_FALSE(Empty->isDistinct());
262 ASSERT_EQ(Empty, MDNode::get(Context, None));
263
264 // distinct !{}
265 MDNode *Distinct1 = MDNode::getDistinct(Context, None);
266 MDNode *Distinct2 = MDNode::getDistinct(Context, None);
267 EXPECT_TRUE(Distinct1->isResolved());
268 EXPECT_TRUE(Distinct2->isDistinct());
269 EXPECT_NE(Empty, Distinct1);
270 EXPECT_NE(Empty, Distinct2);
271 EXPECT_NE(Distinct1, Distinct2);
272
273 // !{}
274 ASSERT_EQ(Empty, MDNode::get(Context, None));
275}
276
Duncan P. N. Exon Smithd1474ee2015-01-12 18:41:26 +0000277TEST_F(MDNodeTest, TempIsDistinct) {
278 MDNode *T = MDNode::getTemporary(Context, None);
279 EXPECT_TRUE(T->isDistinct());
280 MDNode::deleteTemporary(T);
281}
282
Duncan P. N. Exon Smith5e5b8502015-01-07 22:24:46 +0000283TEST_F(MDNodeTest, getDistinctWithUnresolvedOperands) {
284 // temporary !{}
285 MDNodeFwdDecl *Temp = MDNode::getTemporary(Context, None);
286 ASSERT_FALSE(Temp->isResolved());
287
288 // distinct !{temporary !{}}
289 Metadata *Ops[] = {Temp};
290 MDNode *Distinct = MDNode::getDistinct(Context, Ops);
291 EXPECT_TRUE(Distinct->isResolved());
292 EXPECT_EQ(Temp, Distinct->getOperand(0));
293
294 // temporary !{} => !{}
295 MDNode *Empty = MDNode::get(Context, None);
296 Temp->replaceAllUsesWith(Empty);
297 MDNode::deleteTemporary(Temp);
298 EXPECT_EQ(Empty, Distinct->getOperand(0));
299}
300
Duncan P. N. Exon Smith5f461892015-01-12 19:22:04 +0000301TEST_F(MDNodeTest, handleChangedOperandRecursion) {
302 // !0 = !{}
303 MDNode *N0 = MDNode::get(Context, None);
304
305 // !1 = !{!3, null}
Duncan P. N. Exon Smith1e05ea62015-01-13 00:57:27 +0000306 std::unique_ptr<MDNodeFwdDecl> Temp3(MDNode::getTemporary(Context, None));
307 Metadata *Ops1[] = {Temp3.get(), nullptr};
Duncan P. N. Exon Smith5f461892015-01-12 19:22:04 +0000308 MDNode *N1 = MDNode::get(Context, Ops1);
309
310 // !2 = !{!3, !0}
Duncan P. N. Exon Smith1e05ea62015-01-13 00:57:27 +0000311 Metadata *Ops2[] = {Temp3.get(), N0};
Duncan P. N. Exon Smith5f461892015-01-12 19:22:04 +0000312 MDNode *N2 = MDNode::get(Context, Ops2);
313
314 // !3 = !{!2}
315 Metadata *Ops3[] = {N2};
316 MDNode *N3 = MDNode::get(Context, Ops3);
317 Temp3->replaceAllUsesWith(N3);
318
319 // !4 = !{!1}
320 Metadata *Ops4[] = {N1};
321 MDNode *N4 = MDNode::get(Context, Ops4);
322
323 // Confirm that the cycle prevented RAUW from getting dropped.
324 EXPECT_TRUE(N0->isResolved());
325 EXPECT_FALSE(N1->isResolved());
326 EXPECT_FALSE(N2->isResolved());
327 EXPECT_FALSE(N3->isResolved());
328 EXPECT_FALSE(N4->isResolved());
329
330 // Create a couple of distinct nodes to observe what's going on.
331 //
332 // !5 = distinct !{!2}
333 // !6 = distinct !{!3}
334 Metadata *Ops5[] = {N2};
335 MDNode *N5 = MDNode::getDistinct(Context, Ops5);
336 Metadata *Ops6[] = {N3};
337 MDNode *N6 = MDNode::getDistinct(Context, Ops6);
338
339 // Mutate !2 to look like !1, causing a uniquing collision (and an RAUW).
340 // This will ripple up, with !3 colliding with !4, and RAUWing. Since !2
341 // references !3, this can cause a re-entry of handleChangedOperand() when !3
342 // is not ready for it.
343 //
344 // !2->replaceOperandWith(1, nullptr)
345 // !2: !{!3, !0} => !{!3, null}
346 // !2->replaceAllUsesWith(!1)
347 // !3: !{!2] => !{!1}
348 // !3->replaceAllUsesWith(!4)
349 N2->replaceOperandWith(1, nullptr);
350
351 // If all has gone well, N2 and N3 will have been RAUW'ed and deleted from
352 // under us. Just check that the other nodes are sane.
353 //
354 // !1 = !{!4, null}
355 // !4 = !{!1}
356 // !5 = distinct !{!1}
357 // !6 = distinct !{!4}
358 EXPECT_EQ(N4, N1->getOperand(0));
359 EXPECT_EQ(N1, N4->getOperand(0));
360 EXPECT_EQ(N1, N5->getOperand(0));
361 EXPECT_EQ(N4, N6->getOperand(0));
362}
363
Duncan P. N. Exon Smith845755c42015-01-13 00:46:34 +0000364TEST_F(MDNodeTest, replaceResolvedOperand) {
365 // Check code for replacing one resolved operand with another. If doing this
366 // directly (via replaceOperandWith()) becomes illegal, change the operand to
367 // a global value that gets RAUW'ed.
368 //
369 // Use a temporary node to keep N from being resolved.
370 std::unique_ptr<MDNodeFwdDecl> Temp(MDNodeFwdDecl::get(Context, None));
371 Metadata *Ops[] = {nullptr, Temp.get()};
372
373 MDNode *Empty = MDTuple::get(Context, {});
374 MDNode *N = MDTuple::get(Context, Ops);
375 EXPECT_EQ(nullptr, N->getOperand(0));
376 ASSERT_FALSE(N->isResolved());
377
378 // Check code for replacing resolved nodes.
379 N->replaceOperandWith(0, Empty);
380 EXPECT_EQ(Empty, N->getOperand(0));
381
382 // Check code for adding another unresolved operand.
383 N->replaceOperandWith(0, Temp.get());
384 EXPECT_EQ(Temp.get(), N->getOperand(0));
385
386 // Remove the references to Temp; required for teardown.
387 Temp->replaceAllUsesWith(nullptr);
388}
389
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000390typedef MetadataTest MetadataAsValueTest;
391
392TEST_F(MetadataAsValueTest, MDNode) {
393 MDNode *N = MDNode::get(Context, None);
394 auto *V = MetadataAsValue::get(Context, N);
395 EXPECT_TRUE(V->getType()->isMetadataTy());
396 EXPECT_EQ(N, V->getMetadata());
397
398 auto *V2 = MetadataAsValue::get(Context, N);
399 EXPECT_EQ(V, V2);
400}
401
402TEST_F(MetadataAsValueTest, MDNodeMDNode) {
403 MDNode *N = MDNode::get(Context, None);
404 Metadata *Ops[] = {N};
405 MDNode *N2 = MDNode::get(Context, Ops);
406 auto *V = MetadataAsValue::get(Context, N2);
407 EXPECT_TRUE(V->getType()->isMetadataTy());
408 EXPECT_EQ(N2, V->getMetadata());
409
410 auto *V2 = MetadataAsValue::get(Context, N2);
411 EXPECT_EQ(V, V2);
412
413 auto *V3 = MetadataAsValue::get(Context, N);
414 EXPECT_TRUE(V3->getType()->isMetadataTy());
415 EXPECT_NE(V, V3);
416 EXPECT_EQ(N, V3->getMetadata());
417}
418
419TEST_F(MetadataAsValueTest, MDNodeConstant) {
420 auto *C = ConstantInt::getTrue(Context);
421 auto *MD = ConstantAsMetadata::get(C);
422 Metadata *Ops[] = {MD};
423 auto *N = MDNode::get(Context, Ops);
424
425 auto *V = MetadataAsValue::get(Context, MD);
426 EXPECT_TRUE(V->getType()->isMetadataTy());
427 EXPECT_EQ(MD, V->getMetadata());
428
429 auto *V2 = MetadataAsValue::get(Context, N);
430 EXPECT_EQ(MD, V2->getMetadata());
431 EXPECT_EQ(V, V2);
432}
433
Duncan P. N. Exon Smith121eeff2014-12-12 19:24:33 +0000434typedef MetadataTest ValueAsMetadataTest;
435
436TEST_F(ValueAsMetadataTest, UpdatesOnRAUW) {
437 Type *Ty = Type::getInt1PtrTy(Context);
438 std::unique_ptr<GlobalVariable> GV0(
439 new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage));
440 auto *MD = ValueAsMetadata::get(GV0.get());
441 EXPECT_TRUE(MD->getValue() == GV0.get());
442 ASSERT_TRUE(GV0->use_empty());
443
444 std::unique_ptr<GlobalVariable> GV1(
445 new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage));
446 GV0->replaceAllUsesWith(GV1.get());
447 EXPECT_TRUE(MD->getValue() == GV1.get());
448}
449
450typedef MetadataTest TrackingMDRefTest;
451
452TEST_F(TrackingMDRefTest, UpdatesOnRAUW) {
453 Type *Ty = Type::getInt1PtrTy(Context);
454 std::unique_ptr<GlobalVariable> GV0(
455 new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage));
456 TypedTrackingMDRef<ValueAsMetadata> MD(ValueAsMetadata::get(GV0.get()));
457 EXPECT_TRUE(MD->getValue() == GV0.get());
458 ASSERT_TRUE(GV0->use_empty());
459
460 std::unique_ptr<GlobalVariable> GV1(
461 new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage));
462 GV0->replaceAllUsesWith(GV1.get());
463 EXPECT_TRUE(MD->getValue() == GV1.get());
464
465 // Reset it, so we don't inadvertently test deletion.
466 MD.reset();
467}
468
469TEST_F(TrackingMDRefTest, UpdatesOnDeletion) {
470 Type *Ty = Type::getInt1PtrTy(Context);
471 std::unique_ptr<GlobalVariable> GV(
472 new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage));
473 TypedTrackingMDRef<ValueAsMetadata> MD(ValueAsMetadata::get(GV.get()));
474 EXPECT_TRUE(MD->getValue() == GV.get());
475 ASSERT_TRUE(GV->use_empty());
476
477 GV.reset();
478 EXPECT_TRUE(!MD);
479}
480
Devang Patel0924b332009-07-30 00:03:41 +0000481TEST(NamedMDNodeTest, Search) {
Jeffrey Yasskinbd8a7592010-03-04 23:24:19 +0000482 LLVMContext Context;
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000483 ConstantAsMetadata *C =
484 ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Context), 1));
485 ConstantAsMetadata *C2 =
486 ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Context), 2));
Devang Patel0924b332009-07-30 00:03:41 +0000487
Duncan P. N. Exon Smith5bf8fef2014-12-09 18:38:53 +0000488 Metadata *const V = C;
489 Metadata *const V2 = C2;
Jay Foad5514afe2011-04-21 19:59:31 +0000490 MDNode *n = MDNode::get(Context, V);
491 MDNode *n2 = MDNode::get(Context, V2);
Devang Patel0924b332009-07-30 00:03:41 +0000492
Jeffrey Yasskin3d73d1a2010-03-13 01:39:20 +0000493 Module M("MyModule", Context);
Devang Patel0924b332009-07-30 00:03:41 +0000494 const char *Name = "llvm.NMD1";
Dan Gohman2637cc12010-07-21 23:38:33 +0000495 NamedMDNode *NMD = M.getOrInsertNamedMetadata(Name);
496 NMD->addOperand(n);
497 NMD->addOperand(n2);
498
Chris Lattnerbe354a62009-08-23 04:47:35 +0000499 std::string Str;
500 raw_string_ostream oss(Str);
Devang Patel0924b332009-07-30 00:03:41 +0000501 NMD->print(oss);
Chris Lattner1e6e3672009-12-31 02:12:13 +0000502 EXPECT_STREQ("!llvm.NMD1 = !{!0, !1}\n",
Devang Patel0924b332009-07-30 00:03:41 +0000503 oss.str().c_str());
504}
Nick Lewycky49f89192009-04-04 07:22:01 +0000505}