/*** * ==++== * * Copyright (c) Microsoft Corporation. All rights reserved. * 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 * http://www.apache.org/licenses/LICENSE-2.0 * * 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. * * ==--== * =+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ * * Asynchronous File streams * * For the latest on this and related APIs, please see http://casablanca.codeplex.com. * * =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- ****/ #pragma once #ifndef _CASA_FILE_STREAMS_H #define _CASA_FILE_STREAMS_H #include "cpprest/details/fileio.h" #include "cpprest/astreambuf.h" #include "cpprest/streams.h" #include #ifndef _CONCRT_H #ifndef _LWRCASE_CNCRRNCY #define _LWRCASE_CNCRRNCY // Note to reader: we're using lower-case namespace names everywhere, but the 'Concurrency' namespace // is capitalized for historical reasons. The alias let's us pretend that style issue doesn't exist. namespace Concurrency { } namespace concurrency = Concurrency; #endif #endif namespace Concurrency { namespace streams { // Forward declarations template class file_buffer; namespace details { // This operation queue is NOT thread safe class async_operation_queue { pplx::task m_lastOperation; public: async_operation_queue() { m_lastOperation = pplx::task_from_result(); } // It only accepts functors that take no argument and return pplx::task // This function may execute op inline, thus it could throw immediately template auto enqueue_operation(Func &&op) -> decltype(op()) { decltype(op()) res; // res is task , which always has default constructor if (m_lastOperation.is_done()) { res = op(); // Exceptions are expected to be thrown directly without catching if (res.is_done()) return res; } else { res = m_lastOperation.then([=] { return op(); // It will cause task unwrapping }); } m_lastOperation = res.then([&] (decltype(op())) { // This empty task is necessary for keeping the rest of the operations on the list running // even when the previous operation gets error. // Don't observe exception here. }); return res; } void wait() const { m_lastOperation.wait(); } }; /// /// Private stream buffer implementation for file streams. /// The class itself should not be used in application code, it is used by the stream definitions farther down in the header file. /// template class basic_file_buffer : public details::streambuf_state_manager<_CharType> { public: typedef typename basic_streambuf<_CharType>::traits traits; typedef typename basic_streambuf<_CharType>::int_type int_type; typedef typename basic_streambuf<_CharType>::pos_type pos_type; typedef typename basic_streambuf<_CharType>::off_type off_type; virtual ~basic_file_buffer() { if( this->can_read() ) { this->_close_read().wait(); } if (this->can_write()) { this->_close_write().wait(); } } protected: /// /// can_seek is used to determine whether a stream buffer supports seeking. /// virtual bool can_seek() const { return this->is_open(); } /// /// has_size is used to determine whether a stream buffer supports size(). /// virtual bool has_size() const { return this->is_open(); } virtual utility::size64_t size() const { if (!this->is_open()) return 0; return _get_size(m_info, sizeof(_CharType)); } /// /// Gets the stream buffer size, if one has been set. /// /// The direction of buffering (in or out) /// An implementation that does not support buffering will always return '0'. virtual size_t buffer_size(std::ios_base::openmode direction = std::ios_base::in) const { if ( direction == std::ios_base::in ) return m_info->m_buffer_size; else return 0; } /// /// Sets the stream buffer implementation to buffer or not buffer. /// /// The size to use for internal buffering, 0 if no buffering should be done. /// The direction of buffering (in or out) /// An implementation that does not support buffering will silently ignore calls to this function and it will not have /// any effect on what is returned by subsequent calls to buffer_size(). virtual void set_buffer_size(size_t size, std::ios_base::openmode direction = std::ios_base::in) { if ( direction == std::ios_base::out ) return; m_info->m_buffer_size = size; if ( size == 0 && m_info->m_buffer != nullptr ) { delete m_info->m_buffer; m_info->m_buffer = nullptr; } } /// /// For any input stream, in_avail returns the number of characters that are immediately available /// to be consumed without blocking. May be used in conjunction with to read data without /// incurring the overhead of using tasks. /// virtual size_t in_avail() const { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); return _in_avail_unprot(); } size_t _in_avail_unprot() const { if ( !this->is_open() ) return 0; if ( m_info->m_buffer == nullptr || m_info->m_buffill == 0 ) return 0; if ( m_info->m_bufoff > m_info->m_rdpos || (m_info->m_bufoff+m_info->m_buffill) < m_info->m_rdpos ) return 0; msl::safeint3::SafeInt rdpos(m_info->m_rdpos); msl::safeint3::SafeInt buffill(m_info->m_buffill); msl::safeint3::SafeInt bufpos = rdpos - m_info->m_bufoff; return buffill - bufpos; } _file_info * _close_stream() { // indicate that we are no longer open auto fileInfo = m_info; m_info = nullptr; return fileInfo; } static pplx::task _close_file(_In_ _file_info * fileInfo) { pplx::task_completion_event result_tce; auto callback = new _filestream_callback_close(result_tce); if ( !_close_fsb_nolock(&fileInfo, callback) ) { delete callback; return pplx::task_from_result(); } return pplx::create_task(result_tce); } // Workaround GCC compiler bug https://gcc.gnu.org/bugzilla/show_bug.cgi?id=58972 void _invoke_parent_close_read() { streambuf_state_manager<_CharType>::_close_read(); } pplx::task _close_read() { return m_readOps.enqueue_operation([this] { _invoke_parent_close_read(); if (this->can_write()) { return pplx::task_from_result(); } else { // Neither heads are open. Close the underlying device // to indicate that we are no longer open auto fileInfo = _close_stream(); return _close_file(fileInfo); } }); } pplx::task _close_write() { streambuf_state_manager<_CharType>::_close_write(); if (this->can_read()) { // Read head is still open. Just flush the write data return flush_internal(); } else { // Neither heads are open. Close the underlying device // We need to flush all writes if the file was opened for writing. return flush_internal().then([=](pplx::task flushTask) -> pplx::task { // swallow exception from flush try { flushTask.wait(); } catch(...) { } // indicate that we are no longer open auto fileInfo = this->_close_stream(); return this->_close_file(fileInfo); }); } } /// /// Writes a single byte to an output stream. /// /// The byte to write /// A task that holds the value of the byte written. This is EOF if the write operation fails. virtual pplx::task _putc(_CharType ch) { auto result_tce = pplx::task_completion_event(); auto callback = new _filestream_callback_write(m_info, result_tce); // Potentially we should consider deprecating this API, it is TERRIBLY inefficient. std::shared_ptr<_CharType> sharedCh; try { sharedCh = std::make_shared<_CharType>(ch); } catch (const std::bad_alloc &) { delete callback; throw; } size_t written = _putn_fsb(m_info, callback, sharedCh.get(), 1, sizeof(_CharType)); if (written == sizeof(_CharType)) { delete callback; return pplx::task_from_result(ch); } return pplx::create_task(result_tce).then([sharedCh](size_t) { return static_cast(*sharedCh); }); } /// /// Allocates a contiguous memory block and returns it. /// /// The number of characters to allocate. /// A pointer to a block to write to, null if the stream buffer implementation does not support alloc/commit. _CharType* _alloc(size_t) { return nullptr; } /// /// Submits a block already allocated by the stream buffer. /// /// Count of characters to be commited. void _commit(size_t) { } /// /// Gets a pointer to the next already allocated contiguous block of data. /// /// A reference to a pointer variable that will hold the address of the block on success. /// The number of contiguous characters available at the address in 'ptr.' /// true if the operation succeeded, false otherwise. /// /// A return of false does not necessarily indicate that a subsequent read operation would fail, only that /// there is no block to return immediately or that the stream buffer does not support the operation. /// The stream buffer may not de-allocate the block until is called. /// If the end of the stream is reached, the function will return true, a null pointer, and a count of zero; /// a subsequent read will not succeed. /// virtual bool acquire(_Out_ _CharType*& ptr, _Out_ size_t& count) { ptr = nullptr; count = 0; return false; } /// /// Releases a block of data acquired using . This frees the stream buffer to de-allocate the /// memory, if it so desires. Move the read position ahead by the count. /// /// A pointer to the block of data to be released. /// The number of characters that were read. virtual void release(_Out_writes_ (count) _CharType *, _In_ size_t count) { (void)(count); } /// /// Writes a number of characters to the stream. /// /// A pointer to the block of data to be written. /// The number of characters to write. /// A task that holds the number of characters actually written, either 'count' or 0. virtual pplx::task _putn(const _CharType *ptr, size_t count) { auto result_tce = pplx::task_completion_event(); auto callback = new _filestream_callback_write(m_info, result_tce); size_t written = _putn_fsb(m_info, callback, ptr, count, sizeof(_CharType)); if ( written != 0 && written != -1 ) { delete callback; written = written/sizeof(_CharType); return pplx::task_from_result(written); } return pplx::create_task(result_tce); } // Temporarily needed until the deprecated putn is removed. virtual pplx::task _putn(const _CharType *ptr, size_t count, bool copy) { if (copy) { auto sharedData = std::make_shared>(ptr, ptr + count); return _putn(ptr, count).then([sharedData](size_t size) { return size; }); } else { return _putn(ptr, count); } } /// /// Reads a single byte from the stream and advance the read position. /// /// A task that holds the value of the byte read. This is EOF if the read fails. virtual pplx::task _bumpc() { return m_readOps.enqueue_operation([this]()-> pplx::task { if ( _in_avail_unprot() > 0 ) { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); // Check again once the lock is held. if ( _in_avail_unprot() > 0 ) { auto bufoff = m_info->m_rdpos - m_info->m_bufoff; _CharType ch = m_info->m_buffer[bufoff*sizeof(_CharType)]; m_info->m_rdpos += 1; return pplx::task_from_result(ch); } } auto result_tce = pplx::task_completion_event(); auto callback = new _filestream_callback_bumpc(m_info, result_tce); size_t ch = _getn_fsb(m_info, callback, &callback->m_ch, 1, sizeof(_CharType)); if ( ch == sizeof(_CharType) ) { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); m_info->m_rdpos += 1; _CharType ch1 = (_CharType)callback->m_ch; delete callback; return pplx::task_from_result(ch1); } return pplx::create_task(result_tce); }); } /// /// Reads a single byte from the stream and advance the read position. /// /// The value of the byte. EOF if the read fails. if an asynchronous read is required /// This is a synchronous operation, but is guaranteed to never block. virtual int_type _sbumpc() { m_readOps.wait(); if ( m_info->m_atend ) return traits::eof(); if ( _in_avail_unprot() == 0 ) return traits::requires_async(); pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); if ( _in_avail_unprot() == 0 ) return traits::requires_async(); auto bufoff = m_info->m_rdpos - m_info->m_bufoff; _CharType ch = m_info->m_buffer[bufoff*sizeof(_CharType)]; m_info->m_rdpos += 1; return (int_type)ch; } pplx::task _getcImpl() { if ( _in_avail_unprot() > 0 ) { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); // Check again once the lock is held. if ( _in_avail_unprot() > 0 ) { auto bufoff = m_info->m_rdpos - m_info->m_bufoff; _CharType ch = m_info->m_buffer[bufoff*sizeof(_CharType)]; return pplx::task_from_result(ch); } } auto result_tce = pplx::task_completion_event(); auto callback = new _filestream_callback_getc(m_info, result_tce); size_t ch = _getn_fsb(m_info, callback, &callback->m_ch, 1, sizeof(_CharType)); if ( ch == sizeof(_CharType) ) { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); _CharType ch1 = (_CharType)callback->m_ch; delete callback; return pplx::task_from_result(ch1); } return pplx::create_task(result_tce); } /// /// Reads a single byte from the stream without advancing the read position. /// /// The value of the byte. EOF if the read fails. pplx::task _getc() { return m_readOps.enqueue_operation([this]()-> pplx::task { return _getcImpl(); }); } /// /// Reads a single byte from the stream without advancing the read position. /// /// The value of the byte. EOF if the read fails. if an asynchronous read is required /// This is a synchronous operation, but is guaranteed to never block. int_type _sgetc() { m_readOps.wait(); if ( m_info->m_atend ) return traits::eof(); if ( _in_avail_unprot() == 0 ) return traits::requires_async(); pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); if ( _in_avail_unprot() == 0 ) return traits::requires_async(); auto bufoff = m_info->m_rdpos - m_info->m_bufoff; _CharType ch = m_info->m_buffer[bufoff*sizeof(_CharType)]; return (int_type)ch; } /// /// Advances the read position, then return the next character without advancing again. /// /// A task that holds the value of the byte, which is EOF if the read fails. virtual pplx::task _nextc() { return m_readOps.enqueue_operation([this]()-> pplx::task { _seekrdpos_fsb(m_info, m_info->m_rdpos+1, sizeof(_CharType)); if ( m_info->m_atend ) return pplx::task_from_result(basic_file_buffer<_CharType>::traits::eof()); return this->_getcImpl(); }); } /// /// Retreats the read position, then return the current character without advancing. /// /// A task that holds the value of the byte. The value is EOF if the read fails, requires_async if an asynchronous read is required virtual pplx::task _ungetc() { return m_readOps.enqueue_operation([this]()-> pplx::task { if ( m_info->m_rdpos == 0 ) return pplx::task_from_result(basic_file_buffer<_CharType>::traits::eof()); _seekrdpos_fsb(m_info, m_info->m_rdpos-1, sizeof(_CharType)); return this->_getcImpl(); }); } /// /// Reads up to a given number of characters from the stream. /// /// The address of the target memory area /// The maximum number of characters to read /// A task that holds the number of characters read. This number is O if the end of the stream is reached, EOF if there is some error. virtual pplx::task _getn(_Out_writes_ (count) _CharType *ptr, _In_ size_t count) { return m_readOps.enqueue_operation([=] ()-> pplx::task{ if ( m_info->m_atend || count == 0 ) return pplx::task_from_result(0); if ( _in_avail_unprot() >= count ) { pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); // Check again once the lock is held. if ( _in_avail_unprot() >= count ) { auto bufoff = m_info->m_rdpos - m_info->m_bufoff; std::memcpy((void *)ptr, this->m_info->m_buffer+bufoff*sizeof(_CharType), count*sizeof(_CharType)); m_info->m_rdpos += count; return pplx::task_from_result(count); } } auto result_tce = pplx::task_completion_event(); auto callback = new _filestream_callback_read(m_info, result_tce); size_t read = _getn_fsb(m_info, callback, ptr, count, sizeof(_CharType)); if ( read != 0 && read != -1) { delete callback; pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); m_info->m_rdpos += read/sizeof(_CharType); return pplx::task_from_result(read/sizeof(_CharType)); } return pplx::create_task(result_tce); }); } /// /// Reads up to a given number of characters from the stream. /// /// The address of the target memory area /// The maximum number of characters to read /// The number of characters read. O if the end of the stream is reached or an asynchronous read is required. /// This is a synchronous operation, but is guaranteed to never block. size_t _sgetn(_Out_writes_ (count) _CharType *ptr, _In_ size_t count) { m_readOps.wait(); if ( m_info->m_atend ) return 0; if ( count == 0 || in_avail() == 0 ) return 0; pplx::extensibility::scoped_recursive_lock_t lck(m_info->m_lock); size_t available = _in_avail_unprot(); size_t copy = (count < available) ? count : available; auto bufoff = m_info->m_rdpos - m_info->m_bufoff; std::memcpy((void *)ptr, this->m_info->m_buffer+bufoff*sizeof(_CharType), copy*sizeof(_CharType)); m_info->m_rdpos += copy; m_info->m_atend = (copy < count); return copy; } /// /// Copies up to a given number of characters from the stream. /// /// The address of the target memory area /// The maximum number of characters to copy /// The number of characters copied. O if the end of the stream is reached or an asynchronous read is required. /// This is a synchronous operation, but is guaranteed to never block. virtual size_t _scopy(_CharType *, size_t) { return 0; } /// /// Gets the current read or write position in the stream. /// /// The I/O direction to seek (see remarks) /// The current position. EOF if the operation fails. /// Some streams may have separate write and read cursors. /// For such streams, the direction parameter defines whether to move the read or the write cursor. virtual pos_type getpos(std::ios_base::openmode mode) const { return const_cast(this)->seekoff(0, std::ios_base::cur, mode); } /// /// Seeks to the given position. /// /// The offset from the beginning of the stream /// The I/O direction to seek (see remarks) /// The position. EOF if the operation fails. /// Some streams may have separate write and read cursors. /// For such streams, the direction parameter defines whether to move the read or the write cursor. virtual pos_type seekpos(pos_type pos, std::ios_base::openmode mode) { if ( mode == std::ios_base::in ) { m_readOps.wait(); return (pos_type)_seekrdpos_fsb(m_info, size_t(pos), sizeof(_CharType)); } else if ( (m_info->m_mode & std::ios::ios_base::app) == 0 ) { return (pos_type)_seekwrpos_fsb(m_info, size_t(pos), sizeof(_CharType)); } return (pos_type)Concurrency::streams::char_traits<_CharType>::eof(); } /// /// Seeks to a position given by a relative offset. /// /// The relative position to seek to /// The starting point (beginning, end, current) for the seek. /// The I/O direction to seek (see remarks) /// The position. EOF if the operation fails. /// Some streams may have separate write and read cursors. /// For such streams, the mode parameter defines whether to move the read or the write cursor. virtual pos_type seekoff(off_type offset, std::ios_base::seekdir way, std::ios_base::openmode mode) { if ( mode == std::ios_base::in ) { m_readOps.wait(); switch ( way ) { case std::ios_base::beg: return (pos_type)_seekrdpos_fsb(m_info, size_t(offset), sizeof(_CharType)); case std::ios_base::cur: return (pos_type)_seekrdpos_fsb(m_info, size_t(m_info->m_rdpos+offset), sizeof(_CharType)); case std::ios_base::end: return (pos_type)_seekrdtoend_fsb(m_info, int64_t(offset), sizeof(_CharType)); default: // Fail on invalid input (_S_ios_seekdir_end) assert(false); } } else if ( (m_info->m_mode & std::ios::ios_base::app) == 0 ) { switch ( way ) { case std::ios_base::beg: return (pos_type)_seekwrpos_fsb(m_info, size_t(offset), sizeof(_CharType)); case std::ios_base::cur: return (pos_type)_seekwrpos_fsb(m_info, size_t(m_info->m_wrpos+offset), sizeof(_CharType)); case std::ios_base::end: return (pos_type)_seekwrpos_fsb(m_info, size_t(-1), sizeof(_CharType)); default: // Fail on invalid input (_S_ios_seekdir_end) assert(false); } } return (pos_type)traits::eof(); } /// /// For output streams, flush any internally buffered data to the underlying medium. /// virtual pplx::task _sync() { return flush_internal().then([](){return true;}); } private: template friend class ::concurrency::streams::file_buffer; pplx::task flush_internal() { pplx::task_completion_event result_tce; auto callback = utility::details::make_unique<_filestream_callback_write_b>(m_info, result_tce); if ( !_sync_fsb(m_info, callback.get()) ) { return pplx::task_from_exception(std::runtime_error("failure to flush stream")); } callback.release(); return pplx::create_task(result_tce); } basic_file_buffer(_In_ _file_info *info) : streambuf_state_manager<_CharType>(info->m_mode), m_info(info) { } #if !defined(__cplusplus_winrt) static pplx::task>> open( const utility::string_t &_Filename, std::ios_base::openmode _Mode = std::ios_base::out, #ifdef _WIN32 int _Prot = (int)std::ios_base::_Openprot #else int _Prot = 0 // unsupported on Linux, for now #endif ) { auto result_tce = pplx::task_completion_event>>(); auto callback = new _filestream_callback_open(result_tce); _open_fsb_str(callback, _Filename.c_str(), _Mode, _Prot); return pplx::create_task(result_tce); } #else static pplx::task>> open( ::Windows::Storage::StorageFile^ file, std::ios_base::openmode _Mode = std::ios_base::out) { auto result_tce = pplx::task_completion_event>>(); auto callback = new _filestream_callback_open(result_tce); _open_fsb_stf_str(callback, file, _Mode, 0); return pplx::create_task(result_tce); } #endif class _filestream_callback_open : public details::_filestream_callback { public: _filestream_callback_open(const pplx::task_completion_event>> &op) : m_op(op) { } virtual void on_opened(_In_ _file_info *info) { m_op.set(std::shared_ptr>(new basic_file_buffer<_CharType>(info))); delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: pplx::task_completion_event>> m_op; }; class _filestream_callback_close : public details::_filestream_callback { public: _filestream_callback_close(const pplx::task_completion_event &op) : m_op(op) { } virtual void on_closed() { m_op.set(); delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: pplx::task_completion_event m_op; }; template class _filestream_callback_write : public details::_filestream_callback { public: _filestream_callback_write(_In_ _file_info *info, const pplx::task_completion_event &op) : m_info(info), m_op(op) { } virtual void on_completed(size_t result) { m_op.set((ResultType)result / sizeof(_CharType)); delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: _file_info *m_info; pplx::task_completion_event m_op; }; class _filestream_callback_write_b : public details::_filestream_callback { public: _filestream_callback_write_b(_In_ _file_info *info, const pplx::task_completion_event &op) : m_info(info), m_op(op) { } virtual void on_completed(size_t) { m_op.set(); delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: _file_info *m_info; pplx::task_completion_event m_op; }; class _filestream_callback_read : public details::_filestream_callback { public: _filestream_callback_read(_In_ _file_info *info, const pplx::task_completion_event &op) : m_info(info), m_op(op) { } virtual void on_completed(size_t result) { result = result / sizeof(_CharType); m_info->m_rdpos += result; m_op.set(result); delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: _file_info *m_info; pplx::task_completion_event m_op; }; class _filestream_callback_bumpc : public details::_filestream_callback { public: _filestream_callback_bumpc(_In_ _file_info *info, const pplx::task_completion_event &op) : m_ch(0), m_info(info), m_op(op) { } virtual void on_completed(size_t result) { if ( result == sizeof(_CharType) ) { m_info->m_rdpos += 1; m_op.set(m_ch); } else { m_op.set(traits::eof()); } delete this; } virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } int_type m_ch; private: _file_info *m_info; pplx::task_completion_event m_op; }; class _filestream_callback_getc : public details::_filestream_callback { public: _filestream_callback_getc(_In_ _file_info *info, const pplx::task_completion_event &op) : m_ch(0), m_info(info), m_op(op) { } virtual void on_completed(size_t result) { if ( result == sizeof(_CharType) ) { m_op.set(m_ch); } else { m_op.set(traits::eof()); } delete this; } int_type m_ch; virtual void on_error(const std::exception_ptr & e) { m_op.set_exception(e); delete this; } private: _file_info *m_info; pplx::task_completion_event m_op; }; _file_info *m_info; async_operation_queue m_readOps; }; } // namespace details /// /// Stream buffer for file streams. /// /// /// The data type of the basic element of the file_buffer. /// template class file_buffer { public: #if !defined(__cplusplus_winrt) /// /// Open a new stream buffer representing the given file. /// /// The name of the file /// The opening mode of the file /// The file protection mode /// A task that returns an opened stream buffer on completion. static pplx::task> open( const utility::string_t &file_name, std::ios_base::openmode mode = std::ios_base::out, #ifdef _WIN32 int prot = _SH_DENYRD #else int prot = 0 // unsupported on Linux #endif ) { auto bfb = details::basic_file_buffer<_CharType>::open(file_name, mode, prot); return bfb.then([](pplx::task>> op) -> streambuf<_CharType> { return streambuf<_CharType>(op.get()); }); } #else /// /// Open a new stream buffer representing the given file. /// /// The StorageFile instance /// The opening mode of the file /// The file protection mode /// A task that returns an opened stream buffer on completion. static pplx::task> open( ::Windows::Storage::StorageFile^ file, std::ios_base::openmode mode = std::ios_base::out) { auto bfb = details::basic_file_buffer<_CharType>::open(file, mode); return bfb.then([](pplx::task>> op) -> streambuf<_CharType> { return streambuf<_CharType>(op.get()); }); } #endif }; /// /// File stream class containing factory functions for file streams. /// /// /// The data type of the basic element of the file_stream. /// template class file_stream { public: #if !defined(__cplusplus_winrt) /// /// Open a new input stream representing the given file. /// The file should already exist on disk, or an exception will be thrown. /// /// The name of the file /// The opening mode of the file /// The file protection mode /// A task that returns an opened input stream on completion. static pplx::task> open_istream( const utility::string_t &file_name, std::ios_base::openmode mode = std::ios_base::in, #ifdef _WIN32 int prot = (int) std::ios_base::_Openprot #else int prot = 0 #endif ) { mode |= std::ios_base::in; return streams::file_buffer<_CharType>::open(file_name, mode, prot) .then([](streams::streambuf<_CharType> buf) -> basic_istream<_CharType> { return basic_istream<_CharType>(buf); }); } /// /// Open a new ouput stream representing the given file. /// If the file does not exist, it will be create unless the folder or directory /// where it is to be found also does not exist. /// /// The name of the file /// The opening mode of the file /// The file protection mode /// A task that returns an opened output stream on completion. static pplx::task> open_ostream( const utility::string_t &file_name, std::ios_base::openmode mode = std::ios_base::out, #ifdef _WIN32 int prot = (int) std::ios_base::_Openprot #else int prot = 0 #endif ) { mode |= std::ios_base::out; return streams::file_buffer<_CharType>::open(file_name, mode, prot) .then([](streams::streambuf<_CharType> buf) -> basic_ostream<_CharType> { return basic_ostream<_CharType>(buf); }); } #else /// /// Open a new input stream representing the given file. /// The file should already exist on disk, or an exception will be thrown. /// /// The StorageFile reference representing the file /// The opening mode of the file /// A task that returns an opened input stream on completion. static pplx::task> open_istream( ::Windows::Storage::StorageFile^ file, std::ios_base::openmode mode = std::ios_base::in) { mode |= std::ios_base::in; return streams::file_buffer<_CharType>::open(file, mode) .then([](streams::streambuf<_CharType> buf) -> basic_istream<_CharType> { return basic_istream<_CharType>(buf); }); } /// /// Open a new ouput stream representing the given file. /// If the file does not exist, it will be create unless the folder or directory /// where it is to be found also does not exist. /// /// The StorageFile reference representing the file /// The opening mode of the file /// A task that returns an opened output stream on completion. static pplx::task> open_ostream( ::Windows::Storage::StorageFile^ file, std::ios_base::openmode mode = std::ios_base::out) { mode |= std::ios_base::out; return streams::file_buffer<_CharType>::open(file, mode) .then([](streams::streambuf<_CharType> buf) -> basic_ostream<_CharType> { return basic_ostream<_CharType>(buf); }); } #endif }; typedef file_stream fstream; }} // namespace concurrency::streams #endif