OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
segment_health_check.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2017 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 #include <pkt/proto_handler.h>
7 #include <diag/diag_proto.h>
9 #include <oper/health_check.h>
10 #include <oper/metadata_ip.h>
11 
13  DiagTable *diag_table) :
14  DiagEntry(svc->source_ip().to_string(),
15  svc->destination_ip().to_string(), IPPROTO_ICMP, 0, 0, "",
16  ((svc->service()->timeout() * 1000) +
17  (svc->service()->timeout_usecs()/1000)),
18  svc->service()->max_retries(),
19  diag_table), service_(svc), state_(SUCCESS) {
20  delay_msecs_ = GetDelay(svc);
22  (*(diag_table->agent()->event_manager())->io_service(),
23  "SegmentHCDelayTimeoutHandler",
24  TaskScheduler::GetInstance()->GetTaskId("Agent::Diag"),
26 }
27 
28 
32 }
33 
35  const {
36  /* Convert delay from seconds to milliseconds and store */
37  int msecs = svc->service()->delay() * 1000;
38  /* Convert delay_usecs from micro-seconds to milliseconds and update */
39  msecs += svc->service()->delay_usecs() / 1000;
40  return msecs;
41 }
42 
44  const {
45  /* Convert timeout from seconds to milliseconds and store */
46  int msecs = svc->service()->timeout() * 1000;
47  /* Convert timeout_usecs from micro-seconds to milliseconds and update */
48  msecs += svc->service()->timeout_usecs() / 1000;
49  return msecs;
50 }
51 
55  boost::bind(&SegmentHealthCheckPkt::RetryHandler, this));
56 }
57 
59  SendRequest();
60  RestartTimer();
61  /* return false to suppress auto-restart of timer */
62  return false;
63 }
64 
66  data->op_ = htonl(AgentDiagPktData::DIAG_REQUEST);
67  data->key_ = htons(key_);
68  data->seq_no_ = htonl(seq_no_);
69  strcpy(data->data_, DiagTable::kDiagData.c_str());
70 }
71 
73  Agent *agent = diag_table_->agent();
74 
75  //Allocate buffer to hold packet
76  boost::shared_ptr<PktInfo> pkt_info(new PktInfo(agent, kBufferSize,
77  PktHandler::DIAG, 0));
78  uint8_t *msg = pkt_info->packet_buffer()->data();
79  memset(msg, 0, kBufferSize);
80 
81  DiagPktHandler *pkt_handler =
82  new DiagPktHandler(agent, pkt_info,
83  *(agent->event_manager())->io_service());
84  uint16_t len = sizeof(AgentDiagPktData);
85  uint8_t *data = NULL;
86  if (sip_.is_v4()) {
87  //Update pointers to ethernet header, ip header and l4 header
88  pkt_info->UpdateHeaderPtr();
89  switch (proto_) {
90  case IPPROTO_ICMP:
91  len += 8;
92  data = (uint8_t *)pkt_handler->pkt_info()->transp.icmp + 8;
93  pkt_handler->pkt_info()->transp.icmp->icmp_type = ICMP_ECHO;
94  pkt_handler->pkt_info()->transp.icmp->icmp_code = 0;
95  pkt_handler->pkt_info()->transp.icmp->icmp_cksum = 0xffff;
96  break;
97  default:
98  assert(0);
99  }
100  //Add Diag header as ICMP payload
102  len += sizeof(struct ip);
103  pkt_handler->IpHdr(len, ntohl(sip_.to_v4().to_ulong()),
104  ntohl(dip_.to_v4().to_ulong()),
106  len += sizeof(ether_header);
107  InterfaceRef interface = service_->interface();
108 
109  uint32_t service_mode = (static_cast<VmInterface *>(interface.get()))->service_mode();
110  bool l3_mode = false;
111  MacAddress dest_mac;
112  if (service_mode == VmInterface::SERVICE_MODE_ERROR) {
113  delete pkt_handler;
114  return;
115  }
116 
117  if ((service_mode == VmInterface::ROUTED_MODE) ||
118  (service_mode == VmInterface::ROUTED_NAT_MODE)) {
119  l3_mode = true;
120  dest_mac = (static_cast<VmInterface *>(interface.get()))->vm_mac();
121  }
122 
123  if ((static_cast<VmInterface *>(interface.get()))->is_left_si()) {
124  if (l3_mode) {
125  pkt_handler->EthHdr(agent->left_si_mac(), dest_mac,
126  ETHERTYPE_IP);
127  } else {
128  pkt_handler->EthHdr(agent->left_si_mac(), agent->right_si_mac(),
129  ETHERTYPE_IP);
130  }
131  } else {
132  if (l3_mode) {
133  pkt_handler->EthHdr(agent->right_si_mac(), dest_mac,
134  ETHERTYPE_IP);
135  } else {
136  pkt_handler->EthHdr(agent->right_si_mac(), agent->left_si_mac(),
137  ETHERTYPE_IP);
138  }
139  }
140  } else {
141  //TODO: support for IPv6
142  assert(0);
143  }
144  //Increment the attempt count
145  seq_no_++;
146 
147  //Send request out
148  pkt_handler->pkt_info()->set_len(len);
149  pkt_handler->Send(service_->interface()->id(),
150  service_->interface()->vrf_id(), AgentHdr::TX_SWITCH,
151  CMD_PARAM_PACKET_CTRL, CMD_PARAM_1_DIAG,
153  delete pkt_handler;
156  return;
157 }
158 
160  if (seq_no_ >= GetMaxAttempts()) {
161  Notify(FAILURE);
162  seq_no_ = 0;
163  }
164  Retry();
165 }
166 
170  seq_no_ = 0;
171  Notify(SUCCESS);
172  /* Cancel the timeout timer and start delay timer to resend the request */
173  TimerCancel();
174  Retry();
175 }
176 
178  delay_timer_->Cancel();
179 }
180 
182  if (state_ != status) {
183  state_ = status;
184  std::string data = (state_ == SUCCESS) ? "success" : "failure";
185  service_->OnRead(data);
186  }
187 }
188 
190  service_ = service;
191  delay_msecs_ = GetDelay(service);
192  timeout_ = GetTimeout(service);
193  max_attempts_ = service->service()->max_retries();
194 }
IpAddress dip_
Definition: diag.h:57
int timeout_
Definition: diag.h:67
IpAddress sip_
Definition: diag.h:56
InterfaceRef interface() const
Definition: health_check.h:147
uint32_t GetMaxAttempts()
Definition: diag.h:42
#define DEFAULT_IP_TTL
Definition: pkt_handler.h:44
void UpdateService(HealthCheckInstanceService *service)
static const int kBufferSize
union PktInfo::@8 transp
virtual void HandleReply(DiagPktHandler *handler)
DiagProto * diag_proto() const
Definition: diag.h:118
HealthCheckService * service() const
Definition: health_check.h:146
uint8_t proto_
Definition: diag.h:58
void IncrementDiagStats(uint32_t itf_id, DiagStatsType type)
Definition: diag_proto.cc:93
void RequestTimedOut(uint32_t seqno)
static const std::string kDiagData
Definition: diag.h:107
#define DEFAULT_IP_ID
Definition: pkt_handler.h:48
uint32_t seq_no_
Definition: diag.h:84
void set_len(uint32_t len)
uint32_t op_
Definition: diag.h:82
char data_[8]
Definition: diag.h:85
static const MacAddress & left_si_mac()
Definition: agent.h:442
void FillDiagHeader(AgentDiagPktData *data) const
bool TimerCancel()
Definition: diag.h:45
int GetTaskId(const std::string &name)
Definition: task.cc:856
uint32_t timeout() const
Definition: health_check.h:332
int EthHdr(const MacAddress &src, const MacAddress &dest, const uint16_t proto)
Definition: agent.h:358
static TaskScheduler * GetInstance()
Definition: task.cc:547
void RestartTimer()
Definition: diag.cc:57
HealthCheckInstanceService * service_
EventManager * event_manager() const
Definition: agent.h:1103
void Send(uint32_t itf, uint32_t vrf, uint16_t, PktHandler::PktModuleName)
DiagEntry::DiagKey key_
Definition: diag.h:83
int GetTimeout(const HealthCheckInstanceService *svc) const
int GetDelay(const HealthCheckInstanceService *svc) const
uint32_t seq_no_
Definition: diag.h:70
Agent * agent() const
Definition: diag.h:117
SegmentHealthCheckPkt(HealthCheckInstanceService *service, DiagTable *diag_table)
PktInfo * pkt_info() const
Definition: proto_handler.h:88
static Timer * CreateTimer(boost::asio::io_context &service, const std::string &name, int task_id=Timer::GetTimerTaskId(), int task_instance=Timer::GetTimerInstanceId(), bool delete_on_completion=false)
Definition: timer.cc:201
bool Cancel()
Definition: timer.cc:150
static const MacAddress & right_si_mac()
Definition: agent.h:443
uint32_t max_attempts_
Definition: diag.h:69
struct icmp * icmp
Definition: pkt_handler.h:428
uint32_t delay() const
Definition: health_check.h:330
bool Start(int time, Handler handler, ErrorHandler error_handler=NULL)
Definition: timer.cc:108
uint32_t max_retries() const
Definition: health_check.h:334
boost::intrusive_ptr< Interface > InterfaceRef
Definition: agent.h:49
Definition: diag.h:21
uint64_t delay_usecs() const
Definition: health_check.h:331
void OnRead(const std::string &data)
void Notify(Status status)
DiagKey key_
Definition: diag.h:65
DiagTable * diag_table_
Definition: diag.h:64
void IpHdr(uint16_t len, in_addr_t src, in_addr_t dest, uint8_t protocol, uint16_t id, uint8_t ttl)
static bool DeleteTimer(Timer *Timer)
Definition: timer.cc:222
uint64_t timeout_usecs() const
Definition: health_check.h:333