From 9f30ca1dbc783c5cba761a03aca50987c0f3c31f Mon Sep 17 00:00:00 2001 From: Sudharsan Dhamal Gopalarathnam Date: Mon, 1 Nov 2021 08:58:59 -0700 Subject: [PATCH] VxLAN Tunnel Counters and Rates implementation (#1859) * Tunnel stats support --- orchagent/Makefile.am | 3 +- .../flex_counter/flex_counter_manager.cpp | 46 +++++- orchagent/flex_counter/flex_counter_manager.h | 40 ++++- orchagent/flexcounterorch.cpp | 9 ++ orchagent/tunnel_rates.lua | 98 ++++++++++++ orchagent/vxlanorch.cpp | 143 ++++++++++++++++++ orchagent/vxlanorch.h | 18 +++ tests/test_flex_counters.py | 38 ++++- 8 files changed, 391 insertions(+), 4 deletions(-) create mode 100644 orchagent/tunnel_rates.lua diff --git a/orchagent/Makefile.am b/orchagent/Makefile.am index ce56e8438ec8..df3c432aaf00 100644 --- a/orchagent/Makefile.am +++ b/orchagent/Makefile.am @@ -23,7 +23,8 @@ dist_swss_DATA = \ watermark_queue.lua \ watermark_pg.lua \ watermark_bufferpool.lua \ - lagids.lua + lagids.lua \ + tunnel_rates.lua bin_PROGRAMS = orchagent routeresync orchagent_restart_check diff --git a/orchagent/flex_counter/flex_counter_manager.cpp b/orchagent/flex_counter/flex_counter_manager.cpp index 299e238d37cc..c924b269e85c 100644 --- a/orchagent/flex_counter/flex_counter_manager.cpp +++ b/orchagent/flex_counter/flex_counter_manager.cpp @@ -39,17 +39,56 @@ const unordered_map FlexCounterManager::counter_id_field_lo { CounterType::PORT, PORT_COUNTER_ID_LIST }, { CounterType::QUEUE, QUEUE_COUNTER_ID_LIST }, { CounterType::MACSEC_SA_ATTR, MACSEC_SA_ATTR_ID_LIST }, + { CounterType::TUNNEL, TUNNEL_COUNTER_ID_LIST }, }; +FlexManagerDirectory g_FlexManagerDirectory; + +FlexCounterManager *FlexManagerDirectory::createFlexCounterManager(const string& group_name, + const StatsMode stats_mode, + const uint polling_interval, + const bool enabled, + FieldValueTuple fv_plugin) +{ + if (m_managers.find(group_name) != m_managers.end()) + { + if (stats_mode != m_managers[group_name]->getStatsMode()) + { + SWSS_LOG_ERROR("Stats mode mismatch with already created flex counter manager %s", + group_name.c_str()); + return NULL; + } + if (polling_interval != m_managers[group_name]->getPollingInterval()) + { + SWSS_LOG_ERROR("Polling interval mismatch with already created flex counter manager %s", + group_name.c_str()); + return NULL; + } + if (enabled != m_managers[group_name]->getEnabled()) + { + SWSS_LOG_ERROR("Enabled field mismatch with already created flex counter manager %s", + group_name.c_str()); + return NULL; + } + return m_managers[group_name]; + } + FlexCounterManager *fc_manager = new FlexCounterManager(group_name, stats_mode, polling_interval, + enabled, fv_plugin); + m_managers[group_name] = fc_manager; + return fc_manager; +} + FlexCounterManager::FlexCounterManager( const string& group_name, const StatsMode stats_mode, const uint polling_interval, - const bool enabled) : + const bool enabled, + FieldValueTuple fv_plugin) : group_name(group_name), stats_mode(stats_mode), polling_interval(polling_interval), enabled(enabled), + fv_plugin(fv_plugin), flex_counter_db(new DBConnector("FLEX_COUNTER_DB", 0)), flex_counter_group_table(new ProducerTable(flex_counter_db.get(), FLEX_COUNTER_GROUP_TABLE)), flex_counter_table(new ProducerTable(flex_counter_db.get(), FLEX_COUNTER_TABLE)) @@ -86,6 +125,11 @@ void FlexCounterManager::applyGroupConfiguration() FieldValueTuple(FLEX_COUNTER_STATUS_FIELD, status_lookup.at(enabled)) }; + if (!fvField(fv_plugin).empty()) + { + field_values.emplace_back(fv_plugin); + } + flex_counter_group_table->set(group_name, field_values); } diff --git a/orchagent/flex_counter/flex_counter_manager.h b/orchagent/flex_counter/flex_counter_manager.h index 4df99c90bddd..2d531d508442 100644 --- a/orchagent/flex_counter/flex_counter_manager.h +++ b/orchagent/flex_counter/flex_counter_manager.h @@ -4,8 +4,10 @@ #include #include #include +#include #include "dbconnector.h" #include "producertable.h" +#include "table.h" #include extern "C" { @@ -24,6 +26,7 @@ enum class CounterType PORT_DEBUG, SWITCH_DEBUG, MACSEC_SA_ATTR, + TUNNEL, }; // FlexCounterManager allows users to manage a group of flex counters. @@ -38,7 +41,11 @@ class FlexCounterManager const std::string& group_name, const StatsMode stats_mode, const uint polling_interval, - const bool enabled); + const bool enabled, + swss::FieldValueTuple fv_plugin = std::make_pair("","")); + + FlexCounterManager() + {} FlexCounterManager(const FlexCounterManager&) = delete; FlexCounterManager& operator=(const FlexCounterManager&) = delete; @@ -54,6 +61,26 @@ class FlexCounterManager const std::unordered_set& counter_stats); void clearCounterIdList(const sai_object_id_t object_id); + const std::string& getGroupName() const + { + return group_name; + } + + const StatsMode& getStatsMode() const + { + return stats_mode; + } + + const uint& getPollingInterval() const + { + return polling_interval; + } + + const bool& getEnabled() const + { + return enabled; + } + protected: void applyGroupConfiguration(); @@ -68,6 +95,7 @@ class FlexCounterManager StatsMode stats_mode; uint polling_interval; bool enabled; + swss::FieldValueTuple fv_plugin; std::unordered_set installed_counters; std::shared_ptr flex_counter_db = nullptr; @@ -79,4 +107,14 @@ class FlexCounterManager static const std::unordered_map counter_id_field_lookup; }; +class FlexManagerDirectory +{ + public: + FlexCounterManager* createFlexCounterManager(const std::string& group_name, const StatsMode stats_mode, + const uint polling_interval, const bool enabled, + swss::FieldValueTuple fv_plugin = std::make_pair("","")); + private: + std::unordered_map m_managers; +}; + #endif // ORCHAGENT_FLEX_COUNTER_MANAGER_H diff --git a/orchagent/flexcounterorch.cpp b/orchagent/flexcounterorch.cpp index 78368508a525..83e084415596 100644 --- a/orchagent/flexcounterorch.cpp +++ b/orchagent/flexcounterorch.cpp @@ -9,6 +9,7 @@ #include "bufferorch.h" #include "flexcounterorch.h" #include "debugcounterorch.h" +#include "directory.h" extern sai_port_api_t *sai_port_api; @@ -16,6 +17,7 @@ extern PortsOrch *gPortsOrch; extern FabricPortsOrch *gFabricPortsOrch; extern IntfsOrch *gIntfsOrch; extern BufferOrch *gBufferOrch; +extern Directory gDirectory; #define BUFFER_POOL_WATERMARK_KEY "BUFFER_POOL_WATERMARK" #define PORT_KEY "PORT" @@ -23,6 +25,7 @@ extern BufferOrch *gBufferOrch; #define QUEUE_KEY "QUEUE" #define PG_WATERMARK_KEY "PG_WATERMARK" #define RIF_KEY "RIF" +#define TUNNEL_KEY "TUNNEL" unordered_map flexCounterGroupMap = { @@ -38,6 +41,7 @@ unordered_map flexCounterGroupMap = {"RIF", RIF_STAT_COUNTER_FLEX_COUNTER_GROUP}, {"RIF_RATES", RIF_RATE_COUNTER_FLEX_COUNTER_GROUP}, {"DEBUG_COUNTER", DEBUG_COUNTER_FLEX_COUNTER_GROUP}, + {"TUNNEL", TUNNEL_STAT_COUNTER_FLEX_COUNTER_GROUP}, }; @@ -58,6 +62,7 @@ void FlexCounterOrch::doTask(Consumer &consumer) { SWSS_LOG_ENTER(); + VxlanTunnelOrch* vxlan_tunnel_orch = gDirectory.get(); if (gPortsOrch && !gPortsOrch->allPortsReady()) { return; @@ -147,6 +152,10 @@ void FlexCounterOrch::doTask(Consumer &consumer) { gFabricPortsOrch->generateQueueStats(); } + if (vxlan_tunnel_orch && (key== TUNNEL_KEY) && (value == "enable")) + { + vxlan_tunnel_orch->generateTunnelCounterMap(); + } vector fieldValues; fieldValues.emplace_back(FLEX_COUNTER_STATUS_FIELD, value); m_flexCounterGroupTable->set(flexCounterGroupMap[key], fieldValues); diff --git a/orchagent/tunnel_rates.lua b/orchagent/tunnel_rates.lua new file mode 100644 index 000000000000..3b536c29acee --- /dev/null +++ b/orchagent/tunnel_rates.lua @@ -0,0 +1,98 @@ +-- KEYS - rif IDs +-- ARGV[1] - counters db index +-- ARGV[2] - counters table name +-- ARGV[3] - poll time interval +-- return log + +local logtable = {} + +local function logit(msg) + logtable[#logtable+1] = tostring(msg) +end + +local counters_db = ARGV[1] +local counters_table_name = ARGV[2] +local rates_table_name = "RATES" +local sec_to_ms = 1000 + +-- Get configuration +redis.call('SELECT', counters_db) +local smooth_interval = redis.call('HGET', rates_table_name .. ':' .. 'TUNNEL', 'TUNNEL_SMOOTH_INTERVAL') +local alpha = redis.call('HGET', rates_table_name .. ':' .. 'TUNNEL', 'TUNNEL_ALPHA') +if not alpha then + logit("Alpha is not defined") + return logtable +end +local one_minus_alpha = 1.0 - alpha +local delta = tonumber(ARGV[3]) + +local n = table.getn(KEYS) +for i = 1, n do + local state_table = rates_table_name .. ':' .. KEYS[i] .. ':' .. 'TUNNEL' + local initialized = redis.call('HGET', state_table, 'INIT_DONE') + logit(initialized) + + -- Get new COUNTERS values + local in_octets = 0 + local in_packets = 0 + local out_octets = 0 + local out_packets = 0 + + if redis.call('HEXISTS', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_OCTETS') == 1 then + in_octets = redis.call('HGET', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_OCTETS') + end + if redis.call('HEXISTS', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_PACKETS') == 1 then + in_packets = redis.call('HGET', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_PACKETS') + end + if redis.call('HEXISTS', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_OCTETS') == 1 then + out_octets = redis.call('HGET', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_OCTETS') + end + if redis.call('HEXISTS', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_PACKETS') == 1 then + out_packets = redis.call('HGET', counters_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_PACKETS') + end + + if initialized == "DONE" or initialized == "COUNTERS_LAST" then + -- Get old COUNTERS values + local in_octets_last = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_OCTETS_last') + local in_packets_last = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_PACKETS_last') + local out_octets_last = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_OCTETS_last') + local out_packets_last = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_PACKETS_last') + + -- Calculate new rates values + local rx_bps_new = (in_octets - in_octets_last)*sec_to_ms/delta + local tx_bps_new = (out_octets - out_octets_last)*sec_to_ms/delta + local rx_pps_new = (in_packets - in_packets_last)*sec_to_ms/delta + local tx_pps_new = (out_packets - out_packets_last)*sec_to_ms/delta + + if initialized == "DONE" then + -- Get old rates values + local rx_bps_old = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'RX_BPS') + local rx_pps_old = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'RX_PPS') + local tx_bps_old = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'TX_BPS') + local tx_pps_old = redis.call('HGET', rates_table_name .. ':' .. KEYS[i], 'TX_PPS') + + -- Smooth the rates values and store them in DB + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'RX_BPS', alpha*rx_bps_new + one_minus_alpha*rx_bps_old) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'RX_PPS', alpha*rx_pps_new + one_minus_alpha*rx_pps_old) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'TX_BPS', alpha*tx_bps_new + one_minus_alpha*tx_bps_old) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'TX_PPS', alpha*tx_pps_new + one_minus_alpha*tx_pps_old) + else + -- Store unsmoothed initial rates values in DB + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'RX_BPS', rx_bps_new) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'RX_PPS', rx_pps_new) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'TX_BPS', tx_bps_new) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'TX_PPS', tx_pps_new) + redis.call('HSET', state_table, 'INIT_DONE', 'DONE') + end + else + redis.call('HSET', state_table, 'INIT_DONE', 'COUNTERS_LAST') + end + + -- Set old COUNTERS values + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_OCTETS_last', in_octets) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_IN_PACKETS_last', in_packets) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_OCTETS_last', out_octets) + redis.call('HSET', rates_table_name .. ':' .. KEYS[i], 'SAI_TUNNEL_STAT_OUT_PACKETS_last', out_packets) +end + +return logtable diff --git a/orchagent/vxlanorch.cpp b/orchagent/vxlanorch.cpp index 0123ae272d29..fc6a505a1f18 100644 --- a/orchagent/vxlanorch.cpp +++ b/orchagent/vxlanorch.cpp @@ -17,6 +17,8 @@ extern "C" { #include "swssnet.h" #include "warm_restart.h" #include "tokenize.h" +#include "sai_serialize.h" +#include "flex_counter_manager.h" #include "converter.h" /* Global variables */ @@ -27,6 +29,9 @@ extern sai_next_hop_api_t *sai_next_hop_api; extern Directory gDirectory; extern PortsOrch* gPortsOrch; extern sai_object_id_t gUnderlayIfId; +extern FlexManagerDirectory g_FlexManagerDirectory; + +#define FLEX_COUNTER_UPD_INTERVAL 1 const map vxlanTunnelMap = { @@ -60,6 +65,14 @@ const map> vxlanTunnelMapKeyVal = }, }; +const vector tunnel_stat_ids = +{ + SAI_TUNNEL_STAT_IN_OCTETS, + SAI_TUNNEL_STAT_IN_PACKETS, + SAI_TUNNEL_STAT_OUT_OCTETS, + SAI_TUNNEL_STAT_OUT_PACKETS +}; + /* * Manipulators for the above Map */ @@ -485,6 +498,7 @@ bool VxlanTunnel::createTunnel(MAP_T encap, MAP_T decap, uint8_t encap_ttl) { try { + VxlanTunnelOrch* tunnel_orch = gDirectory.get(); sai_ip_address_t ips, ipd, *ip=nullptr; uint8_t mapper_list = 0; swss::copy(ips, src_ip_); @@ -514,6 +528,11 @@ bool VxlanTunnel::createTunnel(MAP_T encap, MAP_T decap, uint8_t encap_ttl) ids_.tunnel_id = create_tunnel(&ids_, ip, NULL, gUnderlayIfId, false, encap_ttl); + if (ids_.tunnel_id != SAI_NULL_OBJECT_ID) + { + tunnel_orch->addTunnelToFlexCounter(ids_.tunnel_id, tunnel_name_); + } + ip = nullptr; if (!dst_ip_.isZero()) { @@ -843,11 +862,14 @@ bool VxlanTunnel::deleteTunnelHw(uint8_t mapper_list, tunnel_map_use_t map_src, { try { + VxlanTunnelOrch* tunnel_orch = gDirectory.get(); + if (with_term) { remove_tunnel_termination(ids_.tunnel_term_id); } + tunnel_orch->removeTunnelFromFlexCounter(ids_.tunnel_id, tunnel_name_); remove_tunnel(ids_.tunnel_id); deleteMapperHw(mapper_list, map_src); } @@ -872,6 +894,7 @@ bool VxlanTunnel::createTunnelHw(uint8_t mapper_list, tunnel_map_use_t map_src, try { + VxlanTunnelOrch* tunnel_orch = gDirectory.get(); sai_ip_address_t ips, ipd, *ip=nullptr; swss::copy(ips, src_ip_); @@ -888,6 +911,11 @@ bool VxlanTunnel::createTunnelHw(uint8_t mapper_list, tunnel_map_use_t map_src, ids_.tunnel_id = create_tunnel(&ids_, &ips, ip, gUnderlayIfId, p2p, encap_ttl); + if (ids_.tunnel_id != SAI_NULL_OBJECT_ID) + { + tunnel_orch->addTunnelToFlexCounter(ids_.tunnel_id, tunnel_name_); + } + if (with_term) { ids_.tunnel_term_id = create_tunnel_termination(ids_.tunnel_id, ips, @@ -1229,6 +1257,120 @@ VxlanTunnelOrch::VxlanTunnelOrch(DBConnector *statedb, DBConnector *db, const st } } } + + FieldValueTuple fv; + string tunnel_rate_plugin = "tunnel_rates.lua"; + m_counter_db = shared_ptr(new DBConnector("COUNTERS_DB", 0)); + m_asic_db = shared_ptr(new DBConnector("ASIC_DB", 0)); + try + { + string tunnel_rate_script = swss::loadLuaScript(tunnel_rate_plugin); + string tunnel_rate_sha = swss::loadRedisScript(m_counter_db.get(), tunnel_rate_script); + fv = FieldValueTuple(TUNNEL_PLUGIN_FIELD, tunnel_rate_sha); + } + catch (const runtime_error &e) + { + SWSS_LOG_WARN("Tunnel flex counter group plugins was not set successfully: %s", e.what()); + } + + tunnel_stat_manager = g_FlexManagerDirectory.createFlexCounterManager(TUNNEL_STAT_COUNTER_FLEX_COUNTER_GROUP, + StatsMode::READ, TUNNEL_STAT_FLEX_COUNTER_POLLING_INTERVAL_MS, false, fv); + + m_tunnelNameTable = unique_ptr(new Table(m_counter_db.get(), COUNTERS_TUNNEL_NAME_MAP)); + m_tunnelTypeTable = unique_ptr
(new Table(m_counter_db.get(), COUNTERS_TUNNEL_TYPE_MAP)); + + m_vidToRidTable = unique_ptr
(new Table(m_asic_db.get(), "VIDTORID")); + + auto intervT = timespec { .tv_sec = FLEX_COUNTER_UPD_INTERVAL , .tv_nsec = 0 }; + m_FlexCounterUpdTimer = new SelectableTimer(intervT); + auto executorT = new ExecutableTimer(m_FlexCounterUpdTimer, this, "FLEX_COUNTER_UPD_TIMER"); + Orch::addExecutor(executorT); + +} + +void VxlanTunnelOrch::doTask(SelectableTimer &timer) +{ + SWSS_LOG_ENTER(); + + for (auto it = m_pendingAddToFlexCntr.begin(); it != m_pendingAddToFlexCntr.end(); ) + { + string value; + const auto id = sai_serialize_object_id(it->first); + + if (m_vidToRidTable->hget("", id, value)) + { + SWSS_LOG_INFO("Registering %s, id %s", it->second.c_str(), id.c_str()); + vector tunnelNameFvs; + vector tunnelTypeFvs; + string type = "SAI_TUNNEL_TYPE_VXLAN"; + + tunnelNameFvs.emplace_back(it->second, id); + tunnelTypeFvs.emplace_back(id, type); + + m_tunnelNameTable->set("", tunnelNameFvs); + m_tunnelTypeTable->set("", tunnelTypeFvs); + auto tunnel_stats = generateTunnelCounterStats(); + + tunnel_stat_manager->setCounterIdList(it->first, CounterType::TUNNEL, + tunnel_stats); + it = m_pendingAddToFlexCntr.erase(it); + } + else + { + ++it; + } + } +} +void VxlanTunnelOrch::addTunnelToFlexCounter(sai_object_id_t oid, const string &name) +{ + m_pendingAddToFlexCntr[oid] = name; +} + +void VxlanTunnelOrch::removeTunnelFromFlexCounter(sai_object_id_t oid, const string &name) +{ + SWSS_LOG_ENTER(); + + if (oid == SAI_NULL_OBJECT_ID) + { + SWSS_LOG_WARN("Not removing NULL OID to flex for tunnel %s", name.c_str()); + return; + } + + if (m_pendingAddToFlexCntr.find(oid) != m_pendingAddToFlexCntr.end()) + { + m_pendingAddToFlexCntr.erase(oid); + return; + } + + string sai_oid = sai_serialize_object_id(oid); + + m_tunnelNameTable->hdel("", name); + m_tunnelTypeTable->hdel("", sai_oid); + tunnel_stat_manager->clearCounterIdList(oid); + SWSS_LOG_DEBUG("Unregistered tunnel %s to Flex counter", name.c_str()); +} + +std::unordered_set VxlanTunnelOrch::generateTunnelCounterStats() +{ + std::unordered_set counter_stats; + + for (const auto& it: tunnel_stat_ids) + { + counter_stats.emplace(sai_serialize_tunnel_stat(it)); + } + return counter_stats; +} + +void VxlanTunnelOrch::generateTunnelCounterMap() +{ + if (m_isTunnelCounterMapGenerated) + { + return; + } + + m_FlexCounterUpdTimer->start(); + + m_isTunnelCounterMapGenerated = true; } sai_object_id_t @@ -1414,6 +1556,7 @@ bool VxlanTunnelOrch::removeVxlanTunnelMap(string tunnelName, uint32_t vni) auto tunnel_id = vxlan_tunnel_table_[tunnelName].get()->getTunnelId(); try { + removeTunnelFromFlexCounter(tunnel_id, tunnelName); remove_tunnel(tunnel_id); } catch(const std::runtime_error& error) diff --git a/orchagent/vxlanorch.h b/orchagent/vxlanorch.h index bd7bc4fabf08..0b56e76faa56 100644 --- a/orchagent/vxlanorch.h +++ b/orchagent/vxlanorch.h @@ -7,6 +7,7 @@ #include "request_parser.h" #include "portsorch.h" #include "vrforch.h" +#include "timer.h" enum class MAP_T { @@ -35,6 +36,8 @@ typedef enum #define IS_TUNNELMAP_SET_VRF(x) ((x)& (1< generateTunnelCounterStats(); + void generateTunnelCounterMap(); + void addTunnelToFlexCounter(sai_object_id_t oid, const std::string &name); + void removeTunnelFromFlexCounter(sai_object_id_t oid, const std::string &name); bool isDipTunnelsSupported(void) { return is_dip_tunnel_supported; @@ -358,12 +365,23 @@ class VxlanTunnelOrch : public Orch2 private: virtual bool addOperation(const Request& request); virtual bool delOperation(const Request& request); + void doTask(swss::SelectableTimer&); VxlanTunnelTable vxlan_tunnel_table_; VxlanTunnelRequest request_; VxlanVniVlanMapTable vxlan_vni_vlan_map_table_; VTEPTable vtep_table_; Table m_stateVxlanTable; + std::map m_pendingAddToFlexCntr; + FlexCounterManager vxlan_tunnel_stat_manager; + bool m_isTunnelCounterMapGenerated = false; + FlexCounterManager *tunnel_stat_manager; + unique_ptr
m_tunnelNameTable; + unique_ptr
m_tunnelTypeTable; + unique_ptr
m_vidToRidTable; + shared_ptr m_counter_db; + shared_ptr m_asic_db; + SelectableTimer* m_FlexCounterUpdTimer = nullptr; bool is_dip_tunnel_supported; }; diff --git a/tests/test_flex_counters.py b/tests/test_flex_counters.py index ecdc844572ed..6fe3e67500ee 100644 --- a/tests/test_flex_counters.py +++ b/tests/test_flex_counters.py @@ -8,6 +8,7 @@ BUFFER_POOL_WATERMARK_KEY = "BUFFER_POOL_WATERMARK" PORT_BUFFER_DROP_KEY = "PORT_BUFFER_DROP" PG_WATERMARK_KEY = "PG_WATERMARK" +TUNNEL_KEY = "TUNNEL" # Counter stats on FlexCountersDB PORT_STAT = "PORT_STAT_COUNTER" @@ -16,6 +17,7 @@ BUFFER_POOL_WATERMARK_STAT = "BUFFER_POOL_WATERMARK_STAT_COUNTER" PORT_BUFFER_DROP_STAT = "PORT_BUFFER_DROP_STAT" PG_WATERMARK_STAT = "PG_WATERMARK_STAT_COUNTER" +TUNNEL_STAT = "TUNNEL_STAT_COUNTER" # Counter maps on CountersDB PORT_MAP = "COUNTERS_PORT_NAME_MAP" @@ -24,7 +26,9 @@ BUFFER_POOL_WATERMARK_MAP = "COUNTERS_BUFFER_POOL_NAME_MAP" PORT_BUFFER_DROP_MAP = "COUNTERS_PORT_NAME_MAP" PG_WATERMARK_MAP = "COUNTERS_PG_NAME_MAP" +TUNNEL_MAP = "COUNTERS_TUNNEL_NAME_MAP" +TUNNEL_TYPE_MAP = "COUNTERS_TUNNEL_TYPE_MAP" NUMBER_OF_RETRIES = 10 CPU_PORT_OID = "0x0" @@ -33,7 +37,9 @@ "rif_counter":[RIF_KEY, RIF_STAT, RIF_MAP], "buffer_pool_watermark_counter":[BUFFER_POOL_WATERMARK_KEY, BUFFER_POOL_WATERMARK_STAT, BUFFER_POOL_WATERMARK_MAP], "port_buffer_drop_counter":[PORT_BUFFER_DROP_KEY, PORT_BUFFER_DROP_STAT, PORT_BUFFER_DROP_MAP], - "pg_watermark_counter":[PG_WATERMARK_KEY, PG_WATERMARK_STAT, PG_WATERMARK_MAP]} + "pg_watermark_counter":[PG_WATERMARK_KEY, PG_WATERMARK_STAT, PG_WATERMARK_MAP], + "vxlan_tunnel_counter":[TUNNEL_KEY, TUNNEL_STAT, TUNNEL_MAP]} + class TestFlexCounters(object): @@ -66,6 +72,15 @@ def verify_no_flex_counters_tables(self, counter_stat): counters_stat_keys = self.flex_db.get_keys("FLEX_COUNTER_TABLE:" + counter_stat) assert len(counters_stat_keys) == 0, "FLEX_COUNTER_TABLE:" + str(counter_stat) + " tables exist before enabling the flex counter group" + def verify_no_flex_counters_tables_after_delete(self, counter_stat): + for retry in range(NUMBER_OF_RETRIES): + counters_stat_keys = self.flex_db.get_keys("FLEX_COUNTER_TABLE:" + counter_stat + ":") + if len(counters_stat_keys) == 0: + return + else: + time.sleep(1) + assert False, "FLEX_COUNTER_TABLE:" + str(counter_stat) + " tables exist after removing the entries" + def verify_flex_counters_populated(self, map, stat): counters_keys = self.counters_db.db_connection.hgetall(map) for counter_entry in counters_keys.items(): @@ -73,6 +88,14 @@ def verify_flex_counters_populated(self, map, stat): oid = counter_entry[1] self.wait_for_id_list(stat, name, oid) + def verify_tunnel_type_vxlan(self, name_map, type_map): + counters_keys = self.counters_db.db_connection.hgetall(name_map) + for counter_entry in counters_keys.items(): + oid = counter_entry[1] + fvs = self.counters_db.get_entry(type_map, "") + assert fvs != {} + assert fvs.get(oid) == "SAI_TUNNEL_TYPE_VXLAN" + def verify_only_phy_ports_created(self): port_counters_keys = self.counters_db.db_connection.hgetall(PORT_MAP) port_counters_stat_keys = self.flex_db.get_keys("FLEX_COUNTER_TABLE:" + PORT_STAT) @@ -102,6 +125,12 @@ def test_flex_counters(self, dvs, counter_type): self.config_db.db_connection.hset('INTERFACE|Ethernet0', "NULL", "NULL") self.config_db.db_connection.hset('INTERFACE|Ethernet0|192.168.0.1/24', "NULL", "NULL") + if counter_type == "vxlan_tunnel_counter": + self.config_db.db_connection.hset("VLAN|Vlan10", "vlanid", "10") + self.config_db.db_connection.hset("VXLAN_TUNNEL|vtep1", "src_ip", "1.1.1.1") + self.config_db.db_connection.hset("VXLAN_TUNNEL_MAP|vtep1|map_100_Vlan10", "vlan", "Vlan10") + self.config_db.db_connection.hset("VXLAN_TUNNEL_MAP|vtep1|map_100_Vlan10", "vni", "100") + self.enable_flex_counter_group(counter_key, counter_map) self.verify_flex_counters_populated(counter_map, counter_stat) @@ -110,3 +139,10 @@ def test_flex_counters(self, dvs, counter_type): if counter_type == "rif_counter": self.config_db.db_connection.hdel('INTERFACE|Ethernet0|192.168.0.1/24', "NULL") + + if counter_type == "vxlan_tunnel_counter": + self.verify_tunnel_type_vxlan(counter_map, TUNNEL_TYPE_MAP) + self.config_db.delete_entry("VLAN","Vlan10") + self.config_db.delete_entry("VLAN_TUNNEL","vtep1") + self.config_db.delete_entry("VLAN_TUNNEL_MAP","vtep1|map_100_Vlan10") + self.verify_no_flex_counters_tables_after_delete(counter_stat)