libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
cosmos::StreamIO Class Reference

Wrapper around file descriptors for streaming I/O access. More...

#include <StreamIO.hxx>

+ Inheritance diagram for cosmos::StreamIO:

Public Types

enum class  SeekType : int {
  SET = SEEK_SET , CUR = SEEK_CUR , END = SEEK_END , DATA = SEEK_DATA ,
  HOLE = SEEK_HOLE
}
 Different methods for changing the file read/write position. More...
 

Public Member Functions

 StreamIO (FileDescriptor &fd)
 
 StreamIO (const StreamIO &)=delete
 
StreamIOoperator= (const StreamIO &)=delete
 
StreamIOoperator= (StreamIO &&) noexcept
 
size_t read (void *buf, size_t length)
 Read up to length bytes from the file into buf.
 
size_t write (const void *buf, size_t length)
 Write up to length bytes from buf into the underlying file.
 
size_t write (const std::string_view data)
 string_view wrapper around write(const void*, size_t).
 
void readAll (void *buf, size_t length)
 Read all length bytes from the underlying file.
 
void readAll (std::string &s, size_t length)
 Like readAll(void*, size_t) using an STL string.
 
void writeAll (const void *buf, size_t length)
 Write all length bytes into the underlying file.
 
void writeAll (const std::string_view data)
 string_view wrapper around writeAll(const void*, size_t).
 
bool read (ReadIOVector &iovec)
 Read data from file into a vector of data regions.
 
bool write (WriteIOVector &iovec)
 Write data to file from a vector of data regions.
 
void readAll (ReadIOVector &iovec)
 Read into all data regions specified in iovec.
 
void writeAll (WriteIOVector &iovec)
 Write all data regions specified in iovec.
 
off_t seek (const SeekType type, off_t off)
 Seek to the given offset based on the given offset type.
 
off_t seekFromStart (off_t off)
 Seek to the given offset relative to the start of the file.
 
off_t seekFromCurrent (off_t off)
 Seek to the given offset relative to the current file position.
 
off_t seekFromEnd (off_t off)
 Seek to the given offset relative to the end of the file.
 

Protected Attributes

FileDescriptorm_stream_fd
 

Detailed Description

Wrapper around file descriptors for streaming I/O access.

Streaming I/O means that a file's read/write position is maintained by the operating system and data is exchanged by means of read/write operations that transfer data from the current process to the file and vice versa.

This is the most common access mode for files but also somewhat inefficient. In contrast e.g. memory mapped files can be more efficient.

Some special devices or file types may also support streaming I/O access. This type can also be used with them - but be sure to understand the special I/O semantics for the respective file type when using it with this wrapper.

Beyond read and write operations this type also offers seek operations. Not all file types are seekable though and the operation can fail.

This type will not take ownership of the provided file descriptor. It is only meant as an access wrapper, not as a permanent representation of the backed file.

StreamIO has a fixed coupling to the assigned file descriptor. It can be used as a mixin class or as a base class.

Definition at line 40 of file StreamIO.hxx.

Member Enumeration Documentation

◆ SeekType

enum class cosmos::StreamIO::SeekType : int
strong

Different methods for changing the file read/write position.

Enumerator
SET 

Set a new absolute position.

CUR 

Set a position relative to the current one.

END 

Set a position relative to the end of the file. Seek to a non-hole position.

DATA 

For files with holes in them this seeks the next position containing data that is equal or greater to the provided offset.

HOLE 

Seek to a hole position.

For files with holes in them this seeks the next position that is part of a hole that is equal or greater to the provided offset. The end-of-file is considered a whole in this context.

Definition at line 44 of file StreamIO.hxx.

44 : int {
45 SET = SEEK_SET,
46 CUR = SEEK_CUR,
47 END = SEEK_END,
49
54 DATA = SEEK_DATA,
56
62 HOLE = SEEK_HOLE
63 };
@ SET
Set a new absolute position.
@ HOLE
Seek to a hole position.
@ END
Set a position relative to the end of the file. Seek to a non-hole position.
@ CUR
Set a position relative to the current one.

Constructor & Destructor Documentation

◆ StreamIO()

cosmos::StreamIO::StreamIO ( FileDescriptor & fd)
inlineexplicit

Definition at line 67 of file StreamIO.hxx.

67 :
68 m_stream_fd{fd}
69 {}

Member Function Documentation

