blob: d5acef2ec744e7b01ec3d5621e307d3a7ae51a8d [file] [log] [blame]
Raphael Isemannb1dfad02018-08-01 06:04:48 +00001//===-- StreamTest.cpp ------------------------------------------*- C++ -*-===//
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#include "lldb/Utility/StreamString.h"
11#include "gtest/gtest.h"
12
13using namespace lldb_private;
14
15namespace {
16struct StreamTest : ::testing::Test {
17 // Note: Stream is an abstract class, so we use StreamString to test it. To
18 // make it easier to change this later, only methods in this class explicitly
19 // refer to the StringStream class.
20 StreamString s;
21 // We return here a std::string because that way gtest can print better
22 // assertion messages.
23 std::string TakeValue() {
24 std::string result = s.GetString().str();
25 s.Clear();
26 return result;
27 }
28};
29}
30
31namespace {
32// A StreamTest where we expect the Stream output to be binary.
33struct BinaryStreamTest : StreamTest {
34 void SetUp() override {
35 s.GetFlags().Set(Stream::eBinary);
36 }
37};
38}
39
40TEST_F(StreamTest, ChangingByteOrder) {
41 s.SetByteOrder(lldb::eByteOrderPDP);
42 EXPECT_EQ(lldb::eByteOrderPDP, s.GetByteOrder());
43}
44
45TEST_F(StreamTest, PutChar) {
46 s.PutChar('a');
47 EXPECT_EQ("a", TakeValue());
48
49 s.PutChar('1');
50 EXPECT_EQ("1", TakeValue());
51}
52
53TEST_F(StreamTest, PutCharWhitespace) {
54 s.PutChar(' ');
55 EXPECT_EQ(" ", TakeValue());
56
57 s.PutChar('\n');
58 EXPECT_EQ("\n", TakeValue());
59
60 s.PutChar('\r');
61 EXPECT_EQ("\r", TakeValue());
62
63 s.PutChar('\t');
64 EXPECT_EQ("\t", TakeValue());
65}
66
67TEST_F(StreamTest, PutCString) {
68 s.PutCString("");
69 EXPECT_EQ("", TakeValue());
70
71 s.PutCString("foobar");
72 EXPECT_EQ("foobar", TakeValue());
73
74 s.PutCString(" ");
75 EXPECT_EQ(" ", TakeValue());
76}
77
78TEST_F(StreamTest, PutCStringWithStringRef) {
79 s.PutCString(llvm::StringRef(""));
80 EXPECT_EQ("", TakeValue());
81
82 s.PutCString(llvm::StringRef("foobar"));
83 EXPECT_EQ("foobar", TakeValue());
84
85 s.PutCString(llvm::StringRef(" "));
86 EXPECT_EQ(" ", TakeValue());
87}
88
89TEST_F(StreamTest, QuotedCString) {
90 s.QuotedCString("foo");
91 EXPECT_EQ(R"("foo")", TakeValue());
92
93 s.QuotedCString("ba r");
94 EXPECT_EQ(R"("ba r")", TakeValue());
95
96 s.QuotedCString(" ");
97 EXPECT_EQ(R"(" ")", TakeValue());
98}
99
100TEST_F(StreamTest, PutCharNull) {
101 s.PutChar('\0');
102 EXPECT_EQ(std::string("\0", 1), TakeValue());
103
104 s.PutChar('a');
105 EXPECT_EQ(std::string("a", 1), TakeValue());
106}
107
108TEST_F(StreamTest, PutCStringAsRawHex8) {
109 s.PutCStringAsRawHex8("");
110 // FIXME: Check that printing 00 on an empty string is the intended behavior.
111 // It seems kind of unexpected that we print the trailing 0 byte for empty
112 // strings, but not for non-empty strings.
113 EXPECT_EQ("00", TakeValue());
114
115 s.PutCStringAsRawHex8("foobar");
116 EXPECT_EQ("666f6f626172", TakeValue());
117
118 s.PutCStringAsRawHex8(" ");
119 EXPECT_EQ("20", TakeValue());
120}
121
122TEST_F(StreamTest, PutHex8) {
123 s.PutHex8((uint8_t)55);
124 EXPECT_EQ("37", TakeValue());
125 s.PutHex8(std::numeric_limits<uint8_t>::max());
126 EXPECT_EQ("ff", TakeValue());
127 s.PutHex8((uint8_t)0);
128 EXPECT_EQ("00", TakeValue());
129}
130
131TEST_F(StreamTest, PutNHex8) {
132 s.PutNHex8(0, (uint8_t)55);
133 EXPECT_EQ("", TakeValue());
134 s.PutNHex8(1, (uint8_t)55);
135 EXPECT_EQ("37", TakeValue());
136 s.PutNHex8(2, (uint8_t)55);
137 EXPECT_EQ("3737", TakeValue());
138 s.PutNHex8(1, (uint8_t)56);
139 EXPECT_EQ("38", TakeValue());
140}
141
142TEST_F(StreamTest, PutHex16ByteOrderLittle) {
143 s.PutHex16(0x1234U, lldb::eByteOrderLittle);
144 EXPECT_EQ("3412", TakeValue());
145 s.PutHex16(std::numeric_limits<uint16_t>::max(), lldb::eByteOrderLittle);
146 EXPECT_EQ("ffff", TakeValue());
147 s.PutHex16(0U, lldb::eByteOrderLittle);
148 EXPECT_EQ("0000", TakeValue());
149}
150
151TEST_F(StreamTest, PutHex16ByteOrderBig) {
152 s.PutHex16(0x1234U, lldb::eByteOrderBig);
153 EXPECT_EQ("1234", TakeValue());
154 s.PutHex16(std::numeric_limits<uint16_t>::max(), lldb::eByteOrderBig);
155 EXPECT_EQ("ffff", TakeValue());
156 s.PutHex16(0U, lldb::eByteOrderBig);
157 EXPECT_EQ("0000", TakeValue());
158}
159
160TEST_F(StreamTest, PutHex32ByteOrderLittle) {
161 s.PutHex32(0x12345678U, lldb::eByteOrderLittle);
162 EXPECT_EQ("78563412", TakeValue());
163 s.PutHex32(std::numeric_limits<uint32_t>::max(), lldb::eByteOrderLittle);
164 EXPECT_EQ("ffffffff", TakeValue());
165 s.PutHex32(0U, lldb::eByteOrderLittle);
166 EXPECT_EQ("00000000", TakeValue());
167}
168
169TEST_F(StreamTest, PutHex32ByteOrderBig) {
170 s.PutHex32(0x12345678U, lldb::eByteOrderBig);
171 EXPECT_EQ("12345678", TakeValue());
172 s.PutHex32(std::numeric_limits<uint32_t>::max(), lldb::eByteOrderBig);
173 EXPECT_EQ("ffffffff", TakeValue());
174 s.PutHex32(0U, lldb::eByteOrderBig);
175 EXPECT_EQ("00000000", TakeValue());
176}
177
178TEST_F(StreamTest, PutHex64ByteOrderLittle) {
179 s.PutHex64(0x1234567890ABCDEFU, lldb::eByteOrderLittle);
180 EXPECT_EQ("efcdab9078563412", TakeValue());
181 s.PutHex64(std::numeric_limits<uint64_t>::max(), lldb::eByteOrderLittle);
182 EXPECT_EQ("ffffffffffffffff", TakeValue());
183 s.PutHex64(0U, lldb::eByteOrderLittle);
184 EXPECT_EQ("0000000000000000", TakeValue());
185}
186
187TEST_F(StreamTest, PutHex64ByteOrderBig) {
188 s.PutHex64(0x1234567890ABCDEFU, lldb::eByteOrderBig);
189 EXPECT_EQ("1234567890abcdef", TakeValue());
190 s.PutHex64(std::numeric_limits<uint64_t>::max(), lldb::eByteOrderBig);
191 EXPECT_EQ("ffffffffffffffff", TakeValue());
192 s.PutHex64(0U, lldb::eByteOrderBig);
193 EXPECT_EQ("0000000000000000", TakeValue());
194}
195
196//------------------------------------------------------------------------------
197// Shift operator tests.
198//------------------------------------------------------------------------------
199
200TEST_F(StreamTest, ShiftOperatorChars) {
201 s << 'a' << 'b';
202 EXPECT_EQ("ab", TakeValue());
203}
204
205TEST_F(StreamTest, ShiftOperatorStrings) {
206 s << "cstring\n";
207 s << llvm::StringRef("llvm::StringRef\n");
208 EXPECT_EQ("cstring\nllvm::StringRef\n", TakeValue());
209}
210
211TEST_F(StreamTest, ShiftOperatorInts) {
212 s << std::numeric_limits<int8_t>::max() << " ";
213 s << std::numeric_limits<int16_t>::max() << " ";
214 s << std::numeric_limits<int32_t>::max() << " ";
215 s << std::numeric_limits<int64_t>::max();
216 EXPECT_EQ("127 32767 2147483647 9223372036854775807", TakeValue());
217}
218
219TEST_F(StreamTest, ShiftOperatorUInts) {
220 s << std::numeric_limits<uint8_t>::max() << " ";
221 s << std::numeric_limits<uint16_t>::max() << " ";
222 s << std::numeric_limits<uint32_t>::max() << " ";
223 s << std::numeric_limits<uint64_t>::max();
224 EXPECT_EQ("ff ffff ffffffff ffffffffffffffff", TakeValue());
225}
226
227TEST_F(StreamTest, ShiftOperatorPtr) {
228 // This test is a bit tricky because pretty much everything related to
229 // pointer printing seems to lead to UB or IB. So let's make the most basic
230 // test that just checks that we print *something*. This way we at least know
231 // that pointer printing doesn't do really bad things (e.g. crashing, reading
232 // OOB/uninitialized memory which the sanitizers would spot).
233
234 // Shift our own pointer to the output.
235 int i = 3;
236 int *ptr = &i;
237 s << ptr;
238
239 EXPECT_TRUE(!TakeValue().empty());
240}
241
242TEST_F(StreamTest, PutPtr) {
243 // See the ShiftOperatorPtr test for the rationale.
244 int i = 3;
245 int *ptr = &i;
246 s.PutPointer(ptr);
247
248 EXPECT_TRUE(!TakeValue().empty());
249}
250
251// Alias to make it more clear that 'invalid' means for the Stream interface
252// that it should use the host byte order.
253const static auto hostByteOrder = lldb::eByteOrderInvalid;
254
255//------------------------------------------------------------------------------
256// PutRawBytes/PutBytesAsRawHex tests.
257//------------------------------------------------------------------------------
258
259TEST_F(StreamTest, PutBytesAsRawHex8ToBigEndian) {
260 uint32_t value = 0x12345678;
261 s.PutBytesAsRawHex8(static_cast<void*>(&value), sizeof(value),
262 hostByteOrder, lldb::eByteOrderBig);
263 EXPECT_EQ("78563412", TakeValue());
264}
265
266TEST_F(StreamTest, PutRawBytesToBigEndian) {
267 uint32_t value = 0x12345678;
268 s.PutRawBytes(static_cast<void*>(&value), sizeof(value),
269 hostByteOrder, lldb::eByteOrderBig);
270 EXPECT_EQ("\x78\x56\x34\x12", TakeValue());
271}
272
273TEST_F(StreamTest, PutBytesAsRawHex8ToLittleEndian) {
274 uint32_t value = 0x12345678;
275 s.PutBytesAsRawHex8(static_cast<void*>(&value), sizeof(value),
276 hostByteOrder, lldb::eByteOrderLittle);
277 EXPECT_EQ("12345678", TakeValue());
278}
279
280TEST_F(StreamTest, PutRawBytesToLittleEndian) {
281 uint32_t value = 0x12345678;
282 s.PutRawBytes(static_cast<void*>(&value), sizeof(value),
283 hostByteOrder, lldb::eByteOrderLittle);
284 EXPECT_EQ("\x12\x34\x56\x78", TakeValue());
285}
286
287TEST_F(StreamTest, PutBytesAsRawHex8ToMixedEndian) {
288 uint32_t value = 0x12345678;
289 s.PutBytesAsRawHex8(static_cast<void*>(&value), sizeof(value),
290 hostByteOrder, lldb::eByteOrderPDP);
291
292 // FIXME: PDP byte order is not actually implemented but Stream just silently
293 // prints the value in some random byte order...
294#if 0
295 EXPECT_EQ("34127856", TakeValue());
296#endif
297}
298
299TEST_F(StreamTest, PutRawBytesToMixedEndian) {
300 uint32_t value = 0x12345678;
301 s.PutRawBytes(static_cast<void*>(&value), sizeof(value),
302 lldb::eByteOrderInvalid, lldb::eByteOrderPDP);
303
304 // FIXME: PDP byte order is not actually implemented but Stream just silently
305 // prints the value in some random byte order...
306#if 0
307 EXPECT_EQ("\x34\x12\x78\x56", TakeValue());
308#endif
309}
310
311//------------------------------------------------------------------------------
312// ULEB128 support for binary streams.
313//------------------------------------------------------------------------------
314
315TEST_F(BinaryStreamTest, PutULEB128OneByte) {
316 auto bytes = s.PutULEB128(0x74ULL);
317 EXPECT_EQ("\x74", TakeValue());
318 EXPECT_EQ(1U, bytes);
319}
320
321TEST_F(BinaryStreamTest, PutULEB128TwoBytes) {
322 auto bytes = s.PutULEB128(0x1985ULL);
323 EXPECT_EQ("\x85\x33", TakeValue());
324 EXPECT_EQ(2U, bytes);
325}
326
327TEST_F(BinaryStreamTest, PutULEB128ThreeBytes) {
328 auto bytes = s.PutULEB128(0x5023ULL);
329 EXPECT_EQ("\xA3\xA0\x1", TakeValue());
330 EXPECT_EQ(3U, bytes);
331}
332
333TEST_F(BinaryStreamTest, PutULEB128FourBytes) {
334 auto bytes = s.PutULEB128(0xA48032ULL);
335 EXPECT_EQ("\xB2\x80\x92\x5", TakeValue());
336 EXPECT_EQ(4U, bytes);
337}
338
339TEST_F(BinaryStreamTest, PutULEB128FiveBytes) {
340 auto bytes = s.PutULEB128(0x12345678ULL);
341 EXPECT_EQ("\xF8\xAC\xD1\x91\x1", TakeValue());
342 EXPECT_EQ(5U, bytes);
343}
344
345TEST_F(BinaryStreamTest, PutULEB128SixBytes) {
346 auto bytes = s.PutULEB128(0xABFE3FAFDFULL);
347 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\x15", TakeValue());
348 EXPECT_EQ(6U, bytes);
349}
350
351TEST_F(BinaryStreamTest, PutULEB128SevenBytes) {
352 auto bytes = s.PutULEB128(0xDABFE3FAFDFULL);
353 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\x3", TakeValue());
354 EXPECT_EQ(7U, bytes);
355}
356
357TEST_F(BinaryStreamTest, PutULEB128EightBytes) {
358 auto bytes = s.PutULEB128(0x7CDABFE3FAFDFULL);
359 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\xF3\x3", TakeValue());
360 EXPECT_EQ(8U, bytes);
361}
362
363TEST_F(BinaryStreamTest, PutULEB128NineBytes) {
364 auto bytes = s.PutULEB128(0x327CDABFE3FAFDFULL);
365 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\xF3\x93\x3", TakeValue());
366 EXPECT_EQ(9U, bytes);
367}
368
369TEST_F(BinaryStreamTest, PutULEB128MaxValue) {
370 auto bytes = s.PutULEB128(std::numeric_limits<uint64_t>::max());
371 EXPECT_EQ("\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\x1", TakeValue());
372 EXPECT_EQ(10U, bytes);
373}
374
375TEST_F(BinaryStreamTest, PutULEB128Zero) {
376 auto bytes = s.PutULEB128(0x0U);
377 EXPECT_EQ(std::string("\0", 1), TakeValue());
378 EXPECT_EQ(1U, bytes);
379}
380
381TEST_F(BinaryStreamTest, PutULEB128One) {
382 auto bytes = s.PutULEB128(0x1U);
383 EXPECT_EQ("\x1", TakeValue());
384 EXPECT_EQ(1U, bytes);
385}
386
387//------------------------------------------------------------------------------
388// SLEB128 support for binary streams.
389//------------------------------------------------------------------------------
390
391TEST_F(BinaryStreamTest, PutSLEB128OneByte) {
392 auto bytes = s.PutSLEB128(0x74LL);
393 EXPECT_EQ(std::string("\xF4\0", 2), TakeValue());
394 EXPECT_EQ(2U, bytes);
395}
396
397TEST_F(BinaryStreamTest, PutSLEB128TwoBytes) {
398 auto bytes = s.PutSLEB128(0x1985LL);
399 EXPECT_EQ("\x85\x33", TakeValue());
400 EXPECT_EQ(2U, bytes);
401}
402
403TEST_F(BinaryStreamTest, PutSLEB128ThreeBytes) {
404 auto bytes = s.PutSLEB128(0x5023LL);
405 EXPECT_EQ("\xA3\xA0\x1", TakeValue());
406 EXPECT_EQ(3U, bytes);
407}
408
409TEST_F(BinaryStreamTest, PutSLEB128FourBytes) {
410 auto bytes = s.PutSLEB128(0xA48032LL);
411 EXPECT_EQ("\xB2\x80\x92\x5", TakeValue());
412 EXPECT_EQ(4U, bytes);
413}
414
415TEST_F(BinaryStreamTest, PutSLEB128FiveBytes) {
416 auto bytes = s.PutSLEB128(0x12345678LL);
417 EXPECT_EQ("\xF8\xAC\xD1\x91\x1", TakeValue());
418 EXPECT_EQ(5U, bytes);
419}
420
421TEST_F(BinaryStreamTest, PutSLEB128SixBytes) {
422 auto bytes = s.PutSLEB128(0xABFE3FAFDFLL);
423 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\x15", TakeValue());
424 EXPECT_EQ(6U, bytes);
425}
426
427TEST_F(BinaryStreamTest, PutSLEB128SevenBytes) {
428 auto bytes = s.PutSLEB128(0xDABFE3FAFDFLL);
429 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\x3", TakeValue());
430 EXPECT_EQ(7U, bytes);
431}
432
433TEST_F(BinaryStreamTest, PutSLEB128EightBytes) {
434 auto bytes = s.PutSLEB128(0x7CDABFE3FAFDFLL);
435 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\xF3\x3", TakeValue());
436 EXPECT_EQ(8U, bytes);
437}
438
439TEST_F(BinaryStreamTest, PutSLEB128NineBytes) {
440 auto bytes = s.PutSLEB128(0x327CDABFE3FAFDFLL);
441 EXPECT_EQ("\xDF\xDF\xFE\xF1\xBF\xB5\xF3\x93\x3", TakeValue());
442 EXPECT_EQ(9U, bytes);
443}
444
445TEST_F(BinaryStreamTest, PutSLEB128MaxValue) {
446 auto bytes = s.PutSLEB128(std::numeric_limits<int64_t>::max());
447 EXPECT_EQ(std::string("\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\0", 10), TakeValue());
448 EXPECT_EQ(10U, bytes);
449}
450
451TEST_F(BinaryStreamTest, PutSLEB128Zero) {
452 auto bytes = s.PutSLEB128(0x0);
453 EXPECT_EQ(std::string("\0", 1), TakeValue());
454 EXPECT_EQ(1U, bytes);
455}
456
457TEST_F(BinaryStreamTest, PutSLEB128One) {
458 auto bytes = s.PutSLEB128(0x1);
459 EXPECT_EQ(std::string("\x1", 1), TakeValue());
460 EXPECT_EQ(1U, bytes);
461}
462
463//------------------------------------------------------------------------------
464// SLEB128/ULEB128 support for non-binary streams.
465//------------------------------------------------------------------------------
466
467// The logic for this is very simple, so it should be enough to test some basic
468// use cases.
469
470TEST_F(StreamTest, PutULEB128) {
471 auto bytes = s.PutULEB128(0x74ULL);
472 EXPECT_EQ("0x74", TakeValue());
473 EXPECT_EQ(4U, bytes);
474}
475
476TEST_F(StreamTest, PutSLEB128) {
477 auto bytes = s.PutSLEB128(0x1985LL);
478 EXPECT_EQ("0x6533", TakeValue());
479 EXPECT_EQ(6U, bytes);
480}