1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
|
#include "Channel_Connector.h"
// $Id$
Channel_Connector::Channel_Connector (void)
{
}
// Override the connection-failure method to add timer support.
// Note that these timers perform "expoential backoff" to
// avoid rapidly trying to reestablish connections when a link
// goes down.
int
Channel_Connector::handle_close (ACE_HANDLE sd, ACE_Reactor_Mask)
{
ACE_Connector<Channel, CHANNEL_PEER_CONNECTOR>::AST *stp = 0;
// Locate the ACE_Svc_Handler corresponding to the socket descriptor.
if (this->handler_map_.find (sd, stp) == -1)
ACE_ERROR_RETURN ((LM_ERROR, "(%t) can't locate channel %d in map, %p\n",
sd, "find"), -1);
Channel *channel = stp->svc_handler ();
// Schedule a reconnection request at some point in the future
// (note that channel uses an exponential backoff scheme).
if (ACE_Service_Config::reactor ()->schedule_timer (channel, 0,
channel->timeout ()) == -1)
ACE_ERROR_RETURN ((LM_ERROR, "(%t) %p\n",
"schedule_timer"), -1);
return 0;
}
// Initiate (or reinitiate) a connection to the Channel.
int
Channel_Connector::initiate_connection (Channel *channel,
ACE_Synch_Options &synch_options)
{
char buf[MAXHOSTNAMELEN];
// Mark ourselves as idle so that the various iterators
// will ignore us until we are reconnected.
channel->state (Channel::IDLE);
if (channel->remote_addr ().addr_to_string (buf, sizeof buf) == -1
|| channel->local_addr ().addr_to_string (buf, sizeof buf) == -1)
ACE_ERROR_RETURN ((LM_ERROR, "(%t) %p\n",
"can't obtain peer's address"), -1);
// Try to connect to the Peer.
if (this->connect (channel, channel->remote_addr (),
synch_options, channel->local_addr ()) == -1)
{
if (errno != EWOULDBLOCK)
{
channel->state (Channel::FAILED);
ACE_DEBUG ((LM_DEBUG, "(%t) %p on address %s\n",
"connect", buf));
// Reschedule ourselves to try and connect again.
if (synch_options[ACE_Synch_Options::USE_REACTOR])
{
if (ACE_Service_Config::reactor ()->schedule_timer
(channel, 0, channel->timeout ()) == 0)
ACE_ERROR_RETURN ((LM_ERROR, "(%t) %p\n",
"schedule_timer"), -1);
}
else
// Failures on synchronous connects are reported as errors
// so that the caller can decide how to proceed.
return -1;
}
else
{
channel->state (Channel::CONNECTING);
ACE_DEBUG ((LM_DEBUG,
"(%t) in the process of connecting %s to %s\n",
synch_options[ACE_Synch_Options::USE_REACTOR]
? "asynchronously" : "synchronously", buf));
}
}
else
{
channel->state (Channel::ESTABLISHED);
ACE_DEBUG ((LM_DEBUG, "(%t) connected to %s on %d\n",
buf, channel->get_handle ()));
}
return 0;
}
|