◆ operator=()

StreamIO & cosmos::StreamIO::operator= ( StreamIO && )
inlinenoexcept

Definition at line 74 of file StreamIO.hxx.

74 {
75 // simply do nothing, the fixed coupling to the FD remains and
76 // should reflect the new object state.
77 return *this;
78 }

◆ read() [1/2]

bool cosmos::StreamIO::read ( ReadIOVector & iovec)

Read data from file into a vector of data regions.

The iovec specifies memory regions into which data from the file should be written. The data will be filled sequentially starting from the first memory region.

Partial reads can occur, thus on return the length and base fields of each vector entry will be updated to reflect this. The return value is a flag indicating whether the complete vector has been filled, or whether a partial read occurred.

These vector I/O operations are useful when structured binary of fixed size is transferred e.g. in network protocols for the different header layers. This way the individual headers can be kept in distinct places while only a single system call is necessary to transfer them.

Definition at line 75 of file StreamIO.cxx.

75 {
76 while (true) {
77 const auto res = ::readv(
78 to_integral(m_stream_fd.raw()), iovec.raw(), iovec.size());
79
80 if (res < 0) {
81 handleIOError("reading to vector from file");
82 continue;
83 }
84
85 return iovec.update(static_cast<size_t>(res));
86 }
87}
FileNum raw() const
Returns the primitive file descriptor contained in the object.

◆ read() [2/2]

size_t cosmos::StreamIO::read ( void * buf,
size_t length )

Read up to length bytes from the file into buf.

An attempt is made to read data from the underlying file object and place it into buf. buf needs to be able to hold at least length bytes. Short reads can occur in which case less bytes will be read. The number of bytes actually read is returned from this function.

A return value of zero indicates that the End-of-File has been reached and no further data can be obtained.

On error conditions an exception is thrown.

Definition at line 26 of file StreamIO.cxx.

26 {
27 while (true) {
28 auto res = ::read(to_integral(m_stream_fd.raw()), buf, length);
29
30 if (res < 0) {
31 handleIOError("reading from file");
32 continue;
33 }
34
35 return static_cast<size_t>(res);
36 }
37}
size_t read(void *buf, size_t length)
Read up to length bytes from the file into buf.
Definition StreamIO.cxx:26

◆ readAll() [1/3]

void cosmos::StreamIO::readAll ( ReadIOVector & iovec)
inline

Read into all data regions specified in iovec.

This is just like read(IOVector&) but it takes care of partial reads and continues until all data of the IOVector has been filled or an error occurs. On return the complete vector has been filled.

Definition at line 189 of file StreamIO.hxx.

189 {
190 while (!read(iovec)) {
191 ;
192 }
193 }

◆ readAll() [2/3]

void cosmos::StreamIO::readAll ( std::string & s,
size_t length )
inline

Like readAll(void*, size_t) using an STL string.

Definition at line 125 of file StreamIO.hxx.

125 {
126 s.resize(length);
127 try {
128 readAll(s.data(), length);
129 } catch(...) {
130 s.clear();
131 throw;
132 }
133 }
void readAll(void *buf, size_t length)
Read all length bytes from the underlying file.
Definition StreamIO.cxx:39

◆ readAll() [3/3]

void cosmos::StreamIO::readAll ( void * buf,
size_t length )

Read all length bytes from the underlying file.

This behaves just like read() with the exception that on short reads the operation will be continued until all length bytes have been obtained from the file.

An End-of-File condition is considered an error in this context and results in a RuntimeError exception. If the function returns normally then all length bytes will have been obtained.

Definition at line 39 of file StreamIO.cxx.

39 {
40 size_t res;
41 while (length != 0) {
42 res = read(buf, length);
43 buf = reinterpret_cast<char*>(buf) + res;
44 length -= res;
45
46 if (res == 0) {
47 cosmos_throw (RuntimeError("unexpected EOF"));
48 }
49 }
50}

◆ seek()

off_t cosmos::StreamIO::seek ( const SeekType type,
off_t off )

Seek to the given offset based on the given offset type.

Definition at line 102 of file StreamIO.cxx.

102 {
103 const auto res = ::lseek(to_integral(m_stream_fd.raw()), off, to_integral(type));
104
105 if (res == static_cast<off_t>(-1)) {
106 cosmos_throw (ApiError("lseek()"));
107 }
108
109 return res;
110}

