Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 1 | //===- BinaryStreamReader.cpp - Reads objects from a binary stream --------===// |
Zachary Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 2 | // |
Chandler Carruth | 2946cd7 | 2019-01-19 08:50:56 +0000 | [diff] [blame^] | 3 | // 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 Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
Zachary Turner | d9dc282 | 2017-03-02 20:52:51 +0000 | [diff] [blame] | 9 | #include "llvm/Support/BinaryStreamReader.h" |
Zachary Turner | d5d37dc | 2016-05-25 20:37:03 +0000 | [diff] [blame] | 10 | |
Zachary Turner | d9dc282 | 2017-03-02 20:52:51 +0000 | [diff] [blame] | 11 | #include "llvm/Support/BinaryStreamError.h" |
| 12 | #include "llvm/Support/BinaryStreamRef.h" |
Zachary Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 13 | |
| 14 | using namespace llvm; |
Zachary Turner | d9a6263 | 2017-05-17 20:23:31 +0000 | [diff] [blame] | 15 | using endianness = llvm::support::endianness; |
Zachary Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 16 | |
Zachary Turner | d9a6263 | 2017-05-17 20:23:31 +0000 | [diff] [blame] | 17 | BinaryStreamReader::BinaryStreamReader(BinaryStreamRef Ref) : Stream(Ref) {} |
| 18 | |
| 19 | BinaryStreamReader::BinaryStreamReader(BinaryStream &Stream) : Stream(Stream) {} |
| 20 | |
| 21 | BinaryStreamReader::BinaryStreamReader(ArrayRef<uint8_t> Data, |
| 22 | endianness Endian) |
| 23 | : Stream(Data, Endian) {} |
| 24 | |
| 25 | BinaryStreamReader::BinaryStreamReader(StringRef Data, endianness Endian) |
| 26 | : Stream(Data, Endian) {} |
Zachary Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 27 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 28 | Error BinaryStreamReader::readLongestContiguousChunk( |
| 29 | ArrayRef<uint8_t> &Buffer) { |
Zachary Turner | 5acb4ac | 2016-06-10 05:09:12 +0000 | [diff] [blame] | 30 | if (auto EC = Stream.readLongestContiguousChunk(Offset, Buffer)) |
| 31 | return EC; |
| 32 | Offset += Buffer.size(); |
| 33 | return Error::success(); |
| 34 | } |
| 35 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 36 | Error BinaryStreamReader::readBytes(ArrayRef<uint8_t> &Buffer, uint32_t Size) { |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 37 | if (auto EC = Stream.readBytes(Offset, Size, Buffer)) |
| 38 | return EC; |
| 39 | Offset += Size; |
| 40 | return Error::success(); |
| 41 | } |
| 42 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 43 | Error BinaryStreamReader::readCString(StringRef &Dest) { |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 44 | uint32_t OriginalOffset = getOffset(); |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 45 | uint32_t FoundOffset = 0; |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 46 | while (true) { |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 47 | uint32_t ThisOffset = getOffset(); |
| 48 | ArrayRef<uint8_t> Buffer; |
| 49 | if (auto EC = readLongestContiguousChunk(Buffer)) |
Zachary Turner | 819e77d | 2016-05-06 20:51:57 +0000 | [diff] [blame] | 50 | return EC; |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 51 | 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 Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 55 | break; |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 56 | } |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 57 | } |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 58 | assert(FoundOffset >= OriginalOffset); |
| 59 | |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 60 | setOffset(OriginalOffset); |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 61 | size_t Length = FoundOffset - OriginalOffset; |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 62 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 63 | if (auto EC = readFixedString(Dest, Length)) |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 64 | return EC; |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 65 | |
Zachary Turner | 95c625e | 2017-05-25 21:12:27 +0000 | [diff] [blame] | 66 | // Now set the offset back to after the null terminator. |
| 67 | setOffset(FoundOffset + 1); |
Zachary Turner | 819e77d | 2016-05-06 20:51:57 +0000 | [diff] [blame] | 68 | return Error::success(); |
Zachary Turner | 6ba65de | 2016-04-29 17:22:58 +0000 | [diff] [blame] | 69 | } |
Zachary Turner | f5c5965 | 2016-05-03 00:28:21 +0000 | [diff] [blame] | 70 | |
Eric Beckmann | 72fb6a8 | 2017-05-30 18:19:06 +0000 | [diff] [blame] | 71 | Error 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 Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 91 | Error BinaryStreamReader::readFixedString(StringRef &Dest, uint32_t Length) { |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 92 | ArrayRef<uint8_t> Bytes; |
Zachary Turner | 0d43c1c | 2016-05-28 05:21:57 +0000 | [diff] [blame] | 93 | if (auto EC = readBytes(Bytes, Length)) |
Zachary Turner | f5c5965 | 2016-05-03 00:28:21 +0000 | [diff] [blame] | 94 | return EC; |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 95 | Dest = StringRef(reinterpret_cast<const char *>(Bytes.begin()), Bytes.size()); |
| 96 | return Error::success(); |
| 97 | } |
| 98 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 99 | Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref) { |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 100 | return readStreamRef(Ref, bytesRemaining()); |
| 101 | } |
| 102 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 103 | Error BinaryStreamReader::readStreamRef(BinaryStreamRef &Ref, uint32_t Length) { |
Zachary Turner | 8dbe362 | 2016-05-27 01:54:44 +0000 | [diff] [blame] | 104 | if (bytesRemaining() < Length) |
Zachary Turner | d0b44fa | 2017-02-28 17:49:34 +0000 | [diff] [blame] | 105 | return make_error<BinaryStreamError>(stream_error_code::stream_too_short); |
Zachary Turner | f4e9c9a | 2016-06-02 19:51:48 +0000 | [diff] [blame] | 106 | Ref = Stream.slice(Offset, Length); |
Zachary Turner | f5c5965 | 2016-05-03 00:28:21 +0000 | [diff] [blame] | 107 | Offset += Length; |
Zachary Turner | 819e77d | 2016-05-06 20:51:57 +0000 | [diff] [blame] | 108 | return Error::success(); |
Zachary Turner | f5c5965 | 2016-05-03 00:28:21 +0000 | [diff] [blame] | 109 | } |
Zachary Turner | 4d49eb9 | 2016-10-20 18:31:19 +0000 | [diff] [blame] | 110 | |
Zachary Turner | a1dcb85 | 2017-06-23 16:38:40 +0000 | [diff] [blame] | 111 | Error BinaryStreamReader::readSubstream(BinarySubstreamRef &Stream, |
| 112 | uint32_t Size) { |
| 113 | Stream.Offset = getOffset(); |
| 114 | return readStreamRef(Stream.StreamData, Size); |
| 115 | } |
| 116 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 117 | Error BinaryStreamReader::skip(uint32_t Amount) { |
Zachary Turner | 4d49eb9 | 2016-10-20 18:31:19 +0000 | [diff] [blame] | 118 | if (Amount > bytesRemaining()) |
Zachary Turner | d0b44fa | 2017-02-28 17:49:34 +0000 | [diff] [blame] | 119 | return make_error<BinaryStreamError>(stream_error_code::stream_too_short); |
Zachary Turner | 4d49eb9 | 2016-10-20 18:31:19 +0000 | [diff] [blame] | 120 | Offset += Amount; |
| 121 | return Error::success(); |
| 122 | } |
| 123 | |
Zachary Turner | d2b418b | 2017-05-17 20:42:52 +0000 | [diff] [blame] | 124 | Error BinaryStreamReader::padToAlignment(uint32_t Align) { |
| 125 | uint32_t NewOffset = alignTo(Offset, Align); |
| 126 | return skip(NewOffset - Offset); |
| 127 | } |
| 128 | |
Zachary Turner | 120faca | 2017-02-27 22:11:43 +0000 | [diff] [blame] | 129 | uint8_t BinaryStreamReader::peek() const { |
Zachary Turner | 4d49eb9 | 2016-10-20 18:31:19 +0000 | [diff] [blame] | 130 | 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 Turner | c504ae3 | 2017-05-03 15:58:37 +0000 | [diff] [blame] | 136 | |
| 137 | std::pair<BinaryStreamReader, BinaryStreamReader> |
| 138 | BinaryStreamReader::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 | } |