2018-05-03 11:28:17 +00:00
|
|
|
|
/**
|
|
|
|
|
* \file io.cc
|
|
|
|
|
* \brief Body for file reading and writing
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "io.hh"
|
2018-06-09 01:18:28 +00:00
|
|
|
|
#include "bitpack.hh"
|
2018-05-25 10:00:58 +00:00
|
|
|
|
#include <array>
|
2018-05-03 11:28:17 +00:00
|
|
|
|
|
2018-06-05 09:38:27 +00:00
|
|
|
|
using std::uint16_t;
|
2018-06-21 15:38:51 +00:00
|
|
|
|
using std::vector;
|
2018-06-05 09:38:27 +00:00
|
|
|
|
using vuint16 = vector<uint16_t>;
|
|
|
|
|
using vvuint16 = vector<vuint16>;
|
2018-05-25 10:00:58 +00:00
|
|
|
|
|
2018-05-03 11:28:17 +00:00
|
|
|
|
/**
|
|
|
|
|
* Écrit dans le fichier \p t_out les chunks passés en paramètre. Le fichier de
|
|
|
|
|
* sortie est composé des éléments suivants :\n
|
2018-06-21 15:38:51 +00:00
|
|
|
|
* - Sur deux octets sont écrit un `uint16_t` (deux octets) déterminant le
|
|
|
|
|
* nombre de chunk composant le fichier\n
|
2018-06-05 09:38:27 +00:00
|
|
|
|
* - Sont ensuite écrits les chunks sur un nombre variable d’octets suivant la
|
|
|
|
|
* taille des chunks\n
|
2018-05-03 11:28:17 +00:00
|
|
|
|
* \n
|
|
|
|
|
* Un chunk est composé de la manière qui suit :\n
|
2018-06-21 15:38:51 +00:00
|
|
|
|
* - Sur deux octets sont écrit un `uint32_t` (quatre octets) déterminant le
|
|
|
|
|
* nombre d’octets composant le chunk\n
|
2018-06-09 20:59:11 +00:00
|
|
|
|
* - Sur le nombre d’octets précisés par le header du chunk se trouvent les
|
|
|
|
|
* données compressées par l’algorithme lzw puis via bit-packing.\n
|
2018-05-03 11:28:17 +00:00
|
|
|
|
*
|
|
|
|
|
* \param[out] t_out Fichier de sortie
|
2018-06-21 15:38:51 +00:00
|
|
|
|
* \param[in] t_chunks Collection ordonnée des chunks à écrire dans \p t_out
|
2018-05-03 11:28:17 +00:00
|
|
|
|
*/
|
2018-06-21 15:38:51 +00:00
|
|
|
|
void write_file(std::ofstream &t_out, const vvuint16 &t_chunks) {
|
|
|
|
|
const auto nr_chunks = static_cast<uint16_t>(t_chunks.size());
|
|
|
|
|
#ifdef Debug
|
|
|
|
|
std::printf("Number of chunks: %u\n", nr_chunks);
|
|
|
|
|
#endif
|
|
|
|
|
t_out.write(reinterpret_cast<const char *>(&nr_chunks), sizeof(nr_chunks));
|
|
|
|
|
for (const auto &chunk : t_chunks) {
|
2018-05-25 10:00:58 +00:00
|
|
|
|
write_chunk(t_out, chunk);
|
2018-05-03 11:28:17 +00:00
|
|
|
|
}
|
2018-05-25 10:00:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2018-06-05 09:38:27 +00:00
|
|
|
|
* Écrit dans le fichier \p t_out le chunk unique \p t_chunk. Se référer à la
|
|
|
|
|
* documentation de \ref write_file pour plus de détails.
|
|
|
|
|
*
|
2018-05-25 10:00:58 +00:00
|
|
|
|
* \param t_out Output file
|
|
|
|
|
* \param t_chunk Chunk to be written to \p t_out
|
|
|
|
|
*/
|
2018-06-21 15:38:51 +00:00
|
|
|
|
void write_chunk(std::ofstream &t_out, const vuint16 &t_chunk) {
|
|
|
|
|
#ifdef Debug
|
|
|
|
|
std::printf("Chunk!\n");
|
|
|
|
|
#endif
|
2018-06-09 01:18:28 +00:00
|
|
|
|
const auto output = pack(t_chunk);
|
2018-06-11 19:26:05 +00:00
|
|
|
|
const auto chunk_size = static_cast<uint32_t>(output.size());
|
2018-06-21 15:38:51 +00:00
|
|
|
|
t_out.write(reinterpret_cast<const char *>(&chunk_size), sizeof(chunk_size));
|
|
|
|
|
t_out.write(reinterpret_cast<const char *>(output.data()),
|
|
|
|
|
sizeof(output[0]) * output.size());
|
2018-05-03 11:28:17 +00:00
|
|
|
|
}
|