libcruft-util/io.hpp

174 lines
4.8 KiB
C++
Raw Normal View History

2011-05-23 17:18:52 +10:00
/*
2015-04-13 18:05:28 +10:00
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
2011-05-23 17:18:52 +10:00
*
2015-04-13 18:05:28 +10:00
* http://www.apache.org/licenses/LICENSE-2.0
2011-05-23 17:18:52 +10:00
*
2015-04-13 18:05:28 +10:00
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
2011-05-23 17:18:52 +10:00
*
* Copyright 2010-2014 Danny Robson <danny@nerdcruft.net>
2011-05-23 17:18:52 +10:00
*/
#ifndef __UTIL_IO_HPP
#define __UTIL_IO_HPP
#include "platform.hpp"
#include "posix/fd.hpp"
2011-05-25 23:02:39 +10:00
#include <cstddef>
2011-05-23 17:18:52 +10:00
#include <cstdint>
#include <cstdio>
#include <experimental/filesystem>
2016-11-17 18:32:08 +11:00
#include <vector>
#include <streambuf>
2011-05-23 17:18:52 +10:00
#ifdef PLATFORM_WIN32
#include <windows.h>
2015-10-29 10:48:11 +11:00
#else
#define O_BINARY 0
#endif
2011-05-23 17:18:52 +10:00
namespace util {
2015-07-02 16:34:17 +10:00
//-------------------------------------------------------------------------
/// Reads an entire file into memory.
template <typename T = std::byte>
std::vector<T> slurp (const std::experimental::filesystem::path&);
template <typename T = std::byte>
std::vector<T> slurp (FILE *);
2015-07-02 16:34:17 +10:00
2015-10-29 17:52:19 +11:00
///////////////////////////////////////////////////////////////////////////
void write (const posix::fd&, const void *restrict data, size_t bytes);
2015-07-02 16:34:17 +10:00
//-------------------------------------------------------------------------
template <
typename ValueT,
typename = std::enable_if_t<
sizeof (util::view<const ValueT*>) == 1,
void
>
>
void
write (const posix::fd &dst, util::view<const ValueT*> data)
{
write (dst, std::data (data), std::size (data));
}
2015-10-20 21:04:14 +11:00
//-------------------------------------------------------------------------
2015-07-02 16:34:17 +10:00
template <typename T>
void write (const posix::fd &_fd, const T &data)
{
return write (_fd, make_view (data));
}
2015-07-02 16:34:17 +10:00
///------------------------------------------------------------------------
/// writes all data from the provided view into the file-like-object
///
/// writing will continually iterate until the entire buffer has been
/// dispatched so as to avoid issues with partial writes. will block until
/// such time as the entire buffer has written.
///
/// an exception may be thrown in the event forward progress is impossible.
/// in this event the progress may not be reported to the caller. in the
/// future an exception member variable may expose the information.
template <
typename DstT,
typename IteratorA,
typename IteratorB
>
2017-12-28 17:49:58 +11:00
util::view<IteratorA,IteratorB>
write (DstT &dst, const util::view<IteratorA, IteratorB> src)
{
2017-12-28 17:49:58 +11:00
auto remain = src;
while (!remain.empty ())
remain = remain.consume (dst.write (remain));
2017-12-28 17:49:58 +11:00
return src;
}
///////////////////////////////////////////////////////////////////////////
2012-11-09 15:16:07 +11:00
class indenter : public std::streambuf {
2012-04-12 14:06:26 +10:00
protected:
std::streambuf* m_dest;
bool m_line_start;
std::string m_indent;
std::ostream* m_owner;
protected:
2015-06-30 22:02:54 +10:00
virtual int overflow (int ch) override;
2012-04-12 14:06:26 +10:00
public:
explicit indenter (std::streambuf* _dest, size_t _indent = 4);
explicit indenter (std::ostream& _dest, size_t _indent = 4);
virtual ~indenter ();
};
2014-06-29 22:49:26 +10:00
//-------------------------------------------------------------------------
template <typename T>
struct indented {
2017-05-23 12:50:51 +10:00
explicit indented (const T &_data);
2014-06-29 22:49:26 +10:00
const T &data;
};
//-------------------------------------------------------------------------
2014-06-29 22:49:26 +10:00
template <typename T>
indented<T>
make_indented (const T &_data);
//-------------------------------------------------------------------------
template <typename T>
std::ostream&
operator<< (std::ostream &os, const util::indented<T> &v);
2014-06-29 22:49:26 +10:00
//-------------------------------------------------------------------------
class scoped_cwd {
public:
scoped_cwd ();
~scoped_cwd ();
protected:
std::string m_original;
};
//-------------------------------------------------------------------------
2012-04-26 18:19:12 +10:00
class path_error : public std::runtime_error {
2016-02-12 13:36:03 +11:00
public:
2017-05-23 12:50:51 +10:00
explicit path_error (const std::experimental::filesystem::path &path);
2016-02-12 13:36:03 +11:00
const std::experimental::filesystem::path& path (void) const noexcept;
2016-02-12 13:36:03 +11:00
private:
const std::experimental::filesystem::path m_path;
2012-04-26 18:19:12 +10:00
};
2016-04-19 14:52:05 +10:00
class stream_error : public std::exception {
public:
virtual const char* what (void) const noexcept
{ return "error on C stream"; }
};
}
2011-05-23 17:18:52 +10:00
#ifdef PLATFORM_WIN32
#include "io_win32.hpp"
#else
#include "io_posix.hpp"
#endif
#include "io.ipp"
2014-06-29 22:49:26 +10:00
2011-05-23 17:18:52 +10:00
#endif