6ad40407cead9d000ced04eb52fd90f6bce3b2a3
[senf.git] / Socket / ClientSocketHandle.ct
1 // $Id$
2 //
3 // Copyright (C) 2006
4 // Fraunhofer Institut fuer offene Kommunikationssysteme (FOKUS)
5 // Kompetenzzentrum fuer Satelitenkommunikation (SatCom)
6 //     Stefan Bund <stefan.bund@fokus.fraunhofer.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief ClientSocketHandle non-inline template implementation
25  */
26
27 #include "ClientSocketHandle.ih"
28
29 // Custom includes
30 #include <algorithm>
31 #include "../Utils/Buffer.hh"
32
33 #define prefix_
34 ///////////////////////////////ct.p////////////////////////////////////////
35
36 ///////////////////////////////////////////////////////////////////////////
37 // senf::detail::ReadRange<Handle,ForwardWritableRange,IsContiguous>
38
39 template <class Handle, class ForwardWritableRange, bool IsContiguous>
40 prefix_ typename boost::range_iterator<ForwardWritableRange>::type
41 senf::detail::ReadRange<Handle,ForwardWritableRange,IsContiguous>::
42 read(Handle & handle, ForwardWritableRange & range)
43 {
44     typename boost::range_size<ForwardWritableRange>::type nread (boost::size(range));
45     SENF_SCOPED_BUFFER(char, buffer, nread);
46     return std::copy(buffer, handle.read(buffer,buffer+nread), boost::begin(range));
47 }
48
49 template <class Handle, class ForwardWritableRange, bool IsContiguous>
50 prefix_ typename boost::range_iterator<ForwardWritableRange>::type
51 senf::detail::ReadRange<Handle,ForwardWritableRange,IsContiguous>::
52 readfrom(Handle & handle, ForwardWritableRange & range, typename Handle::Address & addr)
53 {
54     typename boost::range_size<ForwardWritableRange>::type nread (boost::size(range));
55     SENF_SCOPED_BUFFER(char, buffer, nread);
56     return std::copy(buffer, handle.readfrom(buffer,buffer+nread,addr), boost::begin(range));
57 }
58
59 ///////////////////////////////////////////////////////////////////////////
60 // senf::detail::WriteRange<Handle,ForwardReadableRange,IsContiguous>
61
62 template <class Handle, class ForwardReadableRange, bool IsContiguous>
63 prefix_ typename boost::range_const_iterator<ForwardReadableRange>::type
64 senf::detail::WriteRange<Handle,ForwardReadableRange,IsContiguous>::
65 write(Handle & handle, ForwardReadableRange & range)
66 {
67     typename boost::range_size<ForwardReadableRange>::type nwrite (boost::size(range));
68     typename boost::range_const_iterator<ForwardReadableRange>::type i (boost::begin(range));
69     SENF_SCOPED_BUFFER(char, buffer, nwrite);
70     std::copy(i, boost::end(range), buffer);
71     std::advance(i, handle.write(std::make_pair(buffer, buffer+nwrite)) - buffer);
72     return i;
73 }
74
75 template <class Handle, class ForwardReadableRange, bool IsContiguous>
76 prefix_ typename boost::range_const_iterator<ForwardReadableRange>::type
77 senf::detail::WriteRange<Handle,ForwardReadableRange,IsContiguous>::
78 writeto(Handle & handle, ForwardReadableRange & range, typename Handle::Address const & addr)
79 {
80     typename boost::range_size<ForwardReadableRange>::type nwrite (boost::size(range));
81     typename boost::range_const_iterator<ForwardReadableRange>::type i (boost::begin(range));
82     SENF_SCOPED_BUFFER(char, buffer, nwrite);
83     std::copy(i, boost::end(range), buffer);
84     std::advance(i, handle.writeto(std::make_pair(buffer, buffer+nwrite), addr) - buffer);
85     return i;
86 }
87
88 ///////////////////////////////////////////////////////////////////////////
89 // senf::ClientSocketHandle<Policy>
90
91 ////////////////////////////////////////
92 // reading and writing
93
94 // senf::ClientSocketHandle<Policy>::read
95
96 template <class Policy>
97 prefix_ std::string senf::ClientSocketHandle<Policy>::read(unsigned limit)
98 {
99     std::string rv;
100     this->read(rv,limit);
101     return rv;
102 }
103
104 template <class Policy>
105 template <class Sequence>
106 prefix_ void senf::ClientSocketHandle<Policy>::read(Sequence & container, unsigned limit)
107 {
108     unsigned nread (available());
109     if (limit>0 && nread>limit)
110         nread = limit;
111     container.resize(nread);
112     container.erase(read( std::make_pair(container.begin(), container.end()) ), 
113                     container.end());
114 }
115
116 // senf::ClientSocketHandle<Policy>::readfrom
117
118 template <class Policy>
119 prefix_ std::pair<std::string, typename Policy::AddressingPolicy::Address>
120 senf::ClientSocketHandle<Policy>::readfrom(unsigned limit)
121 {
122     std::string rv;
123     typename Policy::AddressingPolicy::Address addr;
124     this->readfrom(rv,addr,limit);
125     return std::make_pair(rv,addr);
126 }
127
128 template <class Policy>
129 template <class Sequence>
130 prefix_ void senf::ClientSocketHandle<Policy>::readfrom(Sequence & container, Address & from,
131                                                         unsigned limit)
132 {
133     unsigned nread (available());
134     if (limit>0 && nread>limit)
135         nread = limit;
136     container.resize(nread);
137     container.erase(readfrom( std::make_pair(container.begin(), container.end()), from ), 
138                     container.end());
139 }
140
141 ////////////////////////////////////////
142 // private members
143
144 // senf::ClientSocketHandle<Policy>::available
145
146 template <class Policy>
147 prefix_ unsigned senf::ClientSocketHandle<Policy>::available()
148 {
149     unsigned nread = this->protocol().available();
150     if (nread == 0 && this->blocking()) {
151         // We have to block explicitly here so we can return the
152         // number of bytes available explicitly. If no more date can
153         // be expected to arive (i.e. the other end has closed the
154         // connection), the socket will always be in the readable
155         // state. This is the only case when available() will return
156         // 0.
157         this->waitReadable();
158         nread = this->protocol().available();
159     }
160     return nread;
161 }
162
163 ///////////////////////////////ct.e////////////////////////////////////////
164 #undef prefix_
165
166 \f
167 // Local Variables:
168 // mode: c++
169 // fill-column: 100
170 // c-file-style: "senf"
171 // indent-tabs-mode: nil
172 // ispell-local-dictionary: "american"
173 // compile-command: "scons -u test"
174 // comment-column: 40
175 // End: