source: source/ariba/utility/bootstrap/modules/periodicbroadcast/PeriodicBroadcast.h@ 4910

Last change on this file since 4910 was 4910, checked in by huebsch, 15 years ago

bootstrap testing

File size: 8.2 KB
Line 
1// [License]
2// The Ariba-Underlay Copyright
3//
4// Copyright (c) 2008-2009, Institute of Telematics, UniversitÀt Karlsruhe (TH)
5//
6// Institute of Telematics
7// UniversitÀt Karlsruhe (TH)
8// Zirkel 2, 76128 Karlsruhe
9// Germany
10//
11// Redistribution and use in source and binary forms, with or without
12// modification, are permitted provided that the following conditions are
13// met:
14//
15// 1. Redistributions of source code must retain the above copyright
16// notice, this list of conditions and the following disclaimer.
17// 2. Redistributions in binary form must reproduce the above copyright
18// notice, this list of conditions and the following disclaimer in the
19// documentation and/or other materials provided with the distribution.
20//
21// THIS SOFTWARE IS PROVIDED BY THE INSTITUTE OF TELEMATICS ``AS IS'' AND
22// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
24// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE ARIBA PROJECT OR
25// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
26// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
27// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
28// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
29// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
30// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
31// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32//
33// The views and conclusions contained in the software and documentation
34// are those of the authors and should not be interpreted as representing
35// official policies, either expressed or implied, of the Institute of
36// Telematics.
37// [License]
38
39#ifndef __PERIODIC_BROADCAST_H
40#define __PERIODIC_BROADCAST_H
41
42#include "ariba/config.h"
43
44#include <map>
45#include <string>
46#include <ctime>
47#include <iostream>
48#include <boost/asio.hpp>
49#include <boost/foreach.hpp>
50#include <boost/thread/mutex.hpp>
51#include <boost/thread/thread.hpp>
52#include "ariba/utility/bootstrap/modules/BootstrapModule.h"
53#include "ariba/utility/logging/Logging.h"
54#include "ariba/utility/system/Timer.h"
55#include "PeriodicBroadcastMessage.h"
56
57using std::map;
58using std::string;
59using std::cout;
60using boost::asio::ip::udp;
61
62namespace ariba {
63namespace utility {
64
65class PeriodicBroadcast : public BootstrapModule, public Timer {
66 use_logging_h(PeriodicBroadcast);
67public:
68 PeriodicBroadcast(BootstrapInformationCallback* _callback);
69 virtual ~PeriodicBroadcast();
70
71 virtual void start();
72 virtual void stop();
73
74 virtual string getName();
75 virtual string getInformation();
76 virtual bool isFunctional();
77 virtual void publishService(string name, string info1, string info2, string info3);
78 virtual void revokeService(string name);
79
80protected:
81 virtual void eventFunction();
82
83private:
84 void sendLocalServices();
85 void updateRemoteServices();
86
87 static const long timerinterval; // used to send out updates on our services and check for new services
88 static const long servicetimeout; // timeout after that a service is dead when we did not receive updates
89 static const unsigned int serverport_v4;
90 static const unsigned int serverport_v6;
91
92 typedef struct _Service {
93 string name;
94 string info1;
95 string info2;
96 string info3;
97 time_t lastseen;
98
99 _Service()
100 : name(""), info1(""), info2(""), info3(""), lastseen(0){
101 }
102 } Service;
103
104 typedef map<string,Service> ServiceList;
105 ServiceList localServices;
106 boost::mutex localServicesMutex;
107
108 ServiceList remoteServices;
109 boost::mutex remoteServicesMutex;
110
111 ServiceList newRemoteServices;
112 boost::mutex newRemoteServicesMutex;
113
114 boost::asio::io_service io_service;
115
116 class udp_server {
117 private:
118 udp::socket socket_v4_listen;
119 udp::socket socket_v4_send;
120 //udp::socket socket_v6;
121 udp::endpoint remote_endpoint_;
122 boost::array<char, 1500> recv_buffer_;
123 ServiceList* services;
124 boost::mutex* servicesmutex;
125
126 public:
127 udp_server(boost::asio::io_service& io_service, ServiceList* _services, boost::mutex* _servicesmutex)
128 : services(_services), servicesmutex(_servicesmutex),
129 socket_v4_listen(io_service), socket_v4_send(io_service) /*, socket_v6(io_service)*/ {
130
131 boost::asio::ip::udp::endpoint listen_endpoint_v4(
132 boost::asio::ip::udp::v4() /*boost::asio::ip::address_v4::any()*/,
133 PeriodicBroadcast::serverport_v4);
134
135/*
136 boost::asio::ip::udp::endpoint listen_endpoint_v6(
137 boost::asio::ip::address_v6::from_string("ff02::1"),
138 PeriodicBroadcast::serverport_v6);
139*/
140
141 boost::system::error_code err;
142
143 //err = socket_v4.open( listen_endpoint_v4.protocol(), err );
144 //if(err) logging_warn("failed opening ipv4 socket");
145
146 //err = socket_v6.open( listen_endpoint_v6.protocol(), err );
147 //if(err) logging_warn("failed opening ipv6 socket");
148
149 //err = socket_v4.set_option( boost::asio::ip::udp::socket::reuse_address(true), err );
150 //if(err) logging_warn("failed setting reuse address option on ipv4 socket");
151
152 //err = socket_v6.set_option( boost::asio::ip::udp::socket::reuse_address(true), err );
153 //if(err) logging_warn("failed setting reuse address option on ipv6 socket");
154
155 err = socket_v4_listen.set_option( boost::asio::socket_base::broadcast(true), err );
156 err = socket_v4_send.set_option( boost::asio::socket_base::broadcast(true), err );
157 //if(err) logging_warn("failed setting broadcast option on ipv4 socket");
158
159 //err = socket_v6.set_option( boost::asio::socket_base::broadcast(true), err );
160 //if(err) logging_warn("failed setting broadcast option on ipv6 socket");
161
162 //err = socket_v4.bind( listen_endpoint_v4, err );
163 //if(err) logging_warn("failed binding ipv4 socket");
164
165 //socket_v6.bind( listen_endpoint_v6 );
166
167 start_receive();
168 }
169
170 void sendservice(Service service){
171
172 PeriodicBroadcastMessage msg( service.name, service.info1, service.info2, service.info3 );
173 Data data = data_serialize( msg, DEFAULT_V );
174 uint8_t* pnt = data.getBuffer();
175 size_t len = data.getLength() / 8;
176 boost::system::error_code ignored_error;
177
178 cout << "-----------> sending out " << data << std::endl;
179 boost::system::error_code err;
180
181 {
182 udp::endpoint endp(udp::v4(), PeriodicBroadcast::serverport_v4);
183 endp.address( boost::asio::ip::address_v4::broadcast() );
184 socket_v4_send.send_to( boost::asio::buffer(pnt, len), endp, 0, err );
185 if(err) logging_warn("failed sending message through ipv4 socket");
186 }
187 {/*
188 udp::endpoint endp(udp::v6(), PeriodicBroadcast::serverport_v6);
189 endp.address( boost::asio::ip::address_v6::from_string("ff02::1") );
190 socket_v6.send_to( boost::asio::buffer(pnt, len), endp, 0, ignored_error );
191 */}
192 }
193
194 private:
195 void start_receive(){
196 socket_v4_listen.async_receive_from(
197 boost::asio::buffer(recv_buffer_), remote_endpoint_,
198 boost::bind(&udp_server::handle_receive, this,
199 boost::asio::placeholders::error,
200 boost::asio::placeholders::bytes_transferred));
201/*
202 socket_v6.async_receive_from(
203 boost::asio::buffer(recv_buffer_), remote_endpoint_,
204 boost::bind(&udp_server::handle_receive, this,
205 boost::asio::placeholders::error,
206 boost::asio::placeholders::bytes_transferred));
207*/
208 }
209
210 void handle_receive(const boost::system::error_code& error,
211 std::size_t bytes_transferred){
212
213 if (!error || error == boost::asio::error::message_size){
214
215 PeriodicBroadcastMessage msg;
216 Data data( (uint8_t*)recv_buffer_.data(), bytes_transferred*8 );
217 data_deserialize( msg, data );
218
219 cout << "-----------> received " << data << std::endl;
220
221 { // insert new found service
222 boost::mutex::scoped_lock( *servicesmutex );
223
224 ServiceList::iterator it = services->find( msg.getName() );
225 if( it != services->end() ) services->erase( it );
226
227 Service s;
228 s.info1 = msg.getInfo1();
229 s.info2 = msg.getInfo2();
230 s.info3 = msg.getInfo3();
231 s.lastseen = time(NULL);
232 services->insert( std::make_pair(msg.getName(), s) );
233 }
234
235 start_receive();
236 } else {
237
238 logging_warn("failed receiving message on ipv4 socket" << error);
239
240 }
241 }
242
243 void handle_send(boost::shared_ptr<std::string> /*message*/,
244 const boost::system::error_code& error,
245 std::size_t /*bytes_transferred*/){
246
247 if(error)
248 logging_warn("failed sending out message");
249
250 }
251
252 };
253
254 udp_server server;
255};
256
257}} //namespace ariba, utility
258
259#endif // __BLUETOOTH_SDP_H
Note: See TracBrowser for help on using the repository browser.