◆ seekFromCurrent()

off_t cosmos::StreamIO::seekFromCurrent ( off_t off)
inline

Seek to the given offset relative to the current file position.

Definition at line 214 of file StreamIO.hxx.

214{ return seek(SeekType::CUR, off); }
off_t seek(const SeekType type, off_t off)
Seek to the given offset based on the given offset type.
Definition StreamIO.cxx:102

◆ seekFromEnd()

off_t cosmos::StreamIO::seekFromEnd ( off_t off)
inline

Seek to the given offset relative to the end of the file.

Definition at line 216 of file StreamIO.hxx.

216{ return seek(SeekType::END, off); }

◆ seekFromStart()

off_t cosmos::StreamIO::seekFromStart ( off_t off)
inline

Seek to the given offset relative to the start of the file.

Definition at line 212 of file StreamIO.hxx.

212{ return seek(SeekType::SET, off); }

◆ write() [1/3]

size_t cosmos::StreamIO::write ( const std::string_view data)
inline

string_view wrapper around write(const void*, size_t).

Definition at line 108 of file StreamIO.hxx.

108 {
109 return write(data.data(), data.size());
110 }
size_t write(const void *buf, size_t length)
Write up to length bytes from buf into the underlying file.
Definition StreamIO.cxx:52

◆ write() [2/3]

size_t cosmos::StreamIO::write ( const void * buf,
size_t length )

Write up to length bytes from buf into the underlying file.

An attempt is made to write data from the given buf and pass it to the underlying file object. buf needs to hold at least length bytes of data. Short writes can occur in which case less bytes will be written. The number of bytes actually written is returned from this function.

On error conditions an exception is thrown.

Definition at line 52 of file StreamIO.cxx.

52 {
53 while (true) {
54 auto res = ::write(to_integral(m_stream_fd.raw()), buf, length);
55
56 if (res < 0) {
57 handleIOError("writing to file");
58 continue;
59 }
60
61 return static_cast<size_t>(res);
62 }
63}

◆ write() [3/3]

bool cosmos::StreamIO::write ( WriteIOVector & iovec)

Write data to file from a vector of data regions.

The iovec specifies memory regions whose data will be written to the file. The data will be written sequentially starting from the first memory region.

Partial writes can occur, thus on return the length and base fields of each vector entry will be updated to reflect this. The return value is a flag indicating whether the complete vector has been written out, or whether a partial write occurred.

Definition at line 89 of file StreamIO.cxx.

89 {
90 while (true) {
91 auto res = ::writev(to_integral(m_stream_fd.raw()), iovec.raw(), iovec.size());
92
93 if (res < 0) {
94 handleIOError("writing vector to file");
95 continue;
96 }
97
98 return iovec.update(static_cast<size_t>(res));
99 }
100}

◆ writeAll() [1/3]

void cosmos::StreamIO::writeAll ( const std::string_view data)
inline

string_view wrapper around writeAll(const void*, size_t).

Definition at line 147 of file StreamIO.hxx.

147 {
148 return writeAll(data.data(), data.size());
149 }
void writeAll(const void *buf, size_t length)
Write all length bytes into the underlying file.
Definition StreamIO.cxx:65

◆ writeAll() [2/3]

void cosmos::StreamIO::writeAll ( const void * buf,
size_t length )

Write all length bytes into the underlying file.

This behaves just like write() with the exception that on short writes the operation will be continued until all length bytes have been written to the file.

If the function returns normally then all length bytes will have been transferred.

Definition at line 65 of file StreamIO.cxx.

65 {
66 size_t res;
67
68 while (length != 0) {
69 res = write(buf, length);
70 buf = reinterpret_cast<const char*>(buf) + res;
71 length -= res;
72 }
73}

◆ writeAll() [3/3]

void cosmos::StreamIO::writeAll ( WriteIOVector & iovec)
inline

Write all data regions specified in iovec.

This is just like write(const IOVector&) but it takes care of partial writes and continues until all data of the IOVector has been written out or an error occurs. On return the complete vector has been written.

Definition at line 202 of file StreamIO.hxx.

202 {
203 while (!write(iovec)) {
204 ;
205 }
206 }

Member Data Documentation

◆ m_stream_fd

FileDescriptor& cosmos::StreamIO::m_stream_fd
protected

Definition at line 220 of file StreamIO.hxx.


The documentation for this class was generated from the following files: