/* * Copyright 2011, Ben Langmead * * This file is part of Bowtie 2. * * Bowtie 2 is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Bowtie 2 is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Bowtie 2. If not, see . */ #ifndef WORD_IO_H_ #define WORD_IO_H_ #include #include #include #include #include "assert_helpers.h" #include "endian_swap.h" /** * Write a 32-bit unsigned to an output stream being careful to * re-endianize if caller-requested endianness differs from current * host. */ static inline void writeU32(std::ostream& out, uint32_t x, bool toBigEndian) { uint32_t y = endianizeU32(x, toBigEndian); out.write((const char*)&y, 4); } /** * Write a 32-bit unsigned to an output stream using the native * endianness. */ static inline void writeU32(std::ostream& out, uint32_t x) { out.write((const char*)&x, 4); } /** * Write a 32-bit signed int to an output stream being careful to * re-endianize if caller-requested endianness differs from current * host. */ static inline void writeI32(std::ostream& out, int32_t x, bool toBigEndian) { int32_t y = endianizeI32(x, toBigEndian); out.write((const char*)&y, 4); } /** * Write a 32-bit unsigned to an output stream using the native * endianness. */ static inline void writeI32(std::ostream& out, int32_t x) { out.write((const char*)&x, 4); } /** * Write a 16-bit unsigned to an output stream being careful to * re-endianize if caller-requested endianness differs from current * host. */ static inline void writeU16(std::ostream& out, uint16_t x, bool toBigEndian) { uint16_t y = endianizeU16(x, toBigEndian); out.write((const char*)&y, 2); } /** * Write a 16-bit unsigned to an output stream using the native * endianness. */ static inline void writeU16(std::ostream& out, uint16_t x) { out.write((const char*)&x, 2); } /** * Write a 16-bit signed int to an output stream being careful to * re-endianize if caller-requested endianness differs from current * host. */ static inline void writeI16(std::ostream& out, int16_t x, bool toBigEndian) { int16_t y = endianizeI16(x, toBigEndian); out.write((const char*)&y, 2); } /** * Write a 16-bit unsigned to an output stream using the native * endianness. */ static inline void writeI16(std::ostream& out, int16_t x) { out.write((const char*)&x, 2); } /** * Write a 8-bit unsigned to an output stream. */ static inline void writeU8(std::ostream& out, uint8_t x) { out.write((const char*)&x, 1); } /** * Read a 32-bit unsigned from an input stream, inverting endianness * if necessary. */ static inline uint32_t readU32(std::istream& in, bool swap) { uint32_t x; in.read((char *)&x, 4); assert_eq(4, in.gcount()); if(swap) { return endianSwapU32(x); } else { return x; } } /** * Read a 32-bit unsigned from a file descriptor, optionally inverting * endianness. */ #ifdef BOWTIE_MM static inline uint32_t readU32(int in, bool swap) { uint32_t x; if(read(in, (void *)&x, 4) != 4) { assert(false); } if(swap) { return endianSwapU32(x); } else { return x; } } #endif /** * Read a 32-bit unsigned from a FILE*, optionally inverting * endianness. */ static inline uint32_t readU32(FILE* in, bool swap) { uint32_t x; if(fread((void *)&x, 1, 4, in) != 4) { assert(false); } if(swap) { return endianSwapU32(x); } else { return x; } } /** * Read a 32-bit signed from an input stream, inverting endianness * if necessary. */ static inline int32_t readI32(std::istream& in, bool swap) { int32_t x; in.read((char *)&x, 4); assert_eq(4, in.gcount()); if(swap) { return endianSwapI32(x); } else { return x; } } /** * Read a 32-bit unsigned from a file descriptor, optionally inverting * endianness. */ #ifdef BOWTIE_MM static inline uint32_t readI32(int in, bool swap) { int32_t x; if(read(in, (void *)&x, 4) != 4) { assert(false); } if(swap) { return endianSwapI32(x); } else { return x; } } #endif /** * Read a 32-bit unsigned from a FILE*, optionally inverting * endianness. */ static inline uint32_t readI32(FILE* in, bool swap) { int32_t x; if(fread((void *)&x, 1, 4, in) != 4) { assert(false); } if(swap) { return endianSwapI32(x); } else { return x; } } /** * Read a 16-bit unsigned from an input stream, inverting endianness * if necessary. */ static inline uint16_t readU16(std::istream& in, bool swap) { uint16_t x; in.read((char *)&x, 2); assert_eq(2, in.gcount()); if(swap) { return endianSwapU16(x); } else { return x; } } /** * Read a 16-bit unsigned from a file descriptor, optionally inverting * endianness. */ #ifdef BOWTIE_MM static inline uint16_t readU16(int in, bool swap) { uint16_t x; if(read(in, (void *)&x, 2) != 2) { assert(false); } if(swap) { return endianSwapU16(x); } else { return x; } } #endif /** * Read a 16-bit unsigned from a FILE*, optionally inverting * endianness. */ static inline uint16_t readU16(FILE* in, bool swap) { uint16_t x; if(fread((void *)&x, 1, 2, in) != 2) { assert(false); } if(swap) { return endianSwapU32(x); } else { return x; } } /** * Read a 16-bit signed from an input stream, inverting endianness * if necessary. */ static inline int32_t readI16(std::istream& in, bool swap) { int16_t x; in.read((char *)&x, 2); assert_eq(2, in.gcount()); if(swap) { return endianSwapI16(x); } else { return x; } } /** * Read a 16-bit unsigned from a file descriptor, optionally inverting * endianness. */ #ifdef BOWTIE_MM static inline uint16_t readI16(int in, bool swap) { int16_t x; if(read(in, (void *)&x, 2) != 2) { assert(false); } if(swap) { return endianSwapI16(x); } else { return x; } } #endif /** * Read a 16-bit unsigned from a FILE*, optionally inverting * endianness. */ static inline uint16_t readI16(FILE* in, bool swap) { int16_t x; if(fread((void *)&x, 1, 2, in) != 2) { assert(false); } if(swap) { return endianSwapI16(x); } else { return x; } } /** * Read a 8-bit unsigned from an input stream */ static inline uint8_t readU8(std::istream& in) { uint8_t x; in.read((char *)&x, 1); assert_eq(1, in.gcount()); return x; } template void writeIndex(std::ostream& out, index_t x, bool toBigEndian) { index_t y = endianizeIndex(x, toBigEndian); out.write((const char*)&y, sizeof(index_t)); } /** * Read a unsigned from an input stream, inverting endianness * if necessary. */ template static inline index_t readIndex(std::istream& in, bool swap) { index_t x; in.read((char *)&x, sizeof(index_t)); assert_eq(sizeof(index_t), in.gcount()); if(swap) { return endianSwapIndex(x); } else { return x; } } /** * Read a unsigned from a file descriptor, optionally inverting * endianness. */ #ifdef BOWTIE_MM template static inline index_t readIndex(int in, bool swap) { index_t x; if(read(in, (void *)&x, sizeof(index_t)) != sizeof(index_t)) { assert(false); } if(swap) { if(sizeof(index_t) == 8) { assert(false); return 0; } else if(sizeof(index_t) == 4) { return endianSwapU32(x); } else { assert_eq(sizeof(index_t), 2); return endianSwapU16(x); } } else { return x; } } #endif /** * Read a unsigned from a FILE*, optionally inverting * endianness. */ template static inline index_t readIndex(FILE* in, bool swap) { index_t x; if(fread((void *)&x, 1, sizeof(index_t), in) != sizeof(index_t)) { assert(false); } if(swap) { if(sizeof(index_t) == 8) { assert(false); return 0; } else if(sizeof(index_t) == 4) { return endianSwapU32(x); } else { assert_eq(sizeof(index_t), 2); return endianSwapU16(x); } } else { return x; } } #endif /*WORD_IO_H_*/