forked from sonic-net/sonic-swss
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
This code is to add support for fabric asic or NPU with fabric port enabled. What it does includes: 1. Create FabricOrchDaemon for fabric asic 2. Create FabricPortsOrch to manage fabric ports 3. Fabric port and queue stats are collected via FlexCounters The code is waiting for sonic-net#1431 to start fabric flow for fabric asic or NPU with enabled fabric ports.
- Loading branch information
Showing
8 changed files
with
493 additions
and
12 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,338 @@ | ||
#include "fabricportsorch.h" | ||
|
||
#include <inttypes.h> | ||
#include <fstream> | ||
#include <sstream> | ||
#include <tuple> | ||
|
||
#include "logger.h" | ||
#include "schema.h" | ||
#include "sai_serialize.h" | ||
#include "timer.h" | ||
|
||
#define FABRIC_POLLING_INTERVAL_DEFAULT (30) | ||
#define FABRIC_PORT_ERROR 0 | ||
#define FABRIC_PORT_SUCCESS 1 | ||
#define FABRIC_PORT_STAT_COUNTER_FLEX_COUNTER_GROUP "FABRIC_PORT_STAT_COUNTER" | ||
#define FABRIC_PORT_STAT_FLEX_COUNTER_POLLING_INTERVAL_MS 10000 | ||
#define FABRIC_QUEUE_STAT_COUNTER_FLEX_COUNTER_GROUP "FABRIC_QUEUE_STAT_COUNTER" | ||
#define FABRIC_QUEUE_STAT_FLEX_COUNTER_POLLING_INTERVAL_MS 100000 | ||
#define FABRIC_PORT_TABLE_NAME "FABRIC_PORT_TABLE_NAME" | ||
#define FABRIC_COUNTERS_QUEUE_NAME_MAP "FABRIC_COUNTERS_QUEUE_NAME_MAP" | ||
#define FABRIC_COUNTERS_QUEUE_PORT_MAP "FABRIC_COUNTERS_QUEUE_PORT_MAP" | ||
#define FABRIC_COUNTERS_QUEUE_INDEX_MAP "FABRIC_COUNTERS_QUEUE_INDEX_MAP" | ||
#define FABRIC_COUNTERS_QUEUE_TYPE_MAP "FABRIC_COUNTERS_QUEUE_TYPE_MAP" | ||
#define FABRIC_COUNTERS_PORT_NAME_MAP "FABRIC_COUNTERS_PORT_NAME_MAP" | ||
|
||
extern sai_object_id_t gSwitchId; | ||
extern sai_switch_api_t *sai_switch_api; | ||
extern sai_port_api_t *sai_port_api; | ||
extern sai_queue_api_t *sai_queue_api; | ||
|
||
const vector<sai_port_stat_t> port_stat_ids = | ||
{ | ||
SAI_PORT_STAT_IF_IN_OCTETS, | ||
SAI_PORT_STAT_IF_IN_ERRORS, | ||
SAI_PORT_STAT_IF_IN_FABRIC_DATA_UNITS, | ||
SAI_PORT_STAT_IF_IN_FEC_CORRECTABLE_FRAMES, | ||
SAI_PORT_STAT_IF_IN_FEC_NOT_CORRECTABLE_FRAMES, | ||
SAI_PORT_STAT_IF_IN_FEC_SYMBOL_ERRORS, | ||
SAI_PORT_STAT_IF_OUT_OCTETS, | ||
SAI_PORT_STAT_IF_OUT_FABRIC_DATA_UNITS, | ||
}; | ||
|
||
static const vector<sai_queue_stat_t> queue_stat_ids = | ||
{ | ||
SAI_QUEUE_STAT_WATERMARK_LEVEL, | ||
SAI_QUEUE_STAT_CURR_OCCUPANCY_BYTES, | ||
SAI_QUEUE_STAT_CURR_OCCUPANCY_LEVEL, | ||
}; | ||
|
||
FabricPortsOrch::FabricPortsOrch(DBConnector *appl_db, vector<table_name_with_pri_t> &tableNames) : | ||
Orch(appl_db, tableNames), | ||
port_stat_manager(FABRIC_PORT_STAT_COUNTER_FLEX_COUNTER_GROUP, StatsMode::READ, | ||
FABRIC_PORT_STAT_FLEX_COUNTER_POLLING_INTERVAL_MS, true), | ||
queue_stat_manager(FABRIC_QUEUE_STAT_COUNTER_FLEX_COUNTER_GROUP, StatsMode::READ, | ||
FABRIC_QUEUE_STAT_FLEX_COUNTER_POLLING_INTERVAL_MS, true), | ||
m_timer(new SelectableTimer(timespec { .tv_sec = FABRIC_POLLING_INTERVAL_DEFAULT, .tv_nsec = 0 })) | ||
{ | ||
SWSS_LOG_ENTER(); | ||
SWSS_LOG_NOTICE( "FabricPortsOrch constructor" ); | ||
|
||
m_state_db = shared_ptr<DBConnector>(new DBConnector("STATE_DB", 0)); | ||
m_stateTable = unique_ptr<Table>(new Table(m_state_db.get(), FABRIC_PORT_TABLE_NAME)); | ||
|
||
m_counter_db = shared_ptr<DBConnector>(new DBConnector("COUNTERS_DB", 0)); | ||
m_portNameQueueCounterTable = unique_ptr<Table>(new Table(m_counter_db.get(), FABRIC_COUNTERS_QUEUE_NAME_MAP)); | ||
m_queuePortCounterTable = unique_ptr<Table>(new Table(m_counter_db.get(), FABRIC_COUNTERS_QUEUE_PORT_MAP)); | ||
m_queueIndexCounterTable = unique_ptr<Table>(new Table(m_counter_db.get(), FABRIC_COUNTERS_QUEUE_INDEX_MAP)); | ||
m_queueTypeCounterTable = unique_ptr<Table>(new Table(m_counter_db.get(), FABRIC_COUNTERS_QUEUE_TYPE_MAP)); | ||
m_portNamePortCounterTable = unique_ptr<Table>(new Table(m_counter_db.get(), FABRIC_COUNTERS_PORT_NAME_MAP)); | ||
|
||
m_flex_db = shared_ptr<DBConnector>(new DBConnector("FLEX_COUNTER_DB", 0)); | ||
m_flexCounterTable = unique_ptr<ProducerTable>(new ProducerTable(m_flex_db.get(), FABRIC_PORT_TABLE_NAME)); | ||
|
||
getFabricPortList(); | ||
|
||
auto executor = new ExecutableTimer(m_timer, this, "FABRIC_POLL"); | ||
Orch::addExecutor(executor); | ||
m_timer->start(); | ||
} | ||
|
||
int FabricPortsOrch::getFabricPortList() | ||
{ | ||
SWSS_LOG_ENTER(); | ||
SWSS_LOG_NOTICE("Get fabric port list done %d", m_getFabricPortListDone == true ? 1 : 0 ); | ||
|
||
if (m_getFabricPortListDone) { | ||
return FABRIC_PORT_SUCCESS; | ||
} | ||
|
||
uint32_t i; | ||
sai_status_t status; | ||
sai_attribute_t attr; | ||
|
||
attr.id = SAI_SWITCH_ATTR_NUMBER_OF_FABRIC_PORTS; | ||
status = sai_switch_api->get_switch_attribute(gSwitchId, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
SWSS_LOG_ERROR("Failed to get fabric port number, rv:%d", status); | ||
return FABRIC_PORT_ERROR; | ||
} | ||
m_fabricPortCount = attr.value.u32; | ||
SWSS_LOG_NOTICE("Get %d fabric ports", m_fabricPortCount); | ||
|
||
vector<sai_object_id_t> fabric_port_list; | ||
fabric_port_list.resize(m_fabricPortCount); | ||
attr.id = SAI_SWITCH_ATTR_FABRIC_PORT_LIST; | ||
attr.value.objlist.count = (uint32_t)fabric_port_list.size(); | ||
attr.value.objlist.list = fabric_port_list.data(); | ||
status = sai_switch_api->get_switch_attribute(gSwitchId, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get port list failure"); | ||
} | ||
|
||
for (i = 0; i < m_fabricPortCount; i++) | ||
{ | ||
sai_uint32_t lanes[1] = { 0 }; | ||
attr.id = SAI_PORT_ATTR_HW_LANE_LIST; | ||
attr.value.u32list.count = 1; | ||
attr.value.u32list.list = lanes; | ||
status = sai_port_api->get_port_attribute(fabric_port_list[i], 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get port lane failure"); | ||
} | ||
int lane = attr.value.u32list.list[0]; | ||
m_fabricLanePortMap[lane] = fabric_port_list[i]; | ||
SWSS_LOG_NOTICE("Get fabric port pid: %" PRIx64 " lanes: %d", fabric_port_list[i], lane); | ||
} | ||
|
||
generatePortStats(); | ||
|
||
m_getFabricPortListDone = true; | ||
|
||
return FABRIC_PORT_SUCCESS; | ||
} | ||
|
||
bool FabricPortsOrch::allPortsReady() | ||
{ | ||
return m_getFabricPortListDone; | ||
} | ||
|
||
void FabricPortsOrch::generatePortStats() | ||
{ | ||
SWSS_LOG_NOTICE("Generate fabric port stats"); | ||
|
||
vector<FieldValueTuple> portNamePortCounterMap; | ||
for (auto p : m_fabricLanePortMap) | ||
{ | ||
int lane = p.first; | ||
sai_object_id_t port = p.second; | ||
|
||
std::ostringstream portName; | ||
portName << "FabricPort" << lane; | ||
portNamePortCounterMap.emplace_back(portName.str(), sai_serialize_object_id(port)); | ||
|
||
// Install flex counters for port stats | ||
std::unordered_set<std::string> counter_stats; | ||
for (const auto& it: port_stat_ids) | ||
{ | ||
counter_stats.emplace(sai_serialize_port_stat(it)); | ||
} | ||
port_stat_manager.setCounterIdList(port, CounterType::PORT, counter_stats); | ||
} | ||
m_portNamePortCounterTable->set("", portNamePortCounterMap); | ||
} | ||
|
||
void FabricPortsOrch::generateQueueStats() | ||
{ | ||
if (m_isQueueStatsGenerated) return; | ||
if (!m_getFabricPortListDone) return; | ||
|
||
SWSS_LOG_NOTICE("Generate queue map for fabric ports"); | ||
|
||
sai_status_t status; | ||
sai_attribute_t attr; | ||
sai_attribute_t queue_attrs[2]; | ||
|
||
for (auto p : m_fabricLanePortMap) | ||
{ | ||
int lane = p.first; | ||
sai_object_id_t port = p.second; | ||
|
||
attr.id = SAI_PORT_ATTR_QOS_NUMBER_OF_QUEUES; | ||
status = sai_port_api->get_port_attribute(port, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get port queue number failure"); | ||
} | ||
int num_queues = attr.value.u32; | ||
|
||
if (num_queues > 0) | ||
{ | ||
vector<sai_object_id_t> m_queue_ids; | ||
m_queue_ids.resize(num_queues); | ||
|
||
attr.id = SAI_PORT_ATTR_QOS_QUEUE_LIST; | ||
attr.value.objlist.count = (uint32_t) num_queues; | ||
attr.value.objlist.list = m_queue_ids.data(); | ||
|
||
status = sai_port_api->get_port_attribute(port, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get port queue list failure"); | ||
} | ||
|
||
// Maintain queue map and install flex counters for queue stats | ||
vector<FieldValueTuple> portNameQueueMap; | ||
vector<FieldValueTuple> queuePortMap; | ||
vector<FieldValueTuple> queueTypeMap; | ||
vector<FieldValueTuple> queueIndexMap; | ||
|
||
for (size_t q = 0; q < m_queue_ids.size(); q ++) | ||
{ | ||
std::ostringstream portName; | ||
portName << "FabricPort" << lane << ":" << q; | ||
const auto queue = sai_serialize_object_id(m_queue_ids[q]); | ||
|
||
portNameQueueMap.emplace_back(portName.str(), queue); | ||
queuePortMap.emplace_back(queue, sai_serialize_object_id(port)); | ||
|
||
queue_attrs[0].id = SAI_QUEUE_ATTR_TYPE; | ||
queue_attrs[1].id = SAI_QUEUE_ATTR_INDEX; | ||
status = sai_queue_api->get_queue_attribute(m_queue_ids[q], 2, queue_attrs); | ||
if (status == SAI_STATUS_SUCCESS) | ||
{ | ||
string queueType; | ||
string queueIndex; | ||
switch (queue_attrs[0].value.s32) | ||
{ | ||
case SAI_QUEUE_TYPE_ALL: | ||
queueType = "SAI_QUEUE_TYPE_ALL"; | ||
break; | ||
case SAI_QUEUE_TYPE_UNICAST: | ||
queueType = "SAI_QUEUE_TYPE_UNICAST"; | ||
break; | ||
case SAI_QUEUE_TYPE_MULTICAST: | ||
queueType = "SAI_QUEUE_TYPE_MULTICAST"; | ||
break; | ||
default: | ||
throw runtime_error("Got unsupported queue type"); | ||
} | ||
queueTypeMap.emplace_back(queue, queueType); | ||
queueIndex = to_string(queue_attrs[1].value.u8); | ||
queueIndexMap.emplace_back(queue, queueIndex); | ||
} | ||
else | ||
{ | ||
SWSS_LOG_NOTICE("FabricPortsOrch cannot get fabric port queue type and index"); | ||
} | ||
|
||
std::unordered_set<string> counter_stats; | ||
for (const auto& it: queue_stat_ids) | ||
{ | ||
counter_stats.emplace(sai_serialize_queue_stat(it)); | ||
} | ||
queue_stat_manager.setCounterIdList(m_queue_ids[q], CounterType::QUEUE, counter_stats); | ||
} | ||
|
||
m_portNameQueueCounterTable->set("", portNameQueueMap); | ||
m_queuePortCounterTable->set("", queuePortMap); | ||
m_queueTypeCounterTable->set("", queueTypeMap); | ||
m_queueIndexCounterTable->set("", queueIndexMap); | ||
} | ||
} | ||
|
||
m_isQueueStatsGenerated = true; | ||
} | ||
|
||
void FabricPortsOrch::updateFabricPortState() | ||
{ | ||
if (!m_getFabricPortListDone) return; | ||
|
||
sai_status_t status; | ||
sai_attribute_t attr; | ||
|
||
for (auto p : m_fabricLanePortMap) | ||
{ | ||
int lane = p.first; | ||
sai_object_id_t port = p.second; | ||
|
||
string key = "PORT" + to_string(lane); | ||
std::vector<FieldValueTuple> values; | ||
|
||
attr.id = SAI_PORT_ATTR_FABRIC_ATTACHED; | ||
status = sai_port_api->get_port_attribute(port, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get port status failure"); | ||
} | ||
FieldValueTuple s("STATUS", attr.value.booldata ? "up" : "down"); | ||
values.push_back(s); | ||
|
||
if (attr.value.booldata) | ||
{ | ||
attr.id = SAI_PORT_ATTR_FABRIC_ATTACHED_SWITCH_ID; | ||
status = sai_port_api->get_port_attribute(port, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get remote id failure"); | ||
} | ||
FieldValueTuple rm("REMOTE_MOD", to_string(attr.value.u32)); | ||
values.push_back(rm); | ||
|
||
attr.id = SAI_PORT_ATTR_FABRIC_ATTACHED_PORT_INDEX; | ||
status = sai_port_api->get_port_attribute(port, 1, &attr); | ||
if (status != SAI_STATUS_SUCCESS) | ||
{ | ||
throw runtime_error("FabricPortsOrch get remote port index failure"); | ||
} | ||
FieldValueTuple rp("REMOTE_PORT", to_string(attr.value.u32)); | ||
values.push_back(rp); | ||
} | ||
m_stateTable->set(key, values); | ||
} | ||
} | ||
|
||
void FabricPortsOrch::doTask() | ||
{ | ||
} | ||
|
||
void FabricPortsOrch::doTask(Consumer &consumer) | ||
{ | ||
} | ||
|
||
void FabricPortsOrch::doTask(swss::SelectableTimer &timer) | ||
{ | ||
if (!m_getFabricPortListDone) | ||
{ | ||
getFabricPortList(); | ||
} | ||
|
||
if (m_getFabricPortListDone) | ||
{ | ||
updateFabricPortState(); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
#ifndef SWSS_FABRICPORTSORCH_H | ||
#define SWSS_FABRICPORTSORCH_H | ||
|
||
#include <map> | ||
|
||
#include "orch.h" | ||
#include "observer.h" | ||
#include "observer.h" | ||
#include "producertable.h" | ||
#include "flex_counter_manager.h" | ||
|
||
class FabricPortsOrch : public Orch, public Subject | ||
{ | ||
public: | ||
FabricPortsOrch(DBConnector *appl_db, vector<table_name_with_pri_t> &tableNames); | ||
bool allPortsReady(); | ||
void generateQueueStats(); | ||
|
||
private: | ||
shared_ptr<DBConnector> m_state_db; | ||
shared_ptr<DBConnector> m_counter_db; | ||
shared_ptr<DBConnector> m_flex_db; | ||
|
||
unique_ptr<Table> m_stateTable; | ||
unique_ptr<Table> m_portNameQueueCounterTable; | ||
unique_ptr<Table> m_queuePortCounterTable; | ||
unique_ptr<Table> m_queueIndexCounterTable; | ||
unique_ptr<Table> m_queueTypeCounterTable; | ||
unique_ptr<Table> m_portNamePortCounterTable; | ||
unique_ptr<ProducerTable> m_flexCounterTable; | ||
|
||
swss::SelectableTimer *m_timer = nullptr; | ||
|
||
FlexCounterManager port_stat_manager; | ||
FlexCounterManager queue_stat_manager; | ||
|
||
sai_uint32_t m_fabricPortCount; | ||
map<int, sai_object_id_t> m_fabricLanePortMap; | ||
|
||
bool m_getFabricPortListDone = false; | ||
bool m_isQueueStatsGenerated = false; | ||
int getFabricPortList(); | ||
void generatePortStats(); | ||
void updateFabricPortState(); | ||
|
||
void doTask() override; | ||
void doTask(Consumer &consumer); | ||
void doTask(swss::SelectableTimer &timer); | ||
}; | ||
|
||
#endif /* SWSS_FABRICPORTSORCH_H */ |
Oops, something went wrong.