blob: 6f691bf4dcdfa34f76f7fbf733d9b8077b8cc321 [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//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Zachary Turner6ba65de2016-04-29 17:22:58 +00006//
7//===----------------------------------------------------------------------===//
8
Zachary Turnerd9dc2822017-03-02 20:52:51 +00009#include "llvm/Support/BinaryStreamReader.h"
Zachary Turnerd5d37dc2016-05-25 20:37:03 +000010
Zachary Turnerd9dc2822017-03-02 20:52:51 +000011#include "llvm/Support/BinaryStreamError.h"
12#include "llvm/Support/BinaryStreamRef.h"
Zachary Turner6ba65de2016-04-29 17:22:58 +000013
14using namespace llvm;
Zachary Turnerd9a62632017-05-17 20:23:31 +000015using endianness = llvm::support::endianness;
Zachary Turner6ba65de2016-04-29 17:22:58 +000016
Zachary Turnerd9a62632017-05-17 20:23:31 +000017BinaryStreamReader::BinaryStreamReader(BinaryStreamRef Ref) : Stream(Ref) {}
18
19BinaryStreamReader::BinaryStreamReader(BinaryStream &Stream) : Stream(Stream) {}
20
21BinaryStreamReader::BinaryStreamReader(ArrayRef<uint8_t> Data,
22 endianness Endian)
23 : Stream(Data, Endian) {}
24
25BinaryStreamReader::BinaryStreamReader(StringRef Data, endianness Endian)
26 : Stream(Data, Endian) {}
Zachary Turner6ba65de2016-04-29 17:22:58 +000027
Zachary Turner120faca2017-02-27 22:11:43 +000028Error BinaryStreamReader::readLongestContiguousChunk(
29 ArrayRef<uint8_t> &Buffer) {
Zachary Turner5acb4ac2016-06-10 05:09:12 +000030 if (auto EC = Stream.readLongestContiguousChunk(Offset, Buffer))
31 return EC;
32 Offset += Buffer.size();
33 return Error::success();
34}
35
Zachary Turner120faca2017-02-27 22:11:43 +000036Error BinaryStreamReader::readBytes(ArrayRef<uint8_t> &Buffer, uint32_t Size) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000037 if (auto EC = Stream.readBytes(Offset, Size, Buffer))
38 return EC;
39 Offset += Size;
40 return Error::success();
41}
42
Zachary Turner120faca2017-02-27 22:11:43 +000043Error BinaryStreamReader::readCString(StringRef &Dest) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000044 uint32_t OriginalOffset = getOffset();
Zachary Turner95c625e2017-05-25 21:12:27 +000045 uint32_t FoundOffset = 0;
Zachary Turner120faca2017-02-27 22:11:43 +000046 while (true) {
Zachary Turner95c625e2017-05-25 21:12:27 +000047 uint32_t ThisOffset = getOffset();
48 ArrayRef<uint8_t> Buffer;
49 if (auto EC = readLongestContiguousChunk(Buffer))
Zachary Turner819e77d2016-05-06 20:51:57 +000050 return EC;
Zachary Turner95c625e2017-05-25 21:12:27 +000051 StringRef S(reinterpret_cast<const char *>(Buffer.begin()), Buffer.size());
52 size_t Pos = S.find_first_of('\0');
53 if (LLVM_LIKELY(Pos != StringRef::npos)) {
54 FoundOffset = Pos + ThisOffset;
Zachary Turner120faca2017-02-27 22:11:43 +000055 break;
Zachary Turner95c625e2017-05-25 21:12:27 +000056 }
Zachary Turner120faca2017-02-27 22:11:43 +000057 }
Zachary Turner95c625e2017-05-25 21:12:27 +000058 assert(FoundOffset >= OriginalOffset);
59
Zachary Turner8dbe3622016-05-27 01:54:44 +000060 setOffset(OriginalOffset);
Zachary Turner95c625e2017-05-25 21:12:27 +000061 size_t Length = FoundOffset - OriginalOffset;
Zachary Turner8dbe3622016-05-27 01:54:44 +000062
Zachary Turner120faca2017-02-27 22:11:43 +000063 if (auto EC = readFixedString(Dest, Length))
Zachary Turner8dbe3622016-05-27 01:54:44 +000064 return EC;
Zachary Turner8dbe3622016-05-27 01:54:44 +000065
Zachary Turner95c625e2017-05-25 21:12:27 +000066 // Now set the offset back to after the null terminator.
67 setOffset(FoundOffset + 1);
Zachary Turner819e77d2016-05-06 20:51:57 +000068 return Error::success();
Zachary Turner6ba65de2016-04-29 17:22:58 +000069}
Zachary Turnerf5c59652016-05-03 00:28:21 +000070
Eric Beckmann72fb6a82017-05-30 18:19:06 +000071Error BinaryStreamReader::readWideString(ArrayRef<UTF16> &Dest) {
72 uint32_t Length = 0;
73 uint32_t OriginalOffset = getOffset();
74 const UTF16 *C;
75 while (true) {
76 if (auto EC = readObject(C))
77 return EC;
78 if (*C == 0x0000)
79 break;
80 ++Length;
81 }
82 uint32_t NewOffset = getOffset();
83 setOffset(OriginalOffset);
84
85 if (auto EC = readArray(Dest, Length))
86 return EC;
87 setOffset(NewOffset);
88 return Error::success();
89}
90
Zachary Turner120faca2017-02-27 22:11:43 +000091Error BinaryStreamReader::readFixedString(StringRef &Dest, uint32_t Length) {
Zachary Turner8dbe3622016-05-27 01:54:44 +000092 ArrayRef<uint8_t> Bytes;
Zachary Turner0d43c1c2016-05-28 05:21:57 +000093 if (auto EC = readBytes(Bytes, Length))
Zachary Turnerf5c59652016-05-03 00:28:21 +000094 return EC;
Zachary Turner8dbe3622016-05-27 01:54:44 +000095 Dest = StringRef(reinterpret_cast<const char *>(Bytes.begin()), Bytes.size());
96 return Error::success();
97}
98
Zachary Turner120faca2017-02-27 22:11:43 +000099Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref) {
Zachary Turner8dbe3622016-05-27 01:54:44 +0000100 return readStreamRef(Ref, bytesRemaining());
101}
102
Zachary Turner120faca2017-02-27 22:11:43 +0000103Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref, uint32_t Length) {
Zachary Turner8dbe3622016-05-27 01:54:44 +0000104 if (bytesRemaining() < Length)
Zachary Turnerd0b44fa2017-02-28 17:49:34 +0000105 return make_error<BinaryStreamError>(stream_error_code::stream_too_short);
Zachary Turnerf4e9c9a2016-06-02 19:51:48 +0000106 Ref = Stream.slice(Offset, Length);
Zachary Turnerf5c59652016-05-03 00:28:21 +0000107 Offset += Length;
Zachary Turner819e77d2016-05-06 20:51:57 +0000108 return Error::success();
Zachary Turnerf5c59652016-05-03 00:28:21 +0000109}
Zachary Turner4d49eb92016-10-20 18:31:19 +0000110
Zachary Turnera1dcb852017-06-23 16:38:40 +0000111Error BinaryStreamReader::readSubstream(BinarySubstreamRef &Stream,
112 uint32_t Size) {
113 Stream.Offset = getOffset();
114 return readStreamRef(Stream.StreamData, Size);
115}
116
Zachary Turner120faca2017-02-27 22:11:43 +0000117Error BinaryStreamReader::skip(uint32_t Amount) {
Zachary Turner4d49eb92016-10-20 18:31:19 +0000118 if (Amount > bytesRemaining())
Zachary Turnerd0b44fa2017-02-28 17:49:34 +0000119 return make_error<BinaryStreamError>(stream_error_code::stream_too_short);
Zachary Turner4d49eb92016-10-20 18:31:19 +0000120 Offset += Amount;
121 return Error::success();
122}
123
Zachary Turnerd2b418b2017-05-17 20:42:52 +0000124Error BinaryStreamReader::padToAlignment(uint32_t Align) {
125 uint32_t NewOffset = alignTo(Offset, Align);
126 return skip(NewOffset - Offset);
127}
128
Zachary Turner120faca2017-02-27 22:11:43 +0000129uint8_t BinaryStreamReader::peek() const {
Zachary Turner4d49eb92016-10-20 18:31:19 +0000130 ArrayRef<uint8_t> Buffer;
131 auto EC = Stream.readBytes(Offset, 1, Buffer);
132 assert(!EC && "Cannot peek an empty buffer!");
133 llvm::consumeError(std::move(EC));
134 return Buffer[0];
135}
Zachary Turnerc504ae32017-05-03 15:58:37 +0000136
137std::pair<BinaryStreamReader, BinaryStreamReader>
138BinaryStreamReader::split(uint32_t Off) const {
139 assert(getLength() >= Off);
140
141 BinaryStreamRef First = Stream.drop_front(Offset);
142
143 BinaryStreamRef Second = First.drop_front(Off);
144 First = First.keep_front(Off);
145 BinaryStreamReader W1{First};
146 BinaryStreamReader W2{Second};
147 return std::make_pair(W1, W2);
148}