summaryrefslogtreecommitdiff
path: root/ace
diff options
context:
space:
mode:
Diffstat (limited to 'ace')
-rw-r--r--ace/IOStream.cpp406
-rw-r--r--ace/IOStream.h529
2 files changed, 713 insertions, 222 deletions
diff --git a/ace/IOStream.cpp b/ace/IOStream.cpp
index e9e8384b917..79e28154d74 100644
--- a/ace/IOStream.cpp
+++ b/ace/IOStream.cpp
@@ -1,10 +1,15 @@
// IOStream.cpp
// $Id$
-#if !defined (ACE_IOSTREAM_C)
+
+#ifndef ACE_IOSTREAM_C
#define ACE_IOSTREAM_C
#define ACE_BUILD_DLL
#include "ace/IOStream.h"
+#include "ace/Thread.h"
+#include "ace/Handle_Set.h"
+
+///////////////////////////////////////////////////////////////////////////
/* Here's a simple example of how iostream's non-virtual operators can
get you in a mess:
@@ -79,8 +84,28 @@
// function will be invoked by the first >>. Since it returns
// a myiostream&, the second >> will be invoked as desired. */
-template <class STREAM> int
-ACE_Streambuf<STREAM>::underflow (void)
+
+///////////////////////////////////////////////////////////////////////////
+
+#if ! defined( ACE_IOSTREAM_BUILDING_TEMPLATE )
+
+ACE_Time_Value *
+ACE_Streambuf_T::recv_timeout( ACE_Time_Value * tv )
+{
+ ACE_Time_Value * rval = recv_timeout_;
+ if( tv )
+ {
+ recv_timeout_value_ = *tv;
+ recv_timeout_ = &recv_timeout_value_;
+ }
+ else
+ recv_timeout_ = NULL;
+
+ return rval;
+}
+
+int
+ACE_Streambuf_T::underflow (void)
{
// If input mode is not set, any attempt to read from the stream is
// a failure.
@@ -102,7 +127,7 @@ ACE_Streambuf<STREAM>::underflow (void)
// explicitly handle deletion of the TWO buffers at destruction.
//
setb (this->eback_saved_,
- this->eback_saved_ + ACE_STREAMBUF_SIZE, 0);
+ this->eback_saved_ + streambuf_size_, 0);
// Using the new values for base (), initialize the get area.
// This simply sets eback (), gptr () and egptr () described
@@ -151,7 +176,7 @@ ACE_Streambuf<STREAM>::underflow (void)
// to use our private get buffer.
setb (this->eback_saved_,
- this->eback_saved_ + ACE_STREAMBUF_SIZE, 0);
+ this->eback_saved_ + streambuf_size_, 0);
// And restore the previous state of the get pointers.
@@ -198,8 +223,8 @@ ACE_Streambuf<STREAM>::underflow (void)
// Much of this is similar to underflow. I'll just hit the highlights
// rather than repeating a lot of what you've already seen.
-template <class STREAM> int
-ACE_Streambuf<STREAM>::overflow (int c)
+int
+ACE_Streambuf_T::overflow (int c)
{
// Check to see if output is allowed at all.
if (!(mode_ & ios::out))
@@ -211,7 +236,7 @@ ACE_Streambuf<STREAM>::overflow (int c)
// Set base () to use put's private buffer.
//
setb (this->pbase_saved_,
- this->pbase_saved_ + ACE_STREAMBUF_SIZE, 0);
+ this->pbase_saved_ + streambuf_size_, 0);
// Set the put area using the new base () values.
setp (base (), ebuf ());
@@ -236,7 +261,7 @@ ACE_Streambuf<STREAM>::overflow (int c)
setg (0, 0, 0);
// Reconfigure base () and restore the put pointers.
- setb (pbase_saved_, pbase_saved_ + ACE_STREAMBUF_SIZE, 0);
+ setb (pbase_saved_, pbase_saved_ + streambuf_size_, 0);
setp (base (), ebuf ());
// Save the new mode.
@@ -246,8 +271,7 @@ ACE_Streambuf<STREAM>::overflow (int c)
// If there is output to be flushed, do so now. We shouldn't
// get here unless this is the case...
- if (out_waiting ()
- && EOF == syncout ())
+ if (out_waiting () && EOF == syncout ())
return EOF;
}
@@ -269,8 +293,8 @@ ACE_Streambuf<STREAM>::overflow (int c)
// syncin
-template <class STREAM> int
-ACE_Streambuf<STREAM>::syncin (void)
+int
+ACE_Streambuf_T::syncin (void)
{
// As discussed, there really isn't any way to sync input from a socket-like
// device. We specifially override this base-class function so that it won't
@@ -278,12 +302,10 @@ ACE_Streambuf<STREAM>::syncin (void)
return 0;
}
-///// ///// ///// ///// ///// ///// ///// ///// ///// ///// ///// ///// ///// ///// /////
-
// syncout
-template <class STREAM> int
-ACE_Streambuf<STREAM>::syncout (void)
+int
+ACE_Streambuf_T::syncout (void)
{
// Unlike syncin, syncout is a doable thing. All we have to do is
// write whatever is in the output buffer to the peer. flushbuf ()
@@ -295,8 +317,8 @@ ACE_Streambuf<STREAM>::syncout (void)
return 0;
}
-template <class STREAM> int
-ACE_Streambuf<STREAM>::sync (void)
+int
+ACE_Streambuf_T::sync (void)
{
// sync () is fairly traditional in that it syncs both input and output.
// We could have omitted the call to syncin () but someday, we may want it
@@ -314,23 +336,59 @@ ACE_Streambuf<STREAM>::sync (void)
// flushbuf
-template <class STREAM> int
-ACE_Streambuf<STREAM>::flushbuf (void)
+int
+ACE_Streambuf_T::flushbuf (void)
{
// pptr() is one character beyond the last character put
// into the buffer. pbase() points to the beginning of
// the put buffer. Unless pptr() is greater than pbase()
- // there is nothing to be sent to the peer_.
+ // there is nothing to be sent to the peer.
//
if( pptr() <= pbase() )
return 0;
+ // 4/12/97 -- JCEJ
+ // Kludge!!!
+ // If the remote side shuts down the connection, an attempt to
+ // send() to the remote will result in the message 'Broken Pipe'
+ // I think this is an OS message, I've tracked it down to the
+ // ACE_OS::write () function. That's the last one to be called
+ // before the message. I can only test this on Linux though, so
+ // I don't know how other systems will react.
+ //
+ // To get around this gracefully, I do a PEEK recv() with an
+ // immediate (nearly) timeout. recv() is much more graceful
+ // on it's failure. If we get -1 from recv() not due to timeout
+ // then we know we're SOL.
+ //
+ // Q: Is 'errno' threadsafe? Should the section below be a
+ // critical section?
+ //
+ //
+ // char tbuf[1];
+ // ACE_Time_Value to(0,1);
+ // if( this->recv( tbuf, 1, MSG_PEEK, &to ) == -1 )
+ // {
+ // if( errno != ETIME )
+ // {
+ // perror("OOPS preparing to send to peer");
+ // return EOF;
+ // }
+ // }
+ //
+ // The correct way to handle this is for the application to
+ // trap (and ignore?) SIGPIPE. Thanks to Amos Shapira
+ // for reminding me of this.
+ //
+
// Starting at the beginning of the buffer, send as much
- // data as there is waiting. send_n guarantees that all
+ // data as there is waiting. send guarantees that all
// of the data will be sent or an error will be returned.
//
- if( peer_->send_n( pbase(), pptr() - pbase() ) == -1 )
+ if( this->send( pbase(), pptr() - pbase() ) == -1 )
+ {
return EOF;
+ }
// Now that we've sent everything in the output buffer, we reset the
// buffer pointers to appear empty.
@@ -340,86 +398,104 @@ ACE_Streambuf<STREAM>::flushbuf (void)
return 0;
}
-template <class STREAM> int
-ACE_Streambuf<STREAM>::get_one_byte (void)
+int
+ACE_Streambuf_T::get_one_byte ( void )
{
- char * p = base ();
- ssize_t i = peer_->recv_n (p, 1);
-
// The recv function will return immediately if there is no data
// waiting. So, we use recv_n to wait for exactly one byte to come
// from the peer. Later, we can use recv to see if there is
// anything else in the buffer. (Ok, we could use flags to tell it
// to block but I like this better.)
- if (i != 1)
+ if( this->recv_n( base(), 1, MSG_PEEK, recv_timeout_ ) != 1 )
return EOF;
- // Set the get pointers to indicate that we've got new data.
- setg (base (), base (), base () + 1);
-
return 1;
}
-template <class STREAM> int
-ACE_Streambuf<STREAM>::fillbuf (void)
+int
+ACE_Streambuf_T::fillbuf (void)
+ // This will be called when the read (get) buffer has been
+ // exhausted (ie -- gptr == egptr)
{
- char *p;
-
- // Invoke recv_n to get at least one byte from the remote. This
+ // Invoke recv_n to get exactly one byte from the remote. This
// will block until something shows up.
-
- if (this->get_one_byte () == EOF)
+ //
+ if( get_one_byte() == EOF )
return EOF;
// Now, get whatever else may be in the buffer. This will return if
- // there is nothing in the buffer. Notice that we can only read
- // blen ()-1 bytes since we've already read one via <get_one_byte>
+ // there is nothing in the buffer.
- p = base () + 1;
- ssize_t t = peer_->recv (p, blen () - 1);
+ int bc = this->recv ( base(), blen(), recv_timeout_ );
// recv will give us -1 if there was a problem. If there was
// nothing waiting to be read, it will give us 0. That isn't an
// error.
- if (t++ < 0)
+ if (bc < 0)
return EOF;
// Move the get pointer to reflect the number of bytes we just read.
- setg (base (), base (), base () + t);
+ setg (base (), base (), base () + bc );
// Return the byte-read-count including the one from <get_one_byte>
- return t;
+ return bc;
}
-// We will be given a STREAM by the iostream object which creates us.
-// See the ACE_IOStream template for how that works. Like other
-// streambuf objects, we can be input-only, output-only or both.
-
-template <class STREAM>
-ACE_Streambuf<STREAM>::ACE_Streambuf (STREAM *peer, int io_mode)
- : peer_ (peer),
- get_mode_ (1),
+ACE_Streambuf_T::ACE_Streambuf_T (u_int streambuf_size, int io_mode)
+ : get_mode_ (1),
put_mode_ (2),
- mode_ (io_mode)
+ mode_ (io_mode),
+ streambuf_size_ (streambuf_size),
+ recv_timeout_(NULL)
{
- // A streambuf allows for unbuffered IO where every character is
- // read as requested and written as provided. To me, this seems
- // terribly inefficient for socket-type operations, so I've disabled
- // it. All of the work would be done by the underflow/overflow
- // functions anyway and I haven't implemented anything there to
- // support unbuffered IO.
+ (void)reset_get_buffer();
+ (void)reset_put_buffer();
+}
- this->unbuffered (0);
+u_int ACE_Streambuf_T::streambuf_size(void)
+{
+ return streambuf_size_;
+}
- // Linebuffered is similar to unbuffered. Again, I don't have any
- // need for this and I don't see the advantage. I believe this
- // would have to be supported by underflow/overflow to be effective.
-#if !defined (ACE_LACKS_LINEBUFFERED_STREAMBUF)
- this->linebuffered (0);
-#endif /* ! ACE_LACKS_LINEBUFFERED_STREAMBUF */
+u_int ACE_Streambuf_T::get_waiting(void)
+ // Return the number of bytes not yet gotten.
+ // eback + get_waiting = gptr
+{
+ return this->gptr_saved_ - this->eback_saved_;
+}
+
+u_int ACE_Streambuf_T::get_avail(void)
+ // Return the number of bytes in the get area (includes some already gotten);
+ // eback + get_avail = egptr
+{
+ return this->egptr_saved_ - this->eback_saved_;
+}
+
+u_int ACE_Streambuf_T::put_avail(void)
+ // Return the number of bytes to be 'put' onto the stream media.
+ // pbase + put_avail = pptr
+{
+ return this->pptr_saved_ - this->pbase_saved_;
+}
+
+char *
+ACE_Streambuf_T::reset_get_buffer(char * newBuffer, u_int _streambuf_size, u_int _gptr, u_int _egptr )
+//
+// Typical usage:
+//
+// u_int newGptr = otherStream->get_waiting();
+// u_int newEgptr = otherStream->get_avail();
+// char * newBuf = otherStream->reset_get_buffer();
+// char * oldgetbuf = myStream->reset_get_buffer( newBuf, otherStream->streambuf_size(), newGptr, newEgptr );
+//
+// 'myStream' now has the get buffer of 'otherStream' and can use it in any way.
+// 'otherStream' now has a new, empty get buffer.
+//
+{
+ char * rval = this->eback_saved_;
// The get area is where the iostrem will get data from. This is
// our read buffer. There are three pointers which describe the
@@ -437,9 +513,40 @@ ACE_Streambuf<STREAM>::ACE_Streambuf (STREAM *peer, int io_mode)
// the variables below. Initially, they all point to the beginning
// of our read-dedicated buffer.
//
- ACE_NEW (this->eback_saved_, char[ACE_STREAMBUF_SIZE]);
- this->gptr_saved_ = this->eback_saved_;
- this->egptr_saved_ = this->eback_saved_;
+ if( newBuffer )
+ {
+ if( streambuf_size_ != _streambuf_size )
+ return NULL;
+ this->eback_saved_ = newBuffer;
+ }
+ else
+ {
+ ACE_NEW_RETURN (this->eback_saved_, char[streambuf_size_], 0);
+ }
+
+ this->gptr_saved_ = this->eback_saved_ + _gptr;
+ this->egptr_saved_ = this->eback_saved_ + _egptr;
+
+ // Disable the get area initially. This will cause underflow to be
+ // invoked on the first get operation.
+ setg (0, 0, 0);
+
+ reset_base();
+
+ return rval;
+}
+
+char *
+ACE_Streambuf_T::reset_put_buffer(char * newBuffer, u_int _streambuf_size, u_int _pptr )
+//
+// Typical usage:
+//
+// u_int newPptr = otherStream->put_avail();
+// char * newBuf = otherStream->reset_put_buffer();
+// char * oldputbuf = otherStream->reset_put_buffer( newBuf, otherStream->streambuf_size(), newPptr );
+//
+{
+ char * rval = this->pbase_saved_;
// The put area is where the iostream will put data that needs to be
// sent to the peer. This becomes our write buffer. The three
@@ -454,18 +561,32 @@ ACE_Streambuf<STREAM>::ACE_Streambuf (STREAM *peer, int io_mode)
// Again to switch quickly between modes, we keep copies of
// these three pointers.
//
- ACE_NEW (this->pbase_saved_, char[ACE_STREAMBUF_SIZE]);
- this->pptr_saved_ = this->pbase_saved_;
- this->epptr_saved_ = this->pbase_saved_ + ACE_STREAMBUF_SIZE;
+ if( newBuffer )
+ {
+ if( streambuf_size_ != _streambuf_size )
+ return NULL;
+ this->pbase_saved_ = newBuffer;
+ }
+ else
+ {
+ ACE_NEW_RETURN (this->pbase_saved_, char[streambuf_size_], 0);
+ }
- // Disable the get area initially. This will cause underflow to be
- // invoked on the first get operation.
- setg (0, 0, 0);
+ this->pptr_saved_ = this->pbase_saved_ + _pptr;
+ this->epptr_saved_ = this->pbase_saved_ + streambuf_size_;
// Disable the put area. Overflow will be called by the first call
// to any put operator.
setp (0, 0);
+ reset_base();
+
+ return rval;
+}
+
+void
+ACE_Streambuf_T::reset_base(void)
+{
// Until we experience the first get or put operation, we do not
// know what our current IO mode is.
this->cur_mode_ = 0;
@@ -482,20 +603,61 @@ ACE_Streambuf<STREAM>::ACE_Streambuf (STREAM *peer, int io_mode)
// would be deleted when the object destructs. Since we are providing
// separate read/write buffers, it is up to us to manage their memory.
-template <class STREAM>
-ACE_Streambuf<STREAM>::~ACE_Streambuf (void)
+ACE_Streambuf_T::~ACE_Streambuf_T (void)
{
delete [] this->eback_saved_;
delete [] this->pbase_saved_;
}
+#endif // ACE_IOSTREAM_BUILDING_TEMPLATE
+
+///////////////////////////////////////////////////////////////////////////
+
+
+// We will be given a STREAM by the iostream object which creates us.
+// See the ACE_IOStream template for how that works. Like other
+// streambuf objects, we can be input-only, output-only or both.
+
+template <class STREAM>
+ACE_Streambuf<STREAM>::ACE_Streambuf (STREAM *peer, u_int streambuf_size, int io_mode )
+ : ACE_Streambuf_T( streambuf_size, io_mode ), peer_ (peer)
+{
+ // A streambuf allows for unbuffered IO where every character is
+ // read as requested and written as provided. To me, this seems
+ // terribly inefficient for socket-type operations, so I've disabled
+ // it. All of the work would be done by the underflow/overflow
+ // functions anyway and I haven't implemented anything there to
+ // support unbuffered IO.
+
+ this->unbuffered (0);
+
+ // Linebuffered is similar to unbuffered. Again, I don't have any
+ // need for this and I don't see the advantage. I believe this
+ // would have to be supported by underflow/overflow to be effective.
+#if !defined (ACE_LACKS_LINEBUFFERED_STREAMBUF)
+ this->linebuffered (0);
+#endif /* ! ACE_LACKS_LINEBUFFERED_STREAMBUF */
+
+}
+
+
+///////////////////////////////////////////////////////////////////////////
+
+
// The typical constructor. This will initiailze your STREAM and then
// setup the iostream baseclass to use a custom streambuf based on
// STREAM.
template <class STREAM>
-ACE_IOStream<STREAM>::ACE_IOStream (void)
- : iostream (streambuf_ = new ACE_Streambuf<STREAM> ((STREAM *) this))
+ACE_IOStream<STREAM>::ACE_IOStream ( STREAM & stream, u_int streambuf_size )
+ : iostream(streambuf_ = new ACE_Streambuf<STREAM> ((STREAM *) this, streambuf_size)),
+ STREAM(stream)
+{
+}
+
+template <class STREAM>
+ACE_IOStream<STREAM>::ACE_IOStream ( u_int streambuf_size )
+ : iostream(streambuf_ = new ACE_Streambuf<STREAM> ((STREAM *) this, streambuf_size))
{
}
@@ -517,19 +679,54 @@ ACE_IOStream<STREAM>::close (void)
return STREAM::close ();
}
-#if defined (__GNUC__) && !defined (CHORUS)
+template <class STREAM> ACE_IOStream<STREAM> &
+ACE_IOStream<STREAM>::operator>>(ACE_Time_Value *& tv )
+{
+ ACE_Time_Value * old_tv = this->streambuf_->recv_timeout(tv);
+ tv = old_tv;
+ return *this;
+}
+
+#if (defined(__GNUC__) && !defined (CHORUS)) || defined(ACE_WIN32)
+
// A simple string operator. The base iostream has 'em for char* but
// that isn't always the best thing for a String. If we don't provide
// our own here, we may not get what we want.
template <class STREAM> ACE_IOStream<STREAM> &
-ACE_IOStream<STREAM>::operator>> (String & v)
+ACE_IOStream<STREAM>::operator>> (ACE_IOStream_String & v)
+{
+ if( ipfx0() )
+ {
+ char c;
+ iostream::operator>> (c);
+
+ for (v = c ; this->get (c) && !isspace (c) ; v += c)
+ continue;
+ }
+
+ isfx();
+
+ return *this;
+}
+
+
+template <class STREAM> ACE_IOStream<STREAM> &
+ACE_IOStream<STREAM>::operator<< (ACE_IOStream_String & v)
{
- char c;
- iostream::operator>> (c);
+ if( opfx() )
+ {
+#if defined (ACE_WIN32)
+ for( int i = 0 ; i < v.GetLength() ; ++i )
+#else
+ for( u_int i = 0 ; i < (u_int)v.length() ; ++i )
+#endif
+ {
+ this->put( v[i] );
+ }
+ }
- for (v = c ; this->get (c) && !isspace (c) ; v += c)
- continue;
+ osfx();
return *this;
}
@@ -542,6 +739,8 @@ ACE_IOStream<STREAM>::operator>> (String & v)
template <class STREAM> ACE_IOStream<STREAM> &
ACE_IOStream<STREAM>::operator>> (QuotedString & str)
{
+ if( ipfx0() )
+ {
char c;
if( !(*this >> c) ) // eat space up to the first char
@@ -569,18 +768,37 @@ ACE_IOStream<STREAM>::operator>> (QuotedString & str)
}
}
}
+ }
+
+ isfx();
- return *this;
+ return *this;
}
template <class STREAM> ACE_IOStream<STREAM> &
ACE_IOStream<STREAM>::operator<< (QuotedString & str)
{
- *this << '"';
- *this << (String&)str;
- *this << '"';
- return *this;
+ if( opfx() )
+ {
+ this->put('"');
+ for( u_int i = 0 ; i < str.length() ; ++i )
+ {
+ if( str[i] == '"' )
+ this->put('\\');
+ this->put( str[i] );
+ }
+ this->put('"');
+ }
+
+ osfx();
+
+ return *this;
}
-#endif /* __GNUG__ */
+
+///////////////////////////////////////////////////////////////////////////
+
+
+#endif /* (__GNUC__ && ! CHORUS) || ACE_WIN32 */
#endif /* ACE_IOSTREAM_C */
+
diff --git a/ace/IOStream.h b/ace/IOStream.h
index 45d08db7830..86865241ef6 100644
--- a/ace/IOStream.h
+++ b/ace/IOStream.h
@@ -17,30 +17,123 @@
//
// ============================================================================
+/*
+ Changes 4/5/97
+
+ ACE_Streambuf_T
+
+ New public functions added
+
+ char * reset_get_buffer( char * newbuf, u_int bufsiz, u_int inuse )
+ char * reset_put_buffer( char * newbuf, u_int bufsiz, u_int inuse )
+
+ Resets the get/put buffer and returns a pointer to the
+ previous buffer.
+
+ If newbuf is not NULL and bufsiz matches the current
+ streambuf_size_ value, then the current buffer will
+ be replaced by newbuf. If the stream sizes do not match,
+ a NULL pointer is returned and the buffer left unchanged.
+
+ If newbuf is NULL, a new buffer will be allocated from
+ the heap.
+
+ This kind of thing is handy if you're chaining streams.
+ For instance, I have a stream built on a message queue
+ which marshals and sends data from one thread to another.
+ The receiving thread then sends that data to a peer()
+ via a stream built on a socket. It is more efficient
+ to simply replace the socket stream's put buffer with that
+ of the message queue rather than copying the data from
+ one to another.
+
+ u_int streambuf_size( void )
+
+ Get the size of the stream buffers. Note that the get
+ and put buffers are the same size.
+
+ New protected functions added
+
+ void reset_base( void )
+
+ Resets the base() pointers which toggle between get
+ and put mode as well as the cur_mode_ value.
+
+ Constructor changed
+
+ Initialization of the get/put buffers was moved here from
+ the derived class/template ACE_Streambuf<...>
+
+ At the same time, the new functions reset_get/put_buffer
+ are now used instead of the previous monolithic function.
+
+ ACE_Streambuf<...>
+
+ Constructor changed:
+
+ Initialization of the get/put buffers has been moved to the
+ baseclass ACE_Streambuf_T. This will reduce template bloat
+ as well as giving us more functionality as described above.
+ */
+
#if !defined (ACE_IOSTREAM_H)
#define ACE_IOSTREAM_H
#include "ace/OS.h"
+#include "ace/INET_Addr.h"
#include <iomanip.h>
#if defined (__GNUC__) && !defined (CHORUS)
#include <String.h>
+typedef String ACE_IOStream_String;
+#endif /* __GNUC__ */
+//
+#if defined (ACE_WIN32)
+typedef CString ACE_IOStream_String;
+#endif /* ACE_WIN32 */
-class QuotedString : public String
+#if defined(__GNUC__) || defined(ACE_WIN32)
+
+class QuotedString : public ACE_IOStream_String
{
public:
- inline QuotedString & operator =(char c) { return (QuotedString&) String::operator=(c); }
- inline QuotedString & operator =(char* c) { return (QuotedString&) String::operator=(c); }
+ inline QuotedString() {
+ *this = "";
+ }
+ inline QuotedString(const char * c) {
+ *this = ACE_IOStream_String(c);
+ }
+ inline QuotedString(const ACE_IOStream_String& s) {
+ *this = s;
+ }
+ inline QuotedString& operator=(const ACE_IOStream_String& s) {
+ return (QuotedString&) ACE_IOStream_String::operator=(s);
+ }
+ inline QuotedString & operator =(const char c) {
+ return (QuotedString&) ACE_IOStream_String::operator=(c);
+ }
+ inline QuotedString & operator =(const char* c) {
+ return (QuotedString&) ACE_IOStream_String::operator=(c);
+ }
+ inline bool operator < (const QuotedString& s) const {
+ return *(ACE_IOStream_String *)this < (ACE_IOStream_String)s;
+ }
+#if defined (ACE_WIN32)
+ inline int length(void) {
+ return this->GetLength();
+ }
+#endif
};
-#endif /* __GNUC__ */
-template <class STREAM>
-class ACE_Streambuf : public streambuf
+#endif /* __GNUC__ || ACE_WIN32 */
+
+///////////////////////////////////////////////////////////////////////////
+
+class ACE_Export ACE_Streambuf_T : public streambuf
// = TITLE
// Create your custom streambuf by providing and ACE_*_Stream
// object to this template. I have tested it with
// ACE_SOCK_Stream and it should work fine for others as well.
- // I'm hoping to add functionality for ACE_SOCK_Dgram soon...
//
// = DESCRIPTION
// For any iostream object, the real work is done by the
@@ -99,19 +192,51 @@ class ACE_Streambuf : public streambuf
// "syncing" the input. It simply remains buffered.
{
public:
- ACE_Streambuf (STREAM * peer, int io_mode = ios::in | ios::out);
- // We will be given a STREAM by the iostream object which creates
- // us. See the ACE_IOStream template for how that works. Like
- // other streambuf objects, we can be input-only, output-only or
- // both.
- virtual ~ACE_Streambuf (void);
+ virtual ~ACE_Streambuf_T (void);
// If the default allocation strategey were used the common buffer
// would be deleted when the object destructs. Since we are
// providing separate read/write buffers, it is up to us to manage
// their memory.
+ ACE_Time_Value * recv_timeout( ACE_Time_Value * tv = NULL );
+ // Get the current Time_Value pointer and provide a new one.
+
+ char * reset_put_buffer(char * newBuffer = NULL, u_int _streambuf_size = 0, u_int _pptr = 0 );
+ // Use this to allocate a new/different buffer for put operations. If you
+ // do not provide a buffer pointer, one will be allocated. That is the
+ // preferred method. If you do provide a buffer, the size must match that
+ // being used by the get buffer.
+ // If successful, you will receive a pointer to the current put buffer.
+ // It is your responsibility to delete this memory when you are done with it.
+
+ u_int put_avail(void);
+ // Return the number of bytes to be 'put' onto the stream media.
+ // pbase + put_avail = pptr
+
+ char * reset_get_buffer(char * newBuffer = NULL, u_int _streambuf_size = 0, u_int _gptr = 0, u_int _egptr = 0 );
+ // Use this to allocate a new/different buffer for get operations. If you
+ // do not provide a buffer pointer, one will be allocated. That is the
+ // preferred method. If you do provide a buffer, the size must match that
+ // being used by the put buffer.
+ // If successful, you will receive a pointer to the current get buffer.
+ // It is your responsibility to delete this memory when you are done with it.
+
+ u_int get_waiting(void);
+ // Return the number of bytes not yet gotten.
+ // eback + get_waiting = gptr
+ //
+ u_int get_avail(void);
+ // Return the number of bytes in the get area (includes some already gotten);
+ // eback + get_avail = egptr
+ //
+
+ u_int streambuf_size(void);
+ // Query the streambuf for the size of it's buffers.
+
protected:
+ ACE_Streambuf_T (u_int streambuf_size, int io_mode);
+
virtual int sync (void);
// Sync both input and output. See syncin/syncout below for
// descriptions.
@@ -123,10 +248,11 @@ protected:
// The overflow function receives the character which caused the
// overflow.
-private:
- STREAM *peer_;
- // This will be our ACE_SOCK_Stream or similar object.
+ void reset_base(void);
+ // Resets the base() pointer and streambuf mode. This is
+ // used internally when get/put buffers are allocatd.
+protected:
// = Two pointer sets for manipulating the read/write areas.
char *eback_saved_;
char *gptr_saved_;
@@ -146,6 +272,15 @@ private:
// mode tells us if we're working for an istream, ostream, or
// iostream.
+ const u_int streambuf_size_;
+ // This defines the size of the input and output buffers. It can
+ // be set by the object constructor.
+
+ ACE_Time_Value recv_timeout_value_;
+ ACE_Time_Value * recv_timeout_;
+ // We want to allow the user to provide Time_Value pointers to
+ // prevent infinite blocking while waiting to receive data.
+
int syncin (void);
// syncin is called when the input needs to be synced with the
// source file. In a filebuf, this results in the seek() system
@@ -168,28 +303,68 @@ private:
// underflow. It will attempt to fill the read buffer from the
// peer.
- int get_one_byte (void);
+ virtual int get_one_byte ( );
// Used by fillbuf and others to get exactly one byte from the peer.
// recv_n is used to be sure we block until something is available.
+ // It is virtual because we really need to override it for
+ // datagram-derived objects.
+
+ virtual ssize_t send ( char * buf, ssize_t len ) = 0;
+ virtual ssize_t recv ( char * buf, ssize_t len, ACE_Time_Value * tv = NULL ) = 0;
+ virtual ssize_t recv ( char * buf, ssize_t len, int flags, ACE_Time_Value * tv = NULL ) = 0;
+ virtual ssize_t recv_n( char * buf, ssize_t len, int flags = 0, ACE_Time_Value * tv = NULL ) = 0;
+ // Stream connections and "unconnected connections" (ie -- datagrams)
+ // need to work just a little differently. We derive custom
+ // Streambuf objects for them and provide these functions at that time.
+
+ virtual ACE_HANDLE get_handle(void)
+ {
+ return 0;
+ }
};
-// This macro defines the get operator for class MT into datatype DT.
-// We will use it below to quickly override most (all?) iostream get
-// operators. Notice how the ipfx() and isfx() functions are used.
+///////////////////////////////////////////////////////////////////////////
-#define ACE_OPERATORG(MT,DT) \
- inline virtual MT& operator>> (DT v) { \
- if (ipfx ()) iostream::operator>> (v); isfx (); return *this; \
- }
+template <class STREAM>
+class ACE_Streambuf : public ACE_Streambuf_T
+{
+public:
+ ACE_Streambuf (STREAM * peer, u_int streambuf_size = ACE_STREAMBUF_SIZE, int io_mode = ios::in | ios::out);
+ // We will be given a STREAM by the iostream object which creates
+ // us. See the ACE_IOStream template for how that works. Like
+ // other streambuf objects, we can be input-only, output-only or
+ // both.
-// This macro defines the put operator for class MT into datatype DT.
-// We will use it below to quickly override most (all?) iostream put
-// operators. Notice how the opfx() and osfx() functions are used.
+ virtual ssize_t send ( char * buf, ssize_t len )
+ {
+ return peer_->send_n(buf,len);
+ }
+ virtual ssize_t recv ( char * buf, ssize_t len, ACE_Time_Value * tv = NULL )
+ {
+ return this->recv( buf, len, 0, tv );
+ }
+ virtual ssize_t recv ( char * buf, ssize_t len, int flags, ACE_Time_Value * tv = NULL )
+ {
+ ssize_t rval = peer_->recv(buf,len,flags,tv);
+ return rval;
+ }
+ virtual ssize_t recv_n( char * buf, ssize_t len, int flags = 0, ACE_Time_Value * tv = NULL )
+ {
+ ssize_t rval = peer_->recv_n(buf,len,flags,tv);
+ return rval;
+ }
+
+protected:
+ STREAM * peer_;
+ // This will be our ACE_SOCK_Stream or similar object.
+
+ virtual ACE_HANDLE get_handle(void)
+ {
+ return peer_ ? peer_->get_handle() : 0 ;
+ }
+};
-#define ACE_OPERATORP(MT,DT) \
- inline virtual MT& operator<< (DT v) { \
- if (opfx ()) iostream::operator<< (v); osfx (); return *this; \
- }
+///////////////////////////////////////////////////////////////////////////
// These typedefs are provided by G++ (on some systems?) without the
// trailing '_'. Since we can't count on 'em, I've defined them to
@@ -204,83 +379,95 @@ typedef ostream& (*__omanip_)(ostream&);
//
// virtual MT& operator<<(ios& (*func)(ios&)) { (*func)(*this); return *this; }
+
+
+// This macro defines the get operator for class MT into datatype DT.
+// We will use it below to quickly override most (all?) iostream get
+// operators. Notice how the ipfx() and isfx() functions are used.
+
+#define GET_SIG(MT,DT) inline virtual MT& operator>> (DT v)
+#define GET_CODE { if (ipfx ()) iostream::operator>> (v); isfx (); return *this; }
+#define GET_PROT(MT,DT,CODE) GET_SIG(MT,DT) CODE
+#define GET_FUNC(MT,DT) GET_PROT(MT,DT,GET_CODE)
+
+// This macro defines the put operator for class MT into datatype DT.
+// We will use it below to quickly override most (all?) iostream put
+// operators. Notice how the opfx() and osfx() functions are used.
+
+#define PUT_SIG(MT,DT) inline virtual MT& operator<< (DT v)
+#define PUT_CODE { if (opfx ()) iostream::operator<< (v); osfx (); return *this; }
+#define PUT_PROT(MT,DT,CODE) PUT_SIG(MT,DT) CODE
+#define PUT_FUNC(MT,DT) PUT_PROT(MT,DT,PUT_CODE)
+
+
// These are necessary in case somebody wants to derive from us and
// override one of these with a custom approach.
+#define GET_FUNC_SET0(MT,CODE,CODE2) \
+ GET_PROT(MT,short &,CODE) \
+ GET_PROT(MT,u_short &,CODE) \
+ GET_PROT(MT,int &,CODE) \
+ GET_PROT(MT,u_int &,CODE) \
+ GET_PROT(MT,long &,CODE) \
+ GET_PROT(MT,u_long &,CODE) \
+ GET_PROT(MT,float &,CODE) \
+ GET_PROT(MT,double &,CODE) \
+ GET_PROT(MT,long double &,CODE) \
+ GET_PROT(MT,char &,CODE) \
+ GET_PROT(MT,u_char &,CODE) \
+ GET_PROT(MT,char *,CODE) \
+ GET_PROT(MT,u_char *,CODE) \
+ inline virtual MT& operator>>(__omanip_ func) CODE2 \
+ inline virtual MT& operator>>(__manip_ func) CODE2
+
+#define PUT_FUNC_SET0(MT,CODE,CODE2) \
+ PUT_PROT(MT,short,CODE) \
+ PUT_PROT(MT,u_short,CODE) \
+ PUT_PROT(MT,int,CODE) \
+ PUT_PROT(MT,u_int,CODE) \
+ PUT_PROT(MT,long,CODE) \
+ PUT_PROT(MT,u_long,CODE) \
+ PUT_PROT(MT,float,CODE) \
+ PUT_PROT(MT,double,CODE) \
+ PUT_PROT(MT,char,CODE) \
+ PUT_PROT(MT,u_char,CODE) \
+ PUT_PROT(MT,const char *,CODE) \
+ PUT_PROT(MT,const u_char *,CODE) \
+ PUT_PROT(MT,const void *,CODE) \
+ inline virtual MT& operator<<(__omanip_ func) CODE2 \
+ inline virtual MT& operator<<(__manip_ func) CODE2
+
+
#if defined (ACE_LACKS_SIGNED_CHAR)
-#define ACE_OPERATORG_SET(MT) \
- ACE_OPERATORG(MT,short &); \
- ACE_OPERATORG(MT,u_short &); \
- ACE_OPERATORG(MT,int &); \
- ACE_OPERATORG(MT,u_int &); \
- ACE_OPERATORG(MT,long &); \
- ACE_OPERATORG(MT,u_long &); \
- ACE_OPERATORG(MT,float &); \
- ACE_OPERATORG(MT,double &); \
- ACE_OPERATORG(MT,long double &); \
- ACE_OPERATORG(MT,char &) \
- ACE_OPERATORG(MT,u_char &); \
- ACE_OPERATORG(MT,char *) \
- ACE_OPERATORG(MT,u_char *); \
- virtual MT& operator>>(__omanip_ func) { (*func)(*this); return *this; } \
- virtual MT& operator>>(__manip_ func) { (*func)(*this); return *this; }
-
-#define ACE_OPERATORP_SET(MT) \
- ACE_OPERATORP(MT,short); \
- ACE_OPERATORP(MT,u_short); \
- ACE_OPERATORP(MT,int); \
- ACE_OPERATORP(MT,u_int); \
- ACE_OPERATORP(MT,long); \
- ACE_OPERATORP(MT,u_long); \
- ACE_OPERATORP(MT,float); \
- ACE_OPERATORP(MT,double); \
- ACE_OPERATORP(MT,char); \
- ACE_OPERATORP(MT,u_char); \
- ACE_OPERATORP(MT,const char *); \
- ACE_OPERATORP(MT,const u_char *); \
- ACE_OPERATORP(MT,const void *); \
- virtual MT& operator<<(__omanip_ func) { (*func)(*this); return *this; } \
- virtual MT& operator<<(__manip_ func) { (*func)(*this); return *this; }
+ #define GET_FUNC_SET1(MT,CODE,CODE2) GET_FUNC_SET0(MT,CODE,CODE2)
+ #define PUT_FUNC_SET1(MT,CODE,CODE2) PUT_FUNC_SET0(MT,CODE,CODE2)
#else
-#define ACE_OPERATORG_SET(MT) \
- ACE_OPERATORG(MT,short &); \
- ACE_OPERATORG(MT,u_short &); \
- ACE_OPERATORG(MT,int &); \
- ACE_OPERATORG(MT,u_int &); \
- ACE_OPERATORG(MT,long &); \
- ACE_OPERATORG(MT,u_long &); \
- ACE_OPERATORG(MT,float &); \
- ACE_OPERATORG(MT,double &); \
- ACE_OPERATORG(MT,long double &); \
- ACE_OPERATORG(MT,char &) \
- ACE_OPERATORG(MT,u_char &); \
- ACE_OPERATORG(MT,signed char &); \
- ACE_OPERATORG(MT,char *) \
- ACE_OPERATORG(MT,u_char *); \
- ACE_OPERATORG(MT,signed char *); \
- virtual MT& operator>>(__omanip_ func) { (*func)(*this); return *this; } \
- virtual MT& operator>>(__manip_ func) { (*func)(*this); return *this; }
-
-#define ACE_OPERATORP_SET(MT) \
- ACE_OPERATORP(MT,short); \
- ACE_OPERATORP(MT,u_short); \
- ACE_OPERATORP(MT,int); \
- ACE_OPERATORP(MT,u_int); \
- ACE_OPERATORP(MT,long); \
- ACE_OPERATORP(MT,u_long); \
- ACE_OPERATORP(MT,float); \
- ACE_OPERATORP(MT,double); \
- ACE_OPERATORP(MT,char); \
- ACE_OPERATORP(MT,u_char); \
- ACE_OPERATORP(MT,signed char); \
- ACE_OPERATORP(MT,const char *); \
- ACE_OPERATORP(MT,const u_char *); \
- ACE_OPERATORP(MT,const signed char *); \
- ACE_OPERATORP(MT,const void *); \
- virtual MT& operator<<(__omanip_ func) { (*func)(*this); return *this; } \
- virtual MT& operator<<(__manip_ func) { (*func)(*this); return *this; }
+ #define GET_FUNC_SET1(MT,CODE,CODE2) \
+ GET_PROT(MT,signed char &,CODE) \
+ GET_PROT(MT,signed char *,CODE) \
+ GET_FUNC_SET0(MT,CODE,CODE2)
+
+ #define PUT_FUNC_SET1(MT,CODE,CODE2) \
+ PUT_FUNC(MT,signed char) \
+ PUT_FUNC(MT,const signed char *) \
+ PUT_FUNC_SET0(MT,CODE,CODE2)
#endif /* ACE_LACKS_SIGNED_CHAR */
+#define GET_MANIP_CODE { if( ipfx() ) { (*func)(*this); } isfx(); return *this; }
+#define PUT_MANIP_CODE { if( opfx() ) { (*func)(*this); } osfx(); return *this; }
+
+#define GET_FUNC_SET(MT) GET_FUNC_SET1(MT,GET_CODE,GET_MANIP_CODE)
+#define PUT_FUNC_SET(MT) PUT_FUNC_SET1(MT,PUT_CODE,PUT_MANIP_CODE)
+#define GETPUT_FUNC_SET(MT) GET_FUNC_SET(MT) PUT_FUNC_SET(MT)
+
+
+#define GET_SIG_SET(MT) GET_FUNC_SET1(MT,= 0;,= 0;)
+#define PUT_SIG_SET(MT) PUT_FUNC_SET1(MT,= 0;,= 0;)
+#define GETPUT_SIG_SET(MT) GET_SIG_SET(MT) PUT_SIG_SET(MT)
+
+
+///////////////////////////////////////////////////////////////////////////
+
template <class STREAM>
class ACE_IOStream : public iostream, public STREAM
// = TITLE
@@ -316,12 +503,12 @@ class ACE_IOStream : public iostream, public STREAM
// customize only one or two.
{
public:
- ACE_IOStream (void);
+ ACE_IOStream ( STREAM & stream, u_int streambuf_size = ACE_STREAMBUF_SIZE);
+ ACE_IOStream ( u_int streambuf_size = ACE_STREAMBUF_SIZE);
// The default constructor. This will initiailze your STREAM and
// then setup the iostream baseclass to use a custom streambuf based
// on STREAM.
-
virtual ~ACE_IOStream (void);
// We have to get rid of the streambuf_ ourselves since we gave it
// to iostream();
@@ -330,43 +517,35 @@ public:
// The only ambituity in the multiple inheritance is the close()
// function.
-#if defined (__GNUC__)
- virtual ACE_IOStream& operator>>(String & v);
+#if defined(__GNUC__) || defined(ACE_WIN32)
+ virtual ACE_IOStream & operator>>(ACE_IOStream_String & v);
// A simple string operator. The base iostream has 'em for char*
// but that isn't always the best thing for a String. If we don't
// provide our own here, we may not get what we want.
- virtual ACE_IOStream& operator>>(QuotedString &v);
+ virtual ACE_IOStream & operator<<(ACE_IOStream_String & v);
+ // The converse of the String put operator.
+
+ virtual ACE_IOStream & operator>>(QuotedString &v);
// A more clever operator that handles quoted strings so that we
// can get strings containing whitespace!
- virtual ACE_IOStream& operator<<(QuotedString &v);
+ virtual ACE_IOStream & operator<<(QuotedString &v);
// The converse of the QuotedString put operator.
-#endif /* __GNUG__ */
+#endif /* __GNUC__ || ACE_WIN32 */
-#if !defined (ACE_LACKS_IOSTREAM_SETGET)
// = Using the macros to provide get/set operators.
- ACE_OPERATORG_SET (ACE_IOStream<STREAM>);
- ACE_OPERATORP_SET (ACE_IOStream<STREAM>);
+#if !defined (ACE_LACKS_IOSTREAM_SETGET)
+ GETPUT_FUNC_SET (ACE_IOStream<STREAM>)
#endif /* ACE_LACKS_IOSTREAM_SETSET */
- // = These are handy to have around for overriding.
-
- // The *pfx functions are called at the beginning of each get/put
- // operator. The *sfx are called at the end. In a derivative
- // class, I've overridden the osfx() operator such that a space will
- // be inserted after every put operation so that my transmitted
- // "fields" are always separated.
-
#if defined (ACE_LACKS_IOSTREAM_FX)
- // These should be faked out to do the right thing, if we knew
- // what that was. Instead, they're just faked out.
- virtual int ipfx (int need = 0) { ACE_UNUSED_ARG (need); return 1; }
- virtual int ipfx0(void) { return 1; } // Optimized ipfx(0)
- virtual int ipfx1(void) { return 1; } // Optimized ipfx(1)
- virtual void isfx (void) { /* null */ }
- virtual int opfx (void) { return 1; }
- virtual void osfx (void) { /* null */ }
+ virtual int ipfx (int need = 0) { ACE_UNUSED_ARG (need); return good(); }
+ virtual int ipfx0(void) { return good(); } // Optimized ipfx(0)
+ virtual int ipfx1(void) { return good(); } // Optimized ipfx(1)
+ virtual void isfx (void) { }
+ virtual int opfx (void) { return good(); }
+ virtual void osfx (void) { put(' '); }
#else
#if defined (__GNUC__)
virtual int ipfx0(void) { return(iostream::ipfx0()); } // Optimized ipfx(0)
@@ -381,11 +560,17 @@ public:
virtual void osfx (void) { iostream::osfx(); }
#endif /* ACE_LACKS_IOSTREAM_FX */
+ ACE_IOStream & operator>>( ACE_Time_Value *& tv );
+ // Allow the programmer to provide a timeout for read operations.
+ // Give it a pointer to NULL to block forever.
+
protected:
+
ACE_Streambuf<STREAM> *streambuf_;
// This is where all of the action takes place. The
// streambuf_ is the interface to the underlying STREAM.
+private:
// We move these into the private section so that they cannot
// be used by the application programmer. This is necessary
// because streambuf_ will be buffering IO on the STREAM
@@ -397,8 +582,96 @@ protected:
ssize_t recv_n (...);
};
+///////////////////////////////////////////////////////////////////////////
+
+template <class STREAM>
+class ACE_SOCK_Dgram_SC : public STREAM
+// Datagrams don't have the notion of a "peer". Each send and receive
+// on a datagram can go to a different peer if you want. If you're
+// using datagrams for stream activity, you probably want 'em all to
+// go to (and come from) the same place. That's what this class is
+// for. BTW: 'Dgram_SC' is short for 'Datagram-Self-Contained'.
+// Here, we keep an address object so that we can remember who last
+// sent us data. When we write back, we're then able to write back
+// to that same address.
+{
+protected:
+ ACE_INET_Addr peer_;
+
+public:
+ ACE_SOCK_Dgram_SC() : STREAM()
+ {
+ }
+
+ ACE_SOCK_Dgram_SC ( STREAM & source, ACE_INET_Addr & dest )
+ : STREAM( source ), peer_ ( dest )
+ {
+ }
+
+ inline ssize_t send_n( char * buf,ssize_t len )
+ {
+ return STREAM::send(buf,len,peer_);
+ }
+ inline ssize_t recv ( char * buf,ssize_t len, ACE_Time_Value * tv = NULL)
+ {
+ return recv( buf, len, 0, tv );
+ }
+ inline ssize_t recv ( char * buf,ssize_t len, int flags, ACE_Time_Value * tv = NULL)
+ {
+ if( tv != 0 )
+ {
+ ACE_HANDLE handle = this->get_handle();
+ ACE_Handle_Set handle_set;
+
+ handle_set.set_bit (handle);
+
+ switch (ACE_OS::select (int (handle) + 1,
+ (fd_set *) handle_set, // read_fds.
+ (fd_set *) 0, // write_fds.
+ (fd_set *) 0, // exception_fds.
+ tv))
+ {
+ case 0:
+ errno = ETIME;
+ case -1:
+ return -1;
+ default:
+ ; // Do the 'recv' below
+ }
+ }
+
+ int rval = STREAM::recv (buf,len,peer_,flags);
+#ifdef WIN32
+ if( rval == SOCKET_ERROR )
+ if( WSAGetLastError() == WSAEMSGSIZE )
+ if( flags & MSG_PEEK )
+ rval = len;
+#endif
+
+ return (rval<len)?rval:len;
+ }
+ inline ssize_t recv_n( char * buf,ssize_t len, int flags = 0, ACE_Time_Value * tv = NULL )
+ {
+ int rval = this->recv( buf, len, flags, tv );
+ return rval;
+ }
+ inline int get_remote_addr (ACE_INET_Addr & addr) const
+ {
+ addr = peer_;
+ return 0;
+ }
+
+};
+
+///////////////////////////////////////////////////////////////////////////
+
+
#if defined (ACE_TEMPLATES_REQUIRE_SOURCE)
-#include "IOStream.cpp"
+# if ! defined( ACE_IOSTREAM_C )
+# define ACE_IOSTREAM_BUILDING_TEMPLATE
+# endif
+# include "IOStream.cpp"
#endif /* ACE_TEMPLATES_REQUIRE_SOURCE */
#endif /* ACE_IOSTREAM_H */
+