blob: bfb658cfa0b7442aed03ccb6f5d26798fd756028 [file] [log] [blame]
Zachary Turner120faca2017-02-27 22:11:43 +00001//===- BinaryStreamReader.cpp - Reads objects from a binary stream --------===//
Zachary Turner6ba65de2016-04-29 17:22:58 +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
Zachary Turnerd9dc2822017-03-02 20:52:51 +000010#include "llvm/Support/BinaryStreamReader.h"
Zachary Turnerd5d37dc2016-05-25 20:37:03 +000011
Zachary Turnerd9dc2822017-03-02 20:52:51 +000012#include "llvm/Support/BinaryStreamError.h"
13#include "llvm/Support/BinaryStreamRef.h"
Zachary Turner6ba65de2016-04-29 17:22:58 +000014
15using namespace llvm;
Zachary Turnerd9a62632017-05-17 20:23:31 +000016using endianness = llvm::support::endianness;
Zachary Turner6ba65de2016-04-29 17:22:58 +000017
Zachary Turnerd9a62632017-05-17 20:23:31 +000018BinaryStreamReader::BinaryStreamReader(BinaryStreamRef Ref) : Stream(Ref) {}
19
20BinaryStreamReader::BinaryStreamReader(BinaryStream &Stream) : Stream(Stream) {}
21
22BinaryStreamReader::BinaryStreamReader(ArrayRef<uint8_t> Data,
23 endianness Endian)
24 : Stream(Data, Endian) {}
25
26BinaryStreamReader::BinaryStreamReader(StringRef Data, endianness Endian)
27 : Stream(Data, Endian) {}
Zachary Turner6ba65de2016-04-29 17:22:58 +000028
Zachary Turner120faca2017-02-27 22:11:43 +000029Error BinaryStreamReader::readLongestContiguousChunk(
30 ArrayRef<uint8_t> &Buffer) {
Zachary Turner5acb4ac2016-06-10 05:09:12 +000031 if (auto EC = Stream.readLongestContiguousChunk(Offset, Buffer))
32 return EC;
33 Offset += Buffer.size();
34 return Error::success();
35}
36
Zachary Turner120faca2017-02-27 22:11:43 +000037Error BinaryStreamReader::readBytes(ArrayRef<uint8_t> &Buffer, uint32_t Size) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000038 if (auto EC = Stream.readBytes(Offset, Size, Buffer))
39 return EC;
40 Offset += Size;
41 return Error::success();
42}
43
Zachary Turner120faca2017-02-27 22:11:43 +000044Error BinaryStreamReader::readCString(StringRef &Dest) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000045 uint32_t OriginalOffset = getOffset();
Zachary Turner95c625e2017-05-25 21:12:27 +000046 uint32_t FoundOffset = 0;
Zachary Turner120faca2017-02-27 22:11:43 +000047 while (true) {
Zachary Turner95c625e2017-05-25 21:12:27 +000048 uint32_t ThisOffset = getOffset();
49 ArrayRef<uint8_t> Buffer;
50 if (auto EC = readLongestContiguousChunk(Buffer))
Zachary Turner819e77d2016-05-06 20:51:57 +000051 return EC;
Zachary Turner95c625e2017-05-25 21:12:27 +000052 StringRef S(reinterpret_cast<const char *>(Buffer.begin()), Buffer.size());
53 size_t Pos = S.find_first_of('\0');
54 if (LLVM_LIKELY(Pos != StringRef::npos)) {
55 FoundOffset = Pos + ThisOffset;
Zachary Turner120faca2017-02-27 22:11:43 +000056 break;
Zachary Turner95c625e2017-05-25 21:12:27 +000057 }
Zachary Turner120faca2017-02-27 22:11:43 +000058 }
Zachary Turner95c625e2017-05-25 21:12:27 +000059 assert(FoundOffset >= OriginalOffset);
60
Zachary Turner8dbe3622016-05-27 01:54:44 +000061 setOffset(OriginalOffset);
Zachary Turner95c625e2017-05-25 21:12:27 +000062 size_t Length = FoundOffset - OriginalOffset;
Zachary Turner8dbe3622016-05-27 01:54:44 +000063
Zachary Turner120faca2017-02-27 22:11:43 +000064 if (auto EC = readFixedString(Dest, Length))
Zachary Turner8dbe3622016-05-27 01:54:44 +000065 return EC;
Zachary Turner8dbe3622016-05-27 01:54:44 +000066
Zachary Turner95c625e2017-05-25 21:12:27 +000067 // Now set the offset back to after the null terminator.
68 setOffset(FoundOffset + 1);
Zachary Turner819e77d2016-05-06 20:51:57 +000069 return Error::success();
Zachary Turner6ba65de2016-04-29 17:22:58 +000070}
Zachary Turnerf5c59652016-05-03 00:28:21 +000071
Eric Beckmann72fb6a82017-05-30 18:19:06 +000072Error BinaryStreamReader::readWideString(ArrayRef<UTF16> &Dest) {
73 uint32_t Length = 0;
74 uint32_t OriginalOffset = getOffset();
75 const UTF16 *C;
76 while (true) {
77 if (auto EC = readObject(C))
78 return EC;
79 if (*C == 0x0000)
80 break;
81 ++Length;
82 }
83 uint32_t NewOffset = getOffset();
84 setOffset(OriginalOffset);
85
86 if (auto EC = readArray(Dest, Length))
87 return EC;
88 setOffset(NewOffset);
89 return Error::success();
90}
91
Zachary Turner120faca2017-02-27 22:11:43 +000092Error BinaryStreamReader::readFixedString(StringRef &Dest, uint32_t Length) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000093 ArrayRef<uint8_t> Bytes;
Zachary Turner0d43c1c2016-05-28 05:21:57 +000094 if (auto EC = readBytes(Bytes, Length))
Zachary Turnerf5c59652016-05-03 00:28:21 +000095 return EC;
Zachary Turner8dbe3622016-05-27 01:54:44 +000096 Dest = StringRef(reinterpret_cast<const char *>(Bytes.begin()), Bytes.size());
97 return Error::success();
98}
99
Zachary Turner120faca2017-02-27 22:11:43 +0000100Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref) {
Zachary Turner8dbe3622016-05-27 01:54:44 +0000101 return readStreamRef(Ref, bytesRemaining());
102}
103
Zachary Turner120faca2017-02-27 22:11:43 +0000104Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref, uint32_t Length) {
Zachary Turner8dbe3622016-05-27 01:54:44 +0000105 if (bytesRemaining() < Length)
Zachary Turnerd0b44fa2017-02-28 17:49:34 +0000106 return make_error<BinaryStreamError>(stream_error_code::stream_too_short);
Zachary Turnerf4e9c9a2016-06-02 19:51:48 +0000107 Ref = Stream.slice(Offset, Length);
Zachary Turnerf5c59652016-05-03 00:28:21 +0000108 Offset += Length;
Zachary Turner819e77d2016-05-06 20:51:57 +0000109 return Error::success();
Zachary Turnerf5c59652016-05-03 00:28:21 +0000110}
Zachary Turner4d49eb92016-10-20 18:31:19 +0000111
Zachary Turner120faca2017-02-27 22:11:43 +0000112Error BinaryStreamReader::skip(uint32_t Amount) {
Zachary Turner4d49eb92016-10-20 18:31:19 +0000113 if (Amount > bytesRemaining())
Zachary Turnerd0b44fa2017-02-28 17:49:34 +0000114 return make_error<BinaryStreamError>(stream_error_code::stream_too_short);
Zachary Turner4d49eb92016-10-20 18:31:19 +0000115 Offset += Amount;
116 return Error::success();
117}
118
Zachary Turnerd2b418b2017-05-17 20:42:52 +0000119Error BinaryStreamReader::padToAlignment(uint32_t Align) {
120 uint32_t NewOffset = alignTo(Offset, Align);
121 return skip(NewOffset - Offset);
122}
123
Zachary Turner120faca2017-02-27 22:11:43 +0000124uint8_t BinaryStreamReader::peek() const {
Zachary Turner4d49eb92016-10-20 18:31:19 +0000125 ArrayRef<uint8_t> Buffer;
126 auto EC = Stream.readBytes(Offset, 1, Buffer);
127 assert(!EC && "Cannot peek an empty buffer!");
128 llvm::consumeError(std::move(EC));
129 return Buffer[0];
130}
Zachary Turnerc504ae32017-05-03 15:58:37 +0000131
132std::pair<BinaryStreamReader, BinaryStreamReader>
133BinaryStreamReader::split(uint32_t Off) const {
134 assert(getLength() >= Off);
135
136 BinaryStreamRef First = Stream.drop_front(Offset);
137
138 BinaryStreamRef Second = First.drop_front(Off);
139 First = First.keep_front(Off);
140 BinaryStreamReader W1{First};
141 BinaryStreamReader W2{Second};
142 return std::make_pair(W1, W2);
143}