This adds various new functionality and cleanup surrounding the use of the Stream library. Major changes include: * Renaming of all classes for more consistency / meaningfulness * Addition of some new methods for reading multiple values at once. * Full suite of unit tests for reader / writer functionality. * Full set of doxygen comments for all classes. * Streams now store their own endianness. * Fixed some bugs in a few of the classes that were discovered by the unit tests. llvm-svn: 296215
82 lines
2.6 KiB
C++
82 lines
2.6 KiB
C++
//===- BinaryStreamWriter.cpp - Writes objects to a BinaryStream ----------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/DebugInfo/MSF/BinaryStreamWriter.h"
|
|
|
|
#include "llvm/DebugInfo/MSF/BinaryStreamReader.h"
|
|
#include "llvm/DebugInfo/MSF/BinaryStreamRef.h"
|
|
|
|
using namespace llvm;
|
|
|
|
BinaryStreamWriter::BinaryStreamWriter(WritableBinaryStreamRef S)
|
|
: Stream(S), Offset(0) {}
|
|
|
|
Error BinaryStreamWriter::writeBytes(ArrayRef<uint8_t> Buffer) {
|
|
if (auto EC = Stream.writeBytes(Offset, Buffer))
|
|
return EC;
|
|
Offset += Buffer.size();
|
|
return Error::success();
|
|
}
|
|
|
|
Error BinaryStreamWriter::writeInteger(uint64_t Value, uint32_t ByteSize) {
|
|
assert(ByteSize == 1 || ByteSize == 2 || ByteSize == 4 || ByteSize == 8);
|
|
uint8_t Bytes[8];
|
|
MutableArrayRef<uint8_t> Buffer(Bytes);
|
|
Buffer = Buffer.take_front(ByteSize);
|
|
switch (ByteSize) {
|
|
case 1:
|
|
Buffer[0] = static_cast<uint8_t>(Value);
|
|
break;
|
|
case 2:
|
|
llvm::support::endian::write16(Buffer.data(), Value, Stream.getEndian());
|
|
break;
|
|
case 4:
|
|
llvm::support::endian::write32(Buffer.data(), Value, Stream.getEndian());
|
|
break;
|
|
case 8:
|
|
llvm::support::endian::write64(Buffer.data(), Value, Stream.getEndian());
|
|
break;
|
|
}
|
|
return writeBytes(Buffer);
|
|
}
|
|
|
|
Error BinaryStreamWriter::writeCString(StringRef Str) {
|
|
if (auto EC = writeFixedString(Str))
|
|
return EC;
|
|
if (auto EC = writeObject('\0'))
|
|
return EC;
|
|
|
|
return Error::success();
|
|
}
|
|
|
|
Error BinaryStreamWriter::writeFixedString(StringRef Str) {
|
|
return writeBytes(ArrayRef<uint8_t>(Str.bytes_begin(), Str.bytes_end()));
|
|
}
|
|
|
|
Error BinaryStreamWriter::writeStreamRef(BinaryStreamRef Ref) {
|
|
return writeStreamRef(Ref, Ref.getLength());
|
|
}
|
|
|
|
Error BinaryStreamWriter::writeStreamRef(BinaryStreamRef Ref, uint32_t Length) {
|
|
BinaryStreamReader SrcReader(Ref.slice(0, Length));
|
|
// This is a bit tricky. If we just call readBytes, we are requiring that it
|
|
// return us the entire stream as a contiguous buffer. There is no guarantee
|
|
// this can be satisfied by returning a reference straight from the buffer, as
|
|
// an implementation may not store all data in a single contiguous buffer. So
|
|
// we iterate over each contiguous chunk, writing each one in succession.
|
|
while (SrcReader.bytesRemaining() > 0) {
|
|
ArrayRef<uint8_t> Chunk;
|
|
if (auto EC = SrcReader.readLongestContiguousChunk(Chunk))
|
|
return EC;
|
|
if (auto EC = writeBytes(Chunk))
|
|
return EC;
|
|
}
|
|
return Error::success();
|
|
}
|