You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

158 lines
5.9 KiB

/*
* Copyright (C) 2011-2018 Daniel Scharrer
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the author(s) be held liable for any damages
* arising from the use of this software.
*
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
*
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/*!
* \file
*
* Abstraction for reading the embedded or external raw setup data.
*/
#ifndef INNOEXTRACT_STREAM_SLICE_HPP
#define INNOEXTRACT_STREAM_SLICE_HPP
#include <ios>
#include <string>
#include <boost/iostreams/concepts.hpp>
#include <boost/filesystem/path.hpp>
#include "util/fstream.hpp"
namespace stream {
//! Error thrown by \ref slice_reader if there was a problem.
struct slice_error : public std::ios_base::failure {
explicit slice_error(const std::string & msg) : std::ios_base::failure(msg) { }
};
/*!
* Abstraction for reading either data embedded inside the setup executable or from
* multiple external slices.
*
* Setup data contained in the executable is located by a non-zeore
* \ref loader::offsets::data_offset.
*
* The contained data is made up of one or more \ref chunk "chunks"
* (read by \ref chunk_reader), which in turn contain one or more \ref file "files"
* (read by \ref file_reader).
*/
class slice_reader : public boost::iostreams::source {
typedef boost::filesystem::path path_type;
// Information for reading embedded setup data
const boost::uint32_t data_offset;
// Information for eading external setup data
path_type dir; //!< Slice directory specified at construction.
std::string base_file; //!< Base file name for slices.
std::string base_file2; //!< Fallback base filename for slices.
const size_t slices_per_disk; //!< Number of slices grouped into each disk (for names).
// Information about the current slice
size_t current_slice; //!< Number of the currently opened slice.
boost::uint32_t slice_size; //!< Size in bytes of the currently opened slice.
// Streams
util::ifstream ifs; //!< File input stream used when reading from external slices.
std::istream * is; //!< Input stream to read from.
void seek(size_t slice);
bool open_file(const path_type & file);
bool open_file_case_insensitive(const path_type & dir, const path_type & filename);
void open(size_t slice);
public:
static std::string slice_filename(const std::string & basename, size_t slice,
size_t slices_per_disk = 1);
/*!
* Construct a \ref slice_reader to read from data inside the setup file.
* Seeking to anything except the zeroeth slice is not allowed.
*
* \param istream A seekable input stream for the setup executable.
* The initial read position of the stream is ignored.
* \param data_offset The offset within the given stream where the setup data starts.
* This offset is given by \ref loader::offsets::data_offset.
*
* The constructed reader will allow reading the byte range [data_offset, file end)
* from the setup executable and provide this as the range [0, file end - data_offset).
*/
slice_reader(std::istream * istream, boost::uint32_t data_offset);
/*!
* Construct a \ref slice_reader to read from external data slices (aka disks).
*
* Slice files must be located at \c $dir/$base_file-$disk.bin
* or \c $dir/$base_file-$disk$sliceletter.bin if \c slices_per_disk is greater
* than \c 1.
*
* The disk number is given by \code slice / slices_per_disk + 1 \endcode while
* the sliceletter is the ASCII char \code 'a' + (slice % slices_per_disk) \endcode.
*
* \param dir The directory containing the slice files.
* \param basename The base name for slice files.
* \param basename2 Alternative base name for slice files.
* \param slices_per_disk How many slices are grouped into one disk. Must not be \c 0.
*/
slice_reader(const path_type & dir, const std::string & basename, const std::string & basename2,
size_t slices_per_disk);
/*!
* Attempt to seek to an offset within a slice.
*
* \param slice The slice to seek to.
* \param offset The byte offset to seek to within the given slice.
*
* \return \c false if the requested slice could not be opened, or if the requested
* offset is not a valid position in that slice - \c true otherwise.
*/
bool seek(size_t slice, boost::uint32_t offset);
/*!
* Read a number of bytes starting at the current slice and offset within that slice.
*
* \param buffer Buffer to receive the bytes read.
* \param bytes Number of bytes to read.
*
* The current offset will be advanced by the number of bytes read. It is not an error
* to read past the end of the current slice (unless it is the last slice). Doing so
* will automatically seek to the start of the next slice and continue reading from
* there.
*
* \return The number of bytes read or \c -1 if there was an error. Unless we are at the
* end of the last slice, this function blocks until the number of requested
* bytes have been read.
*/
std::streamsize read(char * buffer, std::streamsize bytes);
//! \return the number currently opened slice.
size_t slice() { return current_slice; }
//! \return true a slice is currently open.
bool is_open() { return (is != &ifs || ifs.is_open()); }
};
} // namespace stream
#endif // INNOEXTRACT_STREAM_SLICE_HPP