In the code fragment below, is there a way to handle ENOSPC
?
#include <fstream>
#include <iostream>
#include <boost/iostreams/filtering_streambuf.hpp>
#include <boost/iostreams/copy.hpp>
#include <boost/iostreams/filter/bzip2.hpp>
// open input file stream of the bzip2 file
std::ifstream ifs("file.bz2");
// open output stream to the "full" device
// full device is a "utility-device" to check how applications handle ENOSPC
// more details in "man full"
std::ofstream ofs("/dev/full");
// Setup the iostreams filter
boost::iostreams::filtering_streambuf<boost::iostreams::output> filters;
filters.push(boost::iostreams::bzip2_decompressor());
filters.push(ofs);
// "run" the filter
boost::iostreams::copy(ifs, filters);
If I do strace
of the compiled binary, the code seem to infinitely call writev()
with the same data and returns ENOSPC
error.
writev(4, [{NULL, 0}, {"DATA DATA "..., 4096}], 2) = -1 ENOSPC (No space left on device)
How can this error be handled or made thrown as an error from boost::iostreams::copy()
.
Is it possible to set appropriate exceptions()
on the ofstream
object? I tried ofs.exceptions(std::ios::badbit | std::ios::failbit)
but it didn't make any difference.
The code above was compiled with GCC and run on Linux. Boost version 1.55.
It's stuck in an infinite loop in
non_blocking_adaptor<Device>::write(...)
:iostream::write(device_, ...
keeps returning 0 (so n stays n, amt and result stay 0).It would appear to be a bug in Boost IOstreams. Perhaps it was introduced when preliminary support for non-blocking (synchronous) IO was added. According to the documentation, this should be a work in progress.
Particularly enlightening was: http://www.boost.org/doc/libs/1_55_0/libs/iostreams/doc/guide/asynchronous.html
(bold mine) It does appear as if IOStreams violated this principle by transforming the E_NOSPC condition into a temporary failure notification.
boost::iostreams::copy
special-cases the case where source and destination Indirect devices. In this case, both are indirect. Now the special case wraps the sink in anon_blocking_adaptor
. I don't know why, and this seems to contradict the following general advice taken from the same documentation page:I tried to replace the files with
file_sink
andfile_source
instead, but there was no change. :(Here's my reduced test case which still reproduces the problem:
Perhaps you should report this as a bug with the developers/mailing list.