807639bd1781de81855ed9f9895f0f5e45c0d639
[senf.git] / Socket / SocketHandle.cc
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 SocketHandle non-inline non-template implementation
25  */
26
27 #include "SocketHandle.hh"
28 #include "SocketHandle.ih"
29
30 // Custom includes
31 #include <sstream>
32 #include <sys/socket.h>
33 #include "Utils/TypeInfo.hh"
34
35 //#include "SocketHandle.mpp"
36 #define prefix_
37 ///////////////////////////////cc.p////////////////////////////////////////
38
39 prefix_ void senf::SocketBody::v_close()
40 {
41     if (::shutdown(fd(),SHUT_RDWR) < 0)
42         throw SystemException(errno);
43     if (::close(fd()) < 0)
44         throw SystemException(errno);
45 }
46
47 prefix_ void senf::SocketBody::v_terminate()
48 {
49     struct linger ling;
50     ling.l_onoff = 0;
51     ling.l_linger = 0;
52
53     // We purposely IGNORE any errors: this method is used to try and
54     // terminate the connection ignoring any possible problems
55
56     ::setsockopt(fd(),SOL_SOCKET,SO_LINGER,&ling,sizeof(ling));
57     ::shutdown(fd(),SHUT_RDWR);
58     ::close(fd());
59 }
60
61 prefix_ bool senf::SocketBody::v_eof()
62     const
63 {
64     return protocol().eof();
65 }
66
67 prefix_ void senf::SocketBody::state(SocketStateMap & map, unsigned lod)
68 {
69     map["file.handle"] = fd();
70     map["file.refcount"] = refcount();
71     map["socket.server"] = isServer();
72     map["socket.protocol"] = prettyName(typeid(protocol()));
73     map["socket.policy"] = prettyName(typeid(protocol().policy()));
74     protocol().state(map,lod);
75 }
76
77 ///////////////////////////////////////////////////////////////////////////
78 // senf::detail::StateMapOrdering
79
80 namespace {
81     bool contains(std::string::const_iterator b, std::string::const_iterator e, char c)
82     {
83         for (; b != e; ++b)
84             if (*b == c)
85                 return true;
86         return false;
87     }
88 }
89
90 prefix_ bool senf::detail::StateMapOrdering::operator()(std::string const & a1,
91                                                         std::string const & a2)
92     const
93 {
94     std::string::const_iterator i1 (a1.begin());
95     std::string::const_iterator const i1_end (a1.end());
96     std::string::const_iterator i2 (a2.begin());
97     std::string::const_iterator const i2_end (a2.end());
98     for(; i1 != i1_end && i2 != i2_end && *i1 == *i2; ++i1, ++i2) ;
99     if (i1 == i1_end) {
100         if (i2 == i2_end)
101             // the strings are equal
102             return false;
103         if (contains(i2,i2_end,'.'))
104             // the longer string is a sub-'directory' of the shorter
105             /** \fixme shouldn't this be *i2 == '.' ? */
106             return true;
107         return *i1 < *i2;
108     }
109     else if (i2 == i2_end) { // && i1 != i1_end
110         if (contains(i1,i1_end,'.'))
111             // the longer string is a sub-'directory' of the shorter
112             /** \fixme shouldn't this be *i1 == '.' ? */
113             return false;
114         return *i1 < *i2;
115     }
116     if (contains(i1,i1_end,'.')) {
117         if (contains(i2,i2_end,'.'))
118             return *i1 < *i2;
119         return false;
120     }
121     else if (contains(i2,i2_end,'.'))
122         return true;
123     return *i1 < *i2;
124 }
125
126 prefix_ std::string senf::detail::dumpState(SocketStateMap const & map)
127 {
128     std::stringstream s;
129     SocketStateMap::const_iterator i (map.begin());
130     SocketStateMap::const_iterator i_end (map.end());
131     for (; i != i_end; ++i)
132         s << i->first << ": " << i->second << "\n";
133     return s.str();
134 }
135
136 template <class Policy>
137 prefix_ std::ostream & senf::operator<<(std::ostream & os, SocketHandle<Policy> handle)
138 {
139     os << handle.dumpState();
140     return os;
141 }
142
143 ///////////////////////////////cc.e////////////////////////////////////////
144 #undef prefix_
145 //#include "SocketHandle.mpp"
146
147 \f
148 // Local Variables:
149 // mode: c++
150 // c-file-style: "senf"
151 // End: