mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-19 20:34:38 +00:00
7969dc2bec
returning error codes. Because they don't return an error code, they can return the value read, which simplifies the code and makes the reader more efficient (yaay!). Also eliminate the special case code for little endian machines. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@10871 91177308-0d34-0410-b5e6-96231b3b80d8
98 lines
3.2 KiB
C++
98 lines
3.2 KiB
C++
//===-- ReaderPrimitives.h - Bytecode file format reading prims -*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file was developed by the LLVM research group and is distributed under
|
|
// the University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This header defines some basic functions for reading basic primitive types
|
|
// from a bytecode stream.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef READERPRIMITIVES_H
|
|
#define READERPRIMITIVES_H
|
|
|
|
#include "Support/DataTypes.h"
|
|
#include <string>
|
|
|
|
namespace llvm {
|
|
|
|
static inline unsigned read(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
if (Buf+4 > EndBuf) throw std::string("Ran out of data!");
|
|
Buf += 4;
|
|
return Buf[-4] | (Buf[-3] << 8) | (Buf[-2] << 16) | (Buf[-1] << 24);
|
|
}
|
|
|
|
|
|
// read_vbr - Read an unsigned integer encoded in variable bitrate format.
|
|
//
|
|
static inline unsigned read_vbr_uint(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
unsigned Shift = 0;
|
|
unsigned Result = 0;
|
|
|
|
do {
|
|
if (Buf == EndBuf) throw std::string("Ran out of data!");
|
|
Result |= (unsigned)((*Buf++) & 0x7F) << Shift;
|
|
Shift += 7;
|
|
} while (Buf[-1] & 0x80);
|
|
return Result;
|
|
}
|
|
|
|
static inline uint64_t read_vbr_uint64(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
unsigned Shift = 0;
|
|
uint64_t Result = 0;
|
|
|
|
do {
|
|
if (Buf == EndBuf) throw std::string("Ran out of data!");
|
|
Result |= (uint64_t)((*Buf++) & 0x7F) << Shift;
|
|
Shift += 7;
|
|
} while (Buf[-1] & 0x80);
|
|
return Result;
|
|
}
|
|
|
|
static inline int64_t read_vbr_int64(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
uint64_t R = read_vbr_uint64(Buf, EndBuf);
|
|
if (R & 1)
|
|
return -(int64_t)(R >> 1);
|
|
else
|
|
return (int64_t)(R >> 1);
|
|
}
|
|
|
|
// align32 - Round up to multiple of 32 bits...
|
|
static inline void align32(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
Buf = (const unsigned char *)((unsigned long)(Buf+3) & (~3UL));
|
|
if (Buf > EndBuf) throw std::string("Ran out of data!");
|
|
}
|
|
|
|
static inline std::string read_str(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf) {
|
|
unsigned Size = read_vbr_uint(Buf, EndBuf);
|
|
const unsigned char *OldBuf = Buf;
|
|
Buf += Size;
|
|
if (Buf > EndBuf) // Size invalid?
|
|
throw std::string("Ran out of data reading a string!");
|
|
return std::string((char*)OldBuf, Size);
|
|
}
|
|
|
|
static inline void input_data(const unsigned char *&Buf,
|
|
const unsigned char *EndBuf,
|
|
void *Ptr, void *End) {
|
|
unsigned char *Start = (unsigned char *)Ptr;
|
|
unsigned Amount = (unsigned char *)End - Start;
|
|
if (Buf+Amount > EndBuf) throw std::string("Ran out of data!");
|
|
std::copy(Buf, Buf+Amount, Start);
|
|
Buf += Amount;
|
|
}
|
|
|
|
} // End llvm namespace
|
|
|
|
#endif
|