OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
arp_proto.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 #include "sandesh/sandesh_types.h"
6 #include "sandesh/sandesh.h"
7 #include "base/address_util.h"
8 #include "init/agent_init.h"
9 #include "oper/nexthop.h"
10 #include "oper/tunnel_nh.h"
11 #include "oper/mirror_table.h"
12 #include "oper/route_common.h"
13 #include "pkt/pkt_init.h"
14 #include "services/arp_proto.h"
16 #include "services_init.h"
18 
19 ArpProto::ArpProto(Agent *agent, boost::asio::io_context &io,
20  bool run_with_vrouter) :
21  Proto(agent, "Agent::Services", PktHandler::ARP, io),
22  run_with_vrouter_(run_with_vrouter), ip_fabric_interface_index_(-1),
23  ip_fabric_interface_(NULL), max_retries_(kMaxRetries),
24  retry_timeout_(kRetryTimeout), aging_timeout_(kAgingTimeout) {
25  // limit the number of entries in the workqueue
27  work_queue_.SetBounded(true);
28 
30  boost::bind(&ArpProto::VrfNotify, this, _1, _2));
32  boost::bind(&ArpProto::InterfaceNotify,
33  this, _2));
35  boost::bind(&ArpProto::NextHopNotify, this, _2));
36 }
37 
39 }
40 
42  // we may have arp entries in arp cache without ArpNH, empty them
43  for (ArpIterator it = arp_cache_.begin(); it != arp_cache_.end(); ) {
44  it = DeleteArpEntry(it);
45  }
46 
48  it != gratuitous_arp_cache_.end(); it++) {
49  for (ArpEntrySet::iterator sit = it->second.begin();
50  sit != it->second.end();) {
51  ArpEntry *entry = *sit;
52  it->second.erase(sit++);
53  delete entry;
54  }
55  }
56  gratuitous_arp_cache_.clear();
60 }
61 
62 ProtoHandler *ArpProto::AllocProtoHandler(boost::shared_ptr<PktInfo> info,
63  boost::asio::io_context &io) {
64  return new ArpHandler(agent(), info, io);
65 }
66 
68  VrfEntry *vrf = static_cast<VrfEntry *>(entry);
69  ArpVrfState *state;
70 
71  state = static_cast<ArpVrfState *>(entry->GetState(part->parent(),
73  if (entry->IsDeleted()) {
74  if (state) {
75  for (ArpProto::ArpIterator it = arp_cache_.begin();
76  it != arp_cache_.end();) {
77  ArpEntry *arp_entry = it->second;
78  if (arp_entry->key().vrf == vrf && arp_entry->DeleteArpRoute()) {
79  it = DeleteArpEntry(it);
80  } else
81  it++;
82  }
84  it != gratuitous_arp_cache_.end(); ) {
85  ArpKey key = it->first;
86  if (key.vrf == vrf) {
87  for (ArpEntrySet::iterator sit = it->second.begin();
88  sit != it->second.end();) {
89  ArpEntry *entry = *sit;
90  it->second.erase(sit++);
91  delete entry;
92  }
93  gratuitous_arp_cache_.erase(it++);
94  }else {
95  it++;
96  }
97  }
98  state->Delete();
99  }
100  return;
101  }
102 
103  if (!state){
104  state = new ArpVrfState(agent_, this, vrf,
106  vrf->GetEvpnRouteTable());
107  state->route_table_listener_id = vrf->
108  GetInet4UnicastRouteTable()->
109  Register(boost::bind(&ArpVrfState::RouteUpdate, state, _1, _2));
111  Register(boost::bind(&ArpVrfState::EvpnRouteUpdate, state, _1, _2));
112  entry->SetState(part->parent(), vrf_table_listener_id_, state);
113  }
114 }
115 
117  aps->refcount_.fetch_and_increment();
118 }
119 
121  ArpVrfState *state = aps->vrf_state();
122  int prev = aps->refcount_.fetch_and_decrement();
123  if (prev == 1) {
124  state->Erase(aps->ip(), aps->plen());
125  delete aps;
126  }
127 }
128 
130  uint32_t vrf_id,
131  const IpAddress &ip,
132  uint8_t plen):
133  vrf_state_(state), arp_req_timer_(NULL), vrf_id_(vrf_id),
134  vm_ip_(ip), plen_(plen) {
135  refcount_ = 0;
136 }
137 
139  if (arp_req_timer_) {
142  }
143  assert(refcount_ == 0);
144 }
145 
147  if (arp_req_timer_ == NULL) {
150  "Arp Entry timer for VM",
152  GetTaskId("Agent::Services"), PktHandler::ARP);
153  }
156  this));
157 }
158 
160  &wait_for_traffic_map,
162  &arp_transmitted_map) {
163  bool ret = false;
164  boost::shared_ptr<PktInfo> pkt(new PktInfo(vrf_state_->agent,
166  PktHandler::ARP, 0));
167  ArpHandler arp_handler(vrf_state_->agent, pkt,
169 
170  WaitForTrafficIntfMap::iterator it = wait_for_traffic_map.begin();
171  for (;it != wait_for_traffic_map.end(); it++) {
172  const VmInterface *vm_intf = static_cast<const VmInterface *>(
174  if (!vm_intf) {
175  continue;
176  }
177 
178  InterfaceArpPathPreferenceInfo &data = it->second;
179  bool inserted = arp_transmitted_map.insert(it->first).second;
180  ++data.arp_retry_count;
181  if (inserted == false) {
182  //ARP request already sent due to IP route
183  continue;
184  }
185  if ((plen_ != 32) && (!data.prev_responded_ip.is_unspecified())) {
186  if ((data.arp_send_count >= 1) && (data.arp_reply_count == 0)) {
187  ++data.arp_failure_count;
188  }
190  data.prev_responded_ip = Ip4Address(0);
191  data.arp_failure_count = 0;
192  data.arp_reply_count = 0;
193  data.arp_send_count = 0;
194  }
195  }
196  ++data.arp_send_count;
197 
198  // Post message to delete MIL entry.
200  GetMacIpLearningTable()->GetPairedMacAddress(
201  vm_intf->vrf_id(), ip());
202  if (mil_mac != MacAddress()) {
203  ++data.arp_try_count;
204  if (data.arp_try_count == kArpTryCount) {
205  IpAddress ip = vm_ip_;
206  MacAddress mac = mil_mac;
208  GetMacIpLearningTable()->MacIpEntryUnreachable(
209  vm_intf->vrf_id(), ip, mac);
210  return true;
211  }
212  }
213 
214  MacAddress smac = vm_intf->GetVifMac(vrf_state_->agent);
215  arp_handler.SendArpRequestByPlen(vm_intf, smac, this,
216  data.prev_responded_ip);
217 
218  // reduce the frequency of ARP requests after some tries
219  if (data.arp_send_count >= kMaxRetry) {
220  if ((mac() != MacAddress()) && (mac() != vm_intf->vm_mac())) {
221  arp_req_timer_->Reschedule(5000);
222  } else if (vm_intf->vmi_type() != VmInterface::REMOTE_VM) {
223  // change frequency only if not in gateway mode with remote VMIs
225  }
226  }
227 
228  ret = true;
229  }
230  return ret;
231 }
232 
234  if (l3_wait_for_traffic_map_.size() == 0 &&
235  evpn_wait_for_traffic_map_.size() == 0) {
236  return false;
237  }
238 
239  bool ret = false;
240  ArpTransmittedIntfMap arp_transmitted_map;
241  if (SendArpRequest(l3_wait_for_traffic_map_, arp_transmitted_map)) {
242  ret = true;
243  }
244 
245  if (SendArpRequest(evpn_wait_for_traffic_map_, arp_transmitted_map)) {
246  ret = true;
247  }
248 
249  return ret;
250 }
251 
252 //Send ARP request on interface in Active-BackUp mode
253 //So that preference of route can be incremented if the VM replies to ARP
255  AgentRoute *route) {
256  WaitForTrafficIntfMap new_wait_for_traffic_map;
257  WaitForTrafficIntfMap wait_for_traffic_map = evpn_wait_for_traffic_map_;
258  if (dynamic_cast<const InetUnicastRouteEntry *>(route)) {
259  wait_for_traffic_map = l3_wait_for_traffic_map_;
260  }
261 
262  for (Route::PathList::const_iterator it = route->GetPathList().begin();
263  it != route->GetPathList().end(); it++) {
264  const AgentPath *path = static_cast<const AgentPath *>(it.operator->());
265  if (path->peer() &&
266  path->peer()->GetType() == Peer::LOCAL_VM_PORT_PEER) {
267  const NextHop *nh = path->ComputeNextHop(vrf_state_->agent);
268  if (nh->GetType() != NextHop::INTERFACE) {
269  continue;
270  }
271  if (path->is_health_check_service()) {
272  // skip sending ARP request for Health Check Service IP
273  continue;
274  }
275 
276  const InterfaceNH *intf_nh =
277  static_cast<const InterfaceNH *>(nh);
278  const Interface *intf =
279  static_cast<const Interface *>(intf_nh->GetInterface());
280  if (intf->type() != Interface::VM_INTERFACE) {
281  //Ignore non vm interface nexthop
282  continue;
283  }
284  if (path->subnet_service_ip().is_v4() == false) {
285  continue;
286  }
287  if (path->path_preference().IsDependentRt() == true) {
288  continue;
289  }
290  uint32_t intf_id = intf->id();
291  WaitForTrafficIntfMap::const_iterator wait_for_traffic_it =
292  wait_for_traffic_map.find(intf_id);
293  if (wait_for_traffic_it == wait_for_traffic_map.end()) {
295  new_wait_for_traffic_map.insert(WaitForTrafficIntfPair(intf_id,
296  data));
297  } else {
298  new_wait_for_traffic_map.insert(WaitForTrafficIntfPair(intf_id,
299  wait_for_traffic_it->second));
300  }
301  }
302  }
303 
304  if (dynamic_cast<const InetUnicastRouteEntry *>(route)) {
305  l3_wait_for_traffic_map_ = new_wait_for_traffic_map;
306  } else {
307  evpn_wait_for_traffic_map_ = new_wait_for_traffic_map;
308  }
309  if (new_wait_for_traffic_map.size() > 0) {
310  SendArpRequest();
311  StartTimer();
312  }
313 }
314 
315 ArpDBState::ArpDBState(ArpVrfState *vrf_state, uint32_t vrf_id, IpAddress ip,
316  uint8_t plen) : vrf_state_(vrf_state),
317  sg_list_(), tag_list_(), policy_(false), resolve_route_(false) {
318  if (ip != Ip4Address(0)) {
319  arp_path_preference_state_.reset(vrf_state->Locate(ip, plen));
320  }
321 }
322 
324 }
325 
327 
328  if (nh && arp_path_preference_state_.get()) {
329  arp_path_preference_state_.get()->set_mac(nh->GetDMac());
330  }
331 
332 
333  return;
334 }
335 
337  int plen = rt->prefix_length();
338  uint32_t start_ip = rt->prefix_address().to_v4().to_ulong();
339  ArpKey start_key(start_ip, rt->vrf());
340 
341  ArpProto::ArpIterator start_iter =
343 
344 
345  while (start_iter != vrf_state_->arp_proto->arp_cache().end() &&
346  start_iter->first.vrf == rt->vrf() &&
347  IsIp4SubnetMember(Ip4Address(start_iter->first.ip),
348  rt->prefix_address().to_v4(), plen)) {
349  start_iter->second->Resync(policy_, vn_list_, sg_list_, tag_list_);
350  start_iter++;
351  }
352 }
353 
355  int plen = rt->prefix_length();
356  uint32_t start_ip = rt->prefix_address().to_v4().to_ulong();
357 
358  ArpKey start_key(start_ip, rt->vrf());
359 
360  ArpProto::ArpIterator start_iter =
362 
363  while (start_iter != vrf_state_->arp_proto->arp_cache().end() &&
364  start_iter->first.vrf == rt->vrf() &&
365  IsIp4SubnetMember(Ip4Address(start_iter->first.ip),
366  rt->prefix_address().to_v4(), plen)) {
367  ArpProto::ArpIterator tmp = start_iter++;
368  if (tmp->second->DeleteArpRoute()) {
369  vrf_state_->arp_proto->DeleteArpEntry(tmp->second);
370  }
371  }
372 }
373 
376  arp_path_preference_state_->SendArpRequestForAllIntf(rt);
377  }
378 
379  const InetUnicastRouteEntry *ip_rt =
380  dynamic_cast<const InetUnicastRouteEntry *>(rt);
381  if (ip_rt == NULL) {
382  return;
383  }
384  const NextHop *anh =ip_rt->GetActiveNextHop();
385  if (anh == NULL) {
386  return;
387  }
388 
389  if (anh->GetType() == NextHop::RESOLVE) {
390  resolve_route_ = true;
391  }
392 
393  bool policy = anh->PolicyEnabled();
394  const SecurityGroupList sg = ip_rt->GetActivePath()->sg_list();
395  const TagList tag = ip_rt->GetActivePath()->tag_list();
396 
397 
398  if (policy_ != policy || sg != sg_list_ || tag != tag_list_ ||
399  vn_list_ != ip_rt->GetActivePath()->dest_vn_list()) {
400  policy_ = policy;
401  sg_list_ = sg;
402  tag_list_ = tag;
403  vn_list_ = ip_rt->GetActivePath()->dest_vn_list();
404  if (resolve_route_) {
405  UpdateArpRoutes(ip_rt);
406  }
407  }
408 }
409 
411  EvpnRouteEntry *route = static_cast<EvpnRouteEntry *>(entry);
412  /* Ignore route updates for Non-IPv4 addresses */
413  if (!route->prefix_address().is_v4()) {
414  return;
415  }
416 
417  ArpDBState *state = static_cast<ArpDBState *>(entry->GetState(part->parent(),
419 
420  if (entry->IsDeleted() || deleted) {
421  if (state) {
423  delete state;
424  }
425  return;
426  }
427 
428  if (state == NULL) {
429  state = new ArpDBState(this, route->vrf_id(), route->prefix_address(),
430  route->prefix_length());
431  entry->SetState(part->parent(), evpn_route_table_listener_id, state);
432  }
433 
434  state->Update(route);
435 }
436 
438  InetUnicastRouteEntry *route = static_cast<InetUnicastRouteEntry *>(entry);
439 
440  ArpDBState *state =
441  static_cast<ArpDBState *>
442  (entry->GetState(part->parent(), route_table_listener_id));
443 
444  const InterfaceNH *intf_nh = dynamic_cast<const InterfaceNH *>(
445  route->GetActiveNextHop());
446  const Interface *intf = (intf_nh) ?
447  static_cast<const Interface *>(intf_nh->GetInterface()) : NULL;
448 
449  ArpKey key(route->prefix_address().to_v4().to_ulong(), route->vrf());
450  ArpEntry *arpentry = arp_proto->GratuitousArpEntry(key, intf);
451  if (entry->IsDeleted() || deleted) {
452  if (state) {
454  entry->ClearState(part->parent(), route_table_listener_id);
455  state->Delete(route);
456  delete state;
457  }
458  return;
459  }
460 
461  if (!state) {
462  state = new ArpDBState(this, route->vrf_id(), route->prefix_address(),
463  route->prefix_length());
464  entry->SetState(part->parent(), route_table_listener_id, state);
465  }
466 
467  state->UpdateMac(intf_nh);
468 
469  if (route->vrf()->GetName() == agent->fabric_vrf_name() && intf_nh &&
470  route->GetActiveNextHop()->GetType() == NextHop::RECEIVE &&
471  arp_proto->agent()->router_id() == route->prefix_address().to_v4()) {
472  //Send Grat ARP
475  route->prefix_address().to_v4().to_ulong(), route->vrf(),
477  } else {
478  if (intf_nh) {
479  if (intf->type() == Interface::VM_INTERFACE &&
480  static_cast<const VmInterface*>(intf)->IsActive()) {
481  ArpKey intf_key(route->prefix_address().to_v4().to_ulong(), route->vrf());
482  arp_proto->AddGratuitousArpEntry(intf_key);
484  route->prefix_address().to_v4().to_ulong(), intf->vrf(), intf);
485  }
486  }
487  }
488 
489  //Check if there is a local VM path, if yes send a
490  //ARP request, to trigger route preference state machine
491  if (state && route->vrf()->GetName() != agent->fabric_vrf_name()) {
492  state->Update(route);
493  }
494 }
495 
497  RouteUpdate(part, entry);
498  return true;
499 }
500 
502  DBEntryBase *entry) {
503  EvpnRouteUpdate(part, entry);
504  return true;
505 }
506 
507 
509  if (managed_delete_walk_ref.get() == NULL)
510  return;
511 
513  if (evpn_walk_ref_.get())
515  deleted = true;
516 }
517 
519  if (partition == state->rt_table) {
521  state->managed_delete_walk_ref = NULL;
522  state->l3_walk_completed_ = true;
523  } else {
524  //Get rt table from partition
526  state->evpn_walk_ref_ = NULL;
527  state->evpn_walk_completed_ = true;
528  }
529  if (state->PreWalkDone(partition)) {
530  delete state;
531  }
532 }
533 
535  if (arp_proto->ValidateAndClearVrfState(vrf, this) == false) {
536  return false;
537  }
538 
540  table_delete_ref.Reset(NULL);
541 
544  return true;
545 }
546 
548  ArpPathPreferenceStateKey key(ip, plen);
549  ArpPathPreferenceState* ptr = NULL;
550  ArpPathPreferenceStateMap::iterator it = arp_path_preference_map_.find(key);
551  if (it == arp_path_preference_map_.end()) {
552  ptr = new ArpPathPreferenceState(this, vrf->vrf_id(), ip, plen);
554  } else {
555  ptr = it->second;
556  }
557  return ptr;
558 }
559 
560 void ArpVrfState::Erase(const IpAddress &ip, uint8_t plen) {
561  ArpPathPreferenceStateKey key(ip, plen);
562  ArpPathPreferenceStateMap::iterator it = arp_path_preference_map_.find(key);
563  if (it != arp_path_preference_map_.end()) {
564  arp_path_preference_map_.erase(it);
565  }
566 }
567 
568 
570  ArpPathPreferenceStateKey key(ip, plen);
571  ArpPathPreferenceStateMap::iterator it = arp_path_preference_map_.find(key);
572  if (it != arp_path_preference_map_.end()) {
573  return it->second;
574  }
575  return NULL;
576 }
577 
578 ArpVrfState::ArpVrfState(Agent *agent_ptr, ArpProto *proto, VrfEntry *vrf_entry,
579  AgentRouteTable *table, AgentRouteTable *evpn_table):
580  agent(agent_ptr), arp_proto(proto), vrf(vrf_entry), rt_table(table),
581  evpn_rt_table(evpn_table), route_table_listener_id(DBTableBase::kInvalidId),
582  evpn_route_table_listener_id(DBTableBase::kInvalidId),
583  table_delete_ref(this, table->deleter()),
584  evpn_table_delete_ref(this, evpn_table->deleter()),
585  deleted(false),
586  l3_walk_completed_(false), evpn_walk_completed_(false) {
588  boost::bind(&ArpVrfState::DeleteEvpnRouteState, this, _1, _2),
589  boost::bind(&ArpVrfState::WalkDone, _2, this));
591  boost::bind(&ArpVrfState::DeleteRouteState, this, _1, _2),
592  boost::bind(&ArpVrfState::WalkDone, _2, this));
593 }
594 
596  assert(arp_path_preference_map_.size() == 0);
597 }
598 
600  Interface *intf = static_cast<Interface *>(entry);
601  ArpInterfaceState *state = static_cast<ArpInterfaceState *>
602  (entry->GetState(entry->get_table_partition()->parent(),
604 
605  /* On phy intf down, delete its associated arp route and set state to resolving */
606  if (agent_->is_l3mh() && intf->type() == Interface::PHYSICAL && intf->os_oper_state() == false) {
607  InterfaceArpInfo &intf_entry = ArpMapIndexToEntry(intf->id());
608  ArpKeySet::iterator key_it = intf_entry.arp_key_list.begin();
609  while (key_it != intf_entry.arp_key_list.end()) {
610  ArpKey key = *key_it;
611  ++key_it;
612  ArpEntry *arp_entry = FindArpEntry(key);
613  if (arp_entry) {
614  arp_entry->SetState(ArpEntry::RESOLVING);
615  arp_entry->DeleteArpRoute();
616  }
617  }
618  return;
619  }
620 
621  Interface *itf = static_cast<Interface *>(entry);
622  if (entry->IsDeleted()) {
623  if (state) {
624  intf->ClearState(intf->get_table_partition()->parent(),
626  delete state;
627  }
628 
629  InterfaceArpMap::iterator it = interface_arp_map_.find(itf->id());
630  if (it != interface_arp_map_.end()) {
631  InterfaceArpInfo &intf_entry = it->second;
632  ArpKeySet::iterator key_it = intf_entry.arp_key_list.begin();
633  while (key_it != intf_entry.arp_key_list.end()) {
634  ArpKey key = *key_it;
635  ++key_it;
636  ArpIterator arp_it = arp_cache_.find(key);
637  if (arp_it != arp_cache_.end()) {
638  ArpEntry *arp_entry = arp_it->second;
639  if (arp_entry->DeleteArpRoute()) {
640  DeleteArpEntry(arp_it);
641  }
642  }
643  }
644  intf_entry.arp_key_list.clear();
645  interface_arp_map_.erase(it);
646  }
647 
648  if (itf->type() == Interface::VM_INTERFACE) {
649  ArpKey key(static_cast<VmInterface *>
650  (itf)->primary_ip_addr().to_ulong(), itf->vrf());
651  ArpEntry *arpentry = GratuitousArpEntry(key, itf);
652  if (arpentry) {
653  DeleteGratuitousArpEntry(arpentry);
654  }
655  }
656 
657  if (itf->type() == Interface::PHYSICAL &&
658  itf->name() == agent_->fabric_interface_name()) {
661  }
662  } else {
663  if (state == NULL) {
664  state = new ArpInterfaceState(intf);
665  intf->SetState(entry->get_table_partition()->parent(),
667  }
668 
669  if (itf->type() == Interface::PHYSICAL &&
670  itf->name() == agent_->fabric_interface_name()) {
673  if (run_with_vrouter_) {
675  } else {
677  }
678  }
679 
680  if (itf->type() == Interface::VM_INTERFACE) {
681  const VmInterface *vm_intf =
682  static_cast<const VmInterface *>(itf);
683  VrfEntry *forwarding_vrf = NULL;
684  if (vm_intf->vrf()) {
685  forwarding_vrf = vm_intf->vrf()->forwarding_vrf();
686  }
687  state->SetVrf(vm_intf->vrf(), forwarding_vrf);
688  }
689  }
690 }
691 
693  InterfaceArpMap::iterator it = interface_arp_map_.find(idx);
694  if (it == interface_arp_map_.end()) {
695  InterfaceArpInfo entry;
696  std::pair<InterfaceArpMap::iterator, bool> ret;
697  ret = interface_arp_map_.insert(InterfaceArpPair(idx, entry));
698  return ret.first->second;
699  } else {
700  return it->second;
701  }
702 }
703 
705  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
706  entry.stats.arp_req++;
707 }
708 
710  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
711  entry.stats.arp_replies++;
712 }
713 
715  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
716  entry.stats.resolved++;
717 }
718 
719 uint32_t ArpProto::ArpRequestStatsCounter(uint32_t idx) {
720  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
721  return entry.stats.arp_req;
722 }
723 
724 uint32_t ArpProto::ArpReplyStatsCounter(uint32_t idx) {
725  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
726  return entry.stats.arp_replies;
727 }
728 
729 uint32_t ArpProto::ArpResolvedStatsCounter(uint32_t idx) {
730  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
731  return entry.stats.resolved;
732 }
733 
735  InterfaceArpInfo &entry = ArpMapIndexToEntry(idx);
736  entry.stats.Reset();
737 }
738 
740  NextHop *nh = static_cast<NextHop *>(entry);
741 
742  switch(nh->GetType()) {
743  case NextHop::ARP: {
744  ArpNH *arp_nh = (static_cast<ArpNH *>(nh));
745  if (arp_nh->IsDeleted()) {
746  SendArpIpc(ArpProto::ARP_DELETE, arp_nh->GetIp()->to_ulong(),
747  arp_nh->GetVrf(), arp_nh->GetInterface());
748  } else if (arp_nh->IsValid() == false && arp_nh->GetInterface()) {
749  SendArpIpc(ArpProto::ARP_RESOLVE, arp_nh->GetIp()->to_ulong(),
750  arp_nh->GetVrf(), arp_nh->GetInterface());
751  }
752  break;
753  }
754 
755  default:
756  break;
757  }
758 }
759 
760 bool ArpProto::TimerExpiry(ArpKey &key, uint32_t timer_type,
761  const Interface* itf) {
762  if (arp_cache_.find(key) != arp_cache_.end() ||
763  gratuitous_arp_cache_.find(key) != gratuitous_arp_cache_.end()) {
764  if (itf) {
765  SendArpIpc((ArpProto::ArpMsgType)timer_type, key, itf);
766  }
767  }
768  return false;
769 }
770 
772  ArpEntrySet empty_set;
773  gratuitous_arp_cache_.insert(GratuitousArpCachePair(key, empty_set));
774 }
775 
777  if (!entry)
778  return ;
779 
781  if (iter == gratuitous_arp_cache_.end()) {
782  return;
783  }
784 
785  iter->second.erase(entry);
786  delete entry;
787  if (iter->second.empty()) {
788  gratuitous_arp_cache_.erase(iter);
789  }
790 }
791 
792 ArpEntry *
795  if (it == gratuitous_arp_cache_.end())
796  return NULL;
797 
798  for (ArpEntrySet::iterator sit = it->second.begin();
799  sit != it->second.end(); sit++) {
800  ArpEntry *entry = *sit;
801  if (entry->get_interface() == intf)
802  return *sit;
803  }
804 
805  return NULL;
806 }
807 
809 ArpProto::GratuitousArpEntryIterator(const ArpKey &key, bool *key_valid) {
811  if (it == gratuitous_arp_cache_.end())
812  return it;
813  const VrfEntry *vrf = key.vrf;
814  if (!vrf)
815  return it;
816  const ArpVrfState *state = static_cast<const ArpVrfState *>
817  (vrf->GetState(vrf->get_table_partition()->parent(),
819  // If VRF is delete marked, do not add ARP entries to cache
820  if (state == NULL || state->deleted == true)
821  return it;
822  *key_valid = true;
823  return it;
824 }
825 
827  const VrfEntry *vrf, InterfaceConstRef itf) {
828  ArpIpc *ipc = new ArpIpc(type, ip, vrf, itf);
830 }
831 
833  InterfaceConstRef itf) {
834  ArpIpc *ipc = new ArpIpc(type, key, itf);
836 }
837 
839  const VrfEntry *vrf = entry->key().vrf;
840  const ArpVrfState *state = static_cast<const ArpVrfState *>
841  (vrf->GetState(vrf->get_table_partition()->parent(),
843  // If VRF is delete marked, do not add ARP entries to cache
844  if (state == NULL || state->deleted == true)
845  return false;
846 
847  bool ret = arp_cache_.insert(ArpCachePair(entry->key(), entry)).second;
848  uint32_t intf_id = entry->get_interface()->id();
849  InterfaceArpMap::iterator it = interface_arp_map_.find(intf_id);
850  if (it == interface_arp_map_.end()) {
851  InterfaceArpInfo intf_entry;
852  intf_entry.arp_key_list.insert(entry->key());
853  interface_arp_map_.insert(InterfaceArpPair(intf_id, intf_entry));
854  } else {
855  InterfaceArpInfo &intf_entry = it->second;
856  ArpKeySet::iterator key_it = intf_entry.arp_key_list.find(entry->key());
857  if (key_it == intf_entry.arp_key_list.end()) {
858  intf_entry.arp_key_list.insert(entry->key());
859  }
860  }
861  return ret;
862 }
863 
865  if (!entry)
866  return false;
867 
868  ArpProto::ArpIterator iter = arp_cache_.find(entry->key());
869  if (iter == arp_cache_.end()) {
870  return false;
871  }
872 
873  DeleteArpEntry(iter);
874  return true;
875 }
876 
879  ArpEntry *entry = iter->second;
880  arp_cache_.erase(iter++);
881  delete entry;
882  return iter;
883 }
884 
886  ArpIterator it = arp_cache_.find(key);
887  if (it == arp_cache_.end())
888  return NULL;
889  return it->second;
890 }
891 
893  const ArpVrfState *vrf_state) {
894  if (!vrf_state->deleted) {
895  ARP_TRACE(Trace, "ARP state not cleared - VRF is not delete marked",
896  "", vrf->GetName(), "");
897  return false;
898  }
899 
900  if (vrf_state->l3_walk_completed() == false) {
901  return false;
902  }
903 
904  if (vrf_state->evpn_walk_completed() == false) {
905  return false;
906  }
907 
908  if (vrf_state->managed_delete_walk_ref.get() != NULL ||
909  vrf_state->evpn_walk_ref_.get() != NULL) {
910  ARP_TRACE(Trace, "ARP state not cleared - Route table walk not complete",
911  "", vrf->GetName(), "");
912  return false;
913  }
914 
915  DBState *state = static_cast<DBState *>
916  (vrf->GetState(vrf->get_table_partition()->parent(),
918  if (state) {
919  vrf->ClearState(vrf->get_table_partition()->parent(),
921  }
922  return true;
923 }
924 
927  return arp_cache_.upper_bound(key);
928 }
929 
932  return arp_cache_.lower_bound(key);
933 }
934 
936  WaitForTrafficIntfMap::iterator it = l3_wait_for_traffic_map_.find(itf);
937  if (it == l3_wait_for_traffic_map_.end()) {
938  return;
939  }
940 
941  InterfaceArpPathPreferenceInfo &data = it->second;
942  // reset arp_try_count to 0.
943  data.arp_try_count = 0;
944 
945  if (data.prev_responded_ip == sip) {
946  ++data.arp_reply_count;
947  data.arp_failure_count = 0;
948  } else {
949  data.prev_responded_ip = sip;
950  data.arp_send_count = 0;
951  }
952 }
953 
954 void ArpProto::HandlePathPreferenceArpReply(const VrfEntry *vrf, uint32_t itf,
955  Ip4Address sip) {
956  if (!vrf) {
957  return;
958  }
959  InetUnicastRouteEntry *rt = vrf->GetUcRoute(sip);
960  if (!rt) {
961  return;
962  }
963 
964  ArpVrfState *state = static_cast<ArpVrfState *>
965  (vrf->GetState(vrf->get_table_partition()->parent(),
967  if (!state) {
968  return;
969  }
970  ArpPathPreferenceState* pstate = state->Get(sip, rt->prefix_length());
971  if (!pstate) {
972  return;
973  }
974  pstate->HandleArpReply(sip, itf);
975 }
976 
978  intf_(intf), vrf_(NULL), fabric_vrf_(NULL), walk_ref_(NULL) {
979 }
980 
982  if (walk_ref_.get() && vrf_) {
983  vrf_->GetEvpnRouteTable()->ReleaseWalker(walk_ref_);
984  }
985 }
986 
987 void ArpInterfaceState::SetVrf(VrfEntry *vrf, VrfEntry *fabric_vrf) {
988  bool walk = false;
989 
990  if (vrf_ != vrf) {
991  if (walk_ref_.get() && vrf_) {
992  vrf_->GetEvpnRouteTable()->ReleaseWalker(walk_ref_);
993  }
994  vrf_ = vrf;
995  }
996 
997  if (fabric_vrf_ != fabric_vrf) {
998  fabric_vrf_ = fabric_vrf;
999  walk = true;
1000  }
1001 
1002  if (vrf_ == NULL) {
1003  return;
1004  }
1005 
1006  if (walk && walk_ref_.get() == NULL) {
1007  walk_ref_ = vrf_->GetEvpnRouteTable()->AllocWalker(
1008  boost::bind(&ArpInterfaceState::WalkNotify, this, _1, _2),
1009  boost::bind(&ArpInterfaceState::WalkDone, this, _2));
1010  }
1011 
1012  if (vrf_->vn() && vrf_->vn()->bridging()) {
1013  if (walk) {
1014  vrf_->GetEvpnRouteTable()->WalkAgain(walk_ref_);
1015  }
1016  }
1017 }
1018 
1020 }
1021 
1023  DBEntryBase *e) {
1024  const EvpnRouteEntry *evpn = static_cast<const EvpnRouteEntry *>(e);
1025 
1026  if (evpn->prefix_address().is_v4() == false) {
1027  return true;
1028  }
1029 
1030  if (evpn->prefix_address() == Ip4Address(0)) {
1031  return true;
1032  }
1033 
1034  const InterfaceNH *nh =
1035  dynamic_cast<const InterfaceNH *>(evpn->GetActiveNextHop());
1036  if (nh && nh->GetInterface() == intf_) {
1037  return true;
1038  }
1039 
1040  EvpnAgentRouteTable *table = static_cast<EvpnAgentRouteTable *>(evpn->get_table());
1041  const VmInterface *vmi = static_cast<const VmInterface *>(intf_.get());
1042  Agent *agent = table->agent();
1043  boost::shared_ptr<PktInfo> pkt(new PktInfo(agent, ARP_TX_BUFF_LEN,
1044  PktHandler::ARP, 0));
1045  ArpHandler arp_handler(agent, pkt, *(agent->event_manager()->io_service()));
1046 
1047  MacAddress smac = vmi->GetVifMac(agent);
1048  if (vrf_->forwarding_vrf() == NULL) {
1049  smac = evpn->mac();
1050  }
1051 
1053  arp_handler.SendArp(ARPOP_REQUEST, smac, evpn->prefix_address().to_v4().to_ulong(),
1054  smac, vmi->vm_mac(), evpn->prefix_address().to_v4().to_ulong(),
1055  intf_->id(), intf_->vrf()->vrf_id());
1056  return true;
1057 }
uint8_t prefix_length() const
!
uint32_t vrf_id() const
std::pair< uint32_t, InterfaceArpInfo > InterfaceArpPair
Definition: arp_proto.h:89
void SetBounded(bool bounded)
Definition: queue_task.h:200
const Interface * GetInterface() const
Definition: nexthop.h:1293
int intrusive_ptr_add_ref(const AsPath *cpath)
Definition: bgp_aspath.h:147
DBTableBase::ListenerId evpn_route_table_listener_id
Definition: arp_proto.h:373
const VnListType & dest_vn_list() const
Definition: agent_path.h:258
DBTable::DBTableWalkRef managed_delete_walk_ref
Definition: arp_proto.h:378
const MacAddress & vm_mac() const
Type type() const
Definition: interface.h:112
void Delete()
Definition: arp_proto.cc:508
void SendMessage(PktModuleName mod, InterTaskMsg *msg)
void RouteUpdate(DBTablePartBase *part, DBEntryBase *entry)
Definition: arp_proto.cc:437
std::pair< ArpPathPreferenceStateKey, ArpPathPreferenceState * > ArpPathPreferenceStatePair
Definition: arp_proto.h:342
Definition: vrf.h:86
const TagList & tag_list() const
Definition: agent_path.h:249
void VrfNotify(DBTablePartBase *part, DBEntryBase *entry)
Definition: arp_proto.cc:67
const PathPreference & path_preference() const
Definition: agent_path.h:329
const IpAddress subnet_service_ip() const
Definition: agent_path.h:274
AgentRouteTable * GetEvpnRouteTable() const
Definition: vrf.cc:330
const ArpKey & key() const
Definition: arp_entry.h:36
DBState * GetState(DBTableBase *tbl_base, ListenerId listener) const
Definition: db_entry.cc:37
bool run_with_vrouter_
Definition: arp_proto.h:197
void NextHopNotify(DBEntryBase *entry)
Definition: arp_proto.cc:739
AgentRouteTable * rt_table
Definition: arp_proto.h:370
const uint32_t id() const
Definition: interface.h:123
NextHopTable * nexthop_table() const
Definition: agent.h:475
DBTableBase * get_table() const
Definition: db_entry.cc:119
bool AddArpEntry(ArpEntry *entry)
Definition: arp_proto.cc:838
Agent supports multiple route tables - Inet-unicast (IPv4/IPv6), Inet-multicast, bridge, EVPN (Type2/Type5). This base class contains common code for all types of route tables.
Definition: agent_route.h:109
uint32_t resolved
Definition: arp_proto.h:70
void set_ip_fabric_interface_index(uint32_t ind)
Definition: arp_proto.h:115
bool IsIp4SubnetMember(const Ip4Address &ip, const Ip4Address &prefix_ip, uint16_t plen)
Definition: address_util.cc:19
bool is_l3mh() const
Definition: agent.h:725
ArpDBState(ArpVrfState *vrf_state, uint32_t vrf_id, IpAddress vm_ip_addr, uint8_t plen)
Definition: arp_proto.cc:315
bool IsDeleted() const
Definition: db_entry.h:49
void SetState(DBTableBase *tbl_base, ListenerId listener, DBState *state)
Definition: db_entry.cc:22
boost::asio::ip::address IpAddress
Definition: address.h:13
MacLearningProto * mac_learning_proto() const
Definition: agent.h:1005
static const uint16_t kMaxFailures
Definition: arp_proto.h:22
ArpVrfState(Agent *agent, ArpProto *proto, VrfEntry *vrf, AgentRouteTable *table, AgentRouteTable *evpn_table)
Definition: arp_proto.cc:578
const MacAddress & GetVifMac(const Agent *) const
ArpVrfState * vrf_state_
Definition: arp_proto.h:313
std::vector< int > SecurityGroupList
Definition: agent.h:201
DBTable::DBTableWalkRef walk_ref_
Definition: arp_proto.h:421
void Update(const AgentRoute *route)
Definition: arp_proto.cc:374
void set_ip_fabric_interface_mac(const MacAddress &mac)
Definition: arp_proto.h:118
DBTableWalkRef AllocWalker(WalkFn walk_fn, WalkCompleteFn walk_complete)
Definition: db_table.cc:613
boost::asio::io_context * io_service()
Definition: event_manager.h:42
void IncrementStatsArpReply(uint32_t idx)
Definition: arp_proto.cc:709
InetUnicastAgentRouteTable * GetInet4UnicastRouteTable() const
Definition: vrf.cc:319
DBTableBase * parent()
const ArpCache & arp_cache()
Definition: arp_proto.h:103
VnListType vn_list_
Definition: arp_proto.h:404
InterfaceTable * interface_table() const
Definition: agent.h:465
void Shutdown()
Definition: arp_proto.cc:41
VrfEntry * vrf
Definition: arp_proto.h:369
const string & GetName() const
Definition: vrf.h:100
const MacAddress & mac() const
Definition: interface.h:131
uint32_t ArpResolvedStatsCounter(uint32_t idx)
Definition: arp_proto.cc:729
void InterfaceNotify(DBEntryBase *entry)
Definition: arp_proto.cc:599
virtual ~ArpProto()
Definition: arp_proto.cc:38
bool evpn_walk_completed() const
Definition: arp_proto.h:363
VrfEntryRef fabric_vrf_
Definition: arp_proto.h:420
bool ValidateAndClearVrfState(VrfEntry *vrf, const ArpVrfState *vrf_state)
Definition: arp_proto.cc:892
ArpProto * GetArpProto() const
Definition: agent.h:978
LifetimeRef< ArpVrfState > table_delete_ref
Definition: arp_proto.h:374
InterfaceRef intf_
Definition: arp_proto.h:418
VrfEntry * vrf() const
Definition: interface.h:115
ArpProto * arp_proto
Definition: arp_proto.h:368
bool deleted
Definition: arp_proto.h:376
static const uint32_t kMaxRetry
Definition: arp_proto.h:247
Type GetType() const
Definition: nexthop.h:405
Base class for all Route entries in agent.
Definition: agent_route.h:224
void Erase(const IpAddress &ip, uint8_t plen)
Definition: arp_proto.cc:560
void SendArpRequestByPlen(const VmInterface *vm_interface, const MacAddress &smac, const ArpPathPreferenceState *data, const Ip4Address &tpa)
Definition: arp_handler.cc:392
void Unregister(ListenerId listener)
Definition: db_table.cc:186
void set_ip_fabric_interface(Interface *itf)
Definition: arp_proto.h:114
void ReleaseWalker(DBTableWalkRef &walk)
Definition: db_table.cc:619
void IncrementStatsResolved()
Definition: arp_proto.h:130
const Ip4Address * GetIp() const
Definition: nexthop.h:840
const Interface * GetInterface() const
Definition: nexthop.h:837
LifetimeRef< ArpVrfState > evpn_table_delete_ref
Definition: arp_proto.h:375
void SendArpRequestForAllIntf(const AgentRoute *route)
Definition: arp_proto.cc:254
Interface * ip_fabric_interface() const
Definition: arp_proto.h:107
ArpPathPreferenceState * Locate(const IpAddress &ip, uint8_t plen)
Definition: arp_proto.cc:547
ListenerId Register(ChangeCallback callback, const std::string &name="unspecified")
Definition: db_table.cc:181
const std::string & fabric_vrf_name() const
Definition: agent.h:903
const Type GetType() const
Definition: peer.h:87
const AgentPath * GetActivePath() const
Definition: agent_route.cc:876
void WalkAgain(DBTableWalkRef walk)
Definition: db_table.cc:631
void WalkDone(DBTableBase *part)
Definition: arp_proto.cc:1019
bool DeleteEvpnRouteState(DBTablePartBase *part, DBEntryBase *entry)
Definition: arp_proto.cc:501
std::pair< ArpKey, ArpEntry * > ArpCachePair
Definition: arp_proto.h:29
bool IsValid() const
Definition: nexthop.h:406
ArpPathPreferenceState * Get(const IpAddress ip, uint8_t plen=32)
Definition: arp_proto.cc:569
std::map< ArpKey, ArpEntrySet >::iterator GratuitousArpIterator
Definition: arp_proto.h:35
uint8_t type
Definition: load_balance.h:109
bool IsDependentRt(void) const
Definition: agent_path.h:156
void SetSize(size_t size)
Definition: queue_task.h:196
Definition: agent.h:358
uint32_t vrf_id() const
Definition: interface.cc:621
ArpIterator FindLowerBoundArpEntry(const ArpKey &key)
Definition: arp_proto.cc:931
static TaskScheduler * GetInstance()
Definition: task.cc:547
void Reset(LifetimeActor *actor)
Definition: lifetime.h:82
VrfEntryRef vrf_
Definition: arp_proto.h:419
const std::string & fabric_interface_name() const
Definition: agent.h:1129
const NextHop * GetActiveNextHop() const
Definition: agent_route.cc:881
void HandleArpReply(Ip4Address sip, uint32_t itf)
Definition: arp_proto.cc:935
Ip4Address router_id() const
Definition: agent.h:666
WaitForTrafficIntfMap l3_wait_for_traffic_map_
Definition: arp_proto.h:319
ArpProto(Agent *agent, boost::asio::io_context &io, bool run_with_vrouter)
Definition: arp_proto.cc:19
EventManager * event_manager() const
Definition: agent.h:1103
ArpVrfState * vrf_state_
Definition: arp_proto.h:399
Definition: nexthop.h:820
ArpPathPreferenceStateMap arp_path_preference_map_
Definition: arp_proto.h:379
uint32_t arp_replies
Definition: arp_proto.h:68
bool DeleteArpEntry(ArpEntry *entry)
Definition: arp_proto.cc:864
bool is_health_check_service() const
Definition: agent_path.h:380
Definition: trace.h:220
const Peer * peer() const
Definition: agent_path.h:263
bool os_oper_state() const
Definition: interface.h:132
void UpdateMac(const InterfaceNH *nh)
Definition: arp_proto.cc:326
uint32_t ArpRequestStatsCounter(uint32_t idx)
Definition: arp_proto.cc:719
void SetState(State state)
Definition: arp_entry.h:52
boost::intrusive_ptr< const Interface > InterfaceConstRef
Definition: agent.h:51
Agent * agent
Definition: arp_proto.h:367
PktHandler * pkt_handler() const
Definition: pkt_init.h:31
ArpEntry * FindArpEntry(const ArpKey &key)
Definition: arp_proto.cc:885
MacAddress mac(void) const
Definition: arp_proto.h:274
void ClearInterfaceArpStats(uint32_t idx)
Definition: arp_proto.cc:734
uint32_t ArpReplyStatsCounter(uint32_t idx)
Definition: arp_proto.cc:724
AgentParam * params() const
Definition: agent.h:1218
const Interface * get_interface() const
Definition: arp_entry.h:39
ArpInterfaceState(Interface *intf)
Definition: arp_proto.cc:977
const uint32_t vrf_id() const
Definition: vrf.h:99
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
void ClearState(DBTableBase *tbl_base, ListenerId listener)
Definition: db_entry.cc:73
const MacAddress & mac() const
InterfaceArpMap interface_arp_map_
Definition: arp_proto.h:204
void EvpnRouteUpdate(DBTablePartBase *part, DBEntryBase *entry)
Definition: arp_proto.cc:410
virtual const PrefixType & prefix_address() const
Returns the value of a stored prefix address (IPv4, IPv6 or MAC address)
Definition: agent_route.h:375
boost::asio::ip::address_v4 Ip4Address
Definition: address.h:14
bool TimerExpiry(ArpKey &key, uint32_t timer_type, const Interface *itf)
Definition: arp_proto.cc:760
uint8_t plen() const
Definition: arp_proto.h:272
bool Cancel()
Definition: timer.cc:150
#define ARP_TX_BUFF_LEN
Definition: pkt_handler.h:41
VrfTable * vrf_table() const
Definition: agent.h:485
DBTableBase::ListenerId interface_table_listener_id_
Definition: arp_proto.h:202
InterfaceArpInfo & ArpMapIndexToEntry(uint32_t idx)
Definition: arp_proto.cc:692
void DeleteGratuitousArpEntry(ArpEntry *entry)
Definition: arp_proto.cc:776
ArpPathPreferenceState(ArpVrfState *state, uint32_t vrf_id, const IpAddress &vm_ip, uint8_t plen)
Definition: arp_proto.cc:129
std::set< uint32_t > ArpTransmittedIntfMap
Definition: arp_proto.h:255
bool PreWalkDone(DBTableBase *partition)
Definition: arp_proto.cc:534
std::pair< ArpKey, ArpEntrySet > GratuitousArpCachePair
Definition: arp_proto.h:34
#define ARP_TRACE(obj,...)
Definition: arp_proto.h:12
uint32_t services_queue_limit()
Definition: agent_param.h:417
const MacAddress & GetDMac() const
Definition: nexthop.h:1294
uint8_t prefix_length() const
!
tbb::atomic< int > refcount_
Definition: arp_proto.h:321
const IpAddress & ip() const
Definition: arp_proto.h:271
VmInterface::VmiType vmi_type() const
ProtoHandler * AllocProtoHandler(boost::shared_ptr< PktInfo > info, boost::asio::io_context &io)
Definition: arp_proto.cc:62
bool DeleteArpRoute()
Definition: arp_entry.cc:312
const VrfEntry * GetVrf() const
Definition: nexthop.h:841
bool resolve_route_
Definition: arp_proto.h:403
bool evpn_walk_completed_
Definition: arp_proto.h:381
bool Start(int time, Handler handler, ErrorHandler error_handler=NULL)
Definition: timer.cc:108
bool WalkNotify(DBTablePartBase *partition, DBEntryBase *entry)
Definition: arp_proto.cc:1022
void IncrementStatsVmGarpReq()
Definition: arp_proto.h:133
static const uint32_t kArpTryCount
Definition: arp_proto.h:250
const Interface * FindInterface(size_t index) const
Definition: interface.cc:323
VrfEntry * forwarding_vrf() const
Definition: vrf.h:217
void AddGratuitousArpEntry(ArpKey &key)
Definition: arp_proto.cc:771
VrfEntry * vrf() const
Definition: agent_route.h:275
uint32_t arp_req
Definition: arp_proto.h:67
void intrusive_ptr_release(const AsPath *cpath)
Definition: bgp_aspath.h:155
Agent * agent() const
Definition: agent_route.h:159
void IncrementStatsArpRequest(uint32_t idx)
Definition: arp_proto.cc:704
TagList tag_list_
Definition: arp_proto.h:401
const std::string & name() const
Definition: interface.h:114
void UpdateArpRoutes(const InetUnicastRouteEntry *route)
Definition: arp_proto.cc:336
std::map< uint32_t, InterfaceArpPathPreferenceInfo > WaitForTrafficIntfMap
Definition: arp_proto.h:252
DBTableBase::ListenerId route_table_listener_id
Definition: arp_proto.h:372
void SendArp(uint16_t op, const MacAddress &smac, in_addr_t sip, const MacAddress &tmac, const MacAddress &dmac, in_addr_t tip, uint32_t itf, uint32_t vrf)
Definition: arp_handler.cc:369
bool PolicyEnabled() const
Definition: nexthop.h:407
void SetVrf(VrfEntry *vrf, VrfEntry *fabric_vrf)
Definition: arp_proto.cc:987
virtual const NextHop * ComputeNextHop(Agent *agent) const
Definition: agent_path.cc:91
ArpProto::GratuitousArpIterator GratuitousArpEntryIterator(const ArpKey &key, bool *key_valid)
Definition: arp_proto.cc:809
std::map< ArpKey, ArpEntry * >::iterator ArpIterator
Definition: arp_proto.h:30
void SendArpIpc(ArpProto::ArpMsgType type, in_addr_t ip, const VrfEntry *vrf, InterfaceConstRef itf)
Definition: arp_proto.cc:826
DBTablePartBase * get_table_partition() const
Definition: db_entry.cc:115
const VrfEntry * vrf
Definition: arp_entry.h:18
bool Reschedule(int time)
Definition: timer.cc:137
static const uint32_t kTimeout
Definition: arp_proto.h:249
AgentRouteTable * evpn_rt_table
Definition: arp_proto.h:371
WaitForTrafficIntfMap evpn_wait_for_traffic_map_
Definition: arp_proto.h:320
bool l3_walk_completed_
Definition: arp_proto.h:380
PktModule * pkt() const
Definition: agent.cc:965
std::set< ArpEntry * > ArpEntrySet
Definition: arp_proto.h:32
InetUnicastRouteEntry * GetUcRoute(const IpAddress &addr) const
Definition: vrf.cc:237
DBTableBase::ListenerId vrf_table_listener_id_
Definition: arp_proto.h:201
ArpPathPreferenceStatePtr arp_path_preference_state_
Definition: arp_proto.h:405
ArpEntry * GratuitousArpEntry(const ArpKey &key, const Interface *intf)
Definition: arp_proto.cc:793
ArpIterator FindUpperBoundArpEntry(const ArpKey &key)
Definition: arp_proto.cc:926
void HandlePathPreferenceArpReply(const VrfEntry *vrf, uint32_t itf, Ip4Address sip)
Definition: arp_proto.cc:954
SecurityGroupList sg_list_
Definition: arp_proto.h:400
void Delete(const InetUnicastRouteEntry *rt)
Definition: arp_proto.cc:354
ArpVrfState * vrf_state()
Definition: arp_proto.h:267
const SecurityGroupList & sg_list() const
Definition: agent_path.h:248
bool DeleteRouteState(DBTablePartBase *part, DBEntryBase *entry)
Definition: arp_proto.cc:496
ArpCache arp_cache_
Definition: arp_proto.h:194
static void WalkDone(DBTableBase *partition, ArpVrfState *state)
Definition: arp_proto.cc:518
GratuitousArpCache gratuitous_arp_cache_
Definition: arp_proto.h:196
static bool DeleteTimer(Timer *Timer)
Definition: timer.cc:222
const PathList & GetPathList() const
Definition: route.h:46
DBTableBase::ListenerId nexthop_table_listener_id_
Definition: arp_proto.h:203
DBTable::DBTableWalkRef evpn_walk_ref_
Definition: arp_proto.h:377
std::pair< uint32_t, InterfaceArpPathPreferenceInfo > WaitForTrafficIntfPair
Definition: arp_proto.h:254
std::vector< int > TagList
Definition: agent.h:202
bool l3_walk_completed() const
Definition: arp_proto.h:359
bool policy_
Definition: arp_proto.h:402