From fd75e54d2ad6d73624c64f2ae0f938c6038a0c66 Mon Sep 17 00:00:00 2001 From: Dong Zhang <41927498+dzhangalibaba@users.noreply.github.com> Date: Tue, 11 Sep 2018 09:37:13 -0700 Subject: [PATCH] sairedis: add while loop to make syncd processEvent() handle as many entries as possible in one event call (#335) * sairedis: add while loop to make syncd processEvent() handle as many entries as possible in one event call * add while() loop to handle entries when buffer is Not empty * this change has dependency on the added empty() function in ConsumerTableBase class in swss-common which is submitted in other request * with the changes in swss-common, route performance improved about ~200 routes/sec * sairedis: update SWSS_LOG_INFO to SWSS_LOG_DEBUG * update SWSS_LOG_INFO to SWSS_LOG_DEBUG --- syncd/syncd.cpp | 352 +++++++++++++++++++++++++----------------------- 1 file changed, 180 insertions(+), 172 deletions(-) diff --git a/syncd/syncd.cpp b/syncd/syncd.cpp index 67185f3fbd8e..64778f1b0cc3 100644 --- a/syncd/syncd.cpp +++ b/syncd/syncd.cpp @@ -2478,226 +2478,234 @@ sai_status_t processEvent( swss::KeyOpFieldsValuesTuple kco; - if (isInitViewMode()) - { - /* - * In init mode we put all data to TEMP view and we snoop. We need to - * specify temporary view prefis in consumer since consumer puts data - * to redis db. - */ + sai_status_t status = SAI_STATUS_SUCCESS; - consumer.pop(kco, TEMP_PREFIX); - } - else - { - consumer.pop(kco); - } + do { - const std::string &key = kfvKey(kco); - const std::string &op = kfvOp(kco); + if (isInitViewMode()) + { + /* + * In init mode we put all data to TEMP view and we snoop. We need to + * specify temporary view prefis in consumer since consumer puts data + * to redis db. + */ - /* - * TODO: Key is serialized meta_key, we could use deserialize - * to extract it here. - */ + consumer.pop(kco, TEMP_PREFIX); + } + else + { + consumer.pop(kco); + } - const std::string &str_object_type = key.substr(0, key.find(":")); - const std::string &str_object_id = key.substr(key.find(":") + 1); + const std::string &key = kfvKey(kco); + const std::string &op = kfvOp(kco); - SWSS_LOG_INFO("key: %s op: %s", key.c_str(), op.c_str()); + if (key.length() == 0) { + SWSS_LOG_DEBUG("no elements in m_buffer"); + return status; + } - sai_common_api_t api = SAI_COMMON_API_MAX; + /* + * TODO: Key is serialized meta_key, we could use deserialize + * to extract it here. + */ - if (op == "create") - { - api = SAI_COMMON_API_CREATE; - } - else if (op == "remove") - { - api = SAI_COMMON_API_REMOVE; - } - else if (op == "set") - { - api = SAI_COMMON_API_SET; - } - else if (op == "get") - { - api = SAI_COMMON_API_GET; - } - else if (op == "bulkset") - { - return processBulkEvent((sai_common_api_t)SAI_COMMON_API_BULK_SET, kco); - } - else if (op == "bulkcreate") - { - return processBulkEvent((sai_common_api_t)SAI_COMMON_API_BULK_CREATE, kco); - } - else if (op == "notify") - { - return notifySyncd(key); - } - else if (op == "get_stats") - { - return processGetStatsEvent(kco); - } - else if (op == "flush") - { - return processFdbFlush(kco); - } - else - { - SWSS_LOG_THROW("api '%s' is not implemented", op.c_str()); - } + const std::string &str_object_type = key.substr(0, key.find(":")); + const std::string &str_object_id = key.substr(key.find(":") + 1); - sai_object_type_t object_type; - sai_deserialize_object_type(str_object_type, object_type); + SWSS_LOG_INFO("key: %s op: %s", key.c_str(), op.c_str()); - /* - * TODO: use metadata utils is object type valid. - */ + sai_common_api_t api = SAI_COMMON_API_MAX; - if (object_type == SAI_OBJECT_TYPE_NULL || object_type >= SAI_OBJECT_TYPE_MAX) - { - SWSS_LOG_THROW("undefined object type %s", sai_serialize_object_type(object_type).c_str()); - } + if (op == "create") + { + api = SAI_COMMON_API_CREATE; + } + else if (op == "remove") + { + api = SAI_COMMON_API_REMOVE; + } + else if (op == "set") + { + api = SAI_COMMON_API_SET; + } + else if (op == "get") + { + api = SAI_COMMON_API_GET; + } + else if (op == "bulkset") + { + return processBulkEvent((sai_common_api_t)SAI_COMMON_API_BULK_SET, kco); + } + else if (op == "bulkcreate") + { + return processBulkEvent((sai_common_api_t)SAI_COMMON_API_BULK_CREATE, kco); + } + else if (op == "notify") + { + return notifySyncd(key); + } + else if (op == "get_stats") + { + return processGetStatsEvent(kco); + } + else if (op == "flush") + { + return processFdbFlush(kco); + } + else + { + SWSS_LOG_THROW("api '%s' is not implemented", op.c_str()); + } - const std::vector &values = kfvFieldsValues(kco); + sai_object_type_t object_type; + sai_deserialize_object_type(str_object_type, object_type); - for (const auto &v: values) - { - SWSS_LOG_DEBUG("attr: %s: %s", fvField(v).c_str(), fvValue(v).c_str()); - } + /* + * TODO: use metadata utils is object type valid. + */ - SaiAttributeList list(object_type, values, false); + if (object_type == SAI_OBJECT_TYPE_NULL || object_type >= SAI_OBJECT_TYPE_MAX) + { + SWSS_LOG_THROW("undefined object type %s", sai_serialize_object_type(object_type).c_str()); + } - /* - * Attribute list can't be const since we will use it to translate VID to - * RID inplace. - */ + const std::vector &values = kfvFieldsValues(kco); - sai_attribute_t *attr_list = list.get_attr_list(); - uint32_t attr_count = list.get_attr_count(); + for (const auto &v: values) + { + SWSS_LOG_DEBUG("attr: %s: %s", fvField(v).c_str(), fvValue(v).c_str()); + } - /* - * NOTE: This check pointers must be executed before init view mode, since - * this methods replaces pointers from orchagent memory space to syncd - * memory space. - */ + SaiAttributeList list(object_type, values, false); - if (object_type == SAI_OBJECT_TYPE_SWITCH && (api == SAI_COMMON_API_CREATE || api == SAI_COMMON_API_SET)) - { /* - * We don't need to clear those pointers on switch remove (evan last), - * since those pointers will reside inside attributes, also sairedis - * will internally check whether pointer is null or not, so we here - * will receive all notifications, but redis only those that were set. + * Attribute list can't be const since we will use it to translate VID to + * RID inplace. */ - check_notifications_pointers(attr_count, attr_list); - } - - if (isInitViewMode()) - { - return processEventInInitViewMode(object_type, str_object_id, api, attr_count, attr_list); - } + sai_attribute_t *attr_list = list.get_attr_list(); + uint32_t attr_count = list.get_attr_count(); - if (api != SAI_COMMON_API_GET) - { /* - * TODO we can also call translate on get, if sairedis will clean - * buffer so then all OIDs will be NULL, and translation will also - * convert them to NULL. + * NOTE: This check pointers must be executed before init view mode, since + * this methods replaces pointers from orchagent memory space to syncd + * memory space. */ - SWSS_LOG_DEBUG("translating VID to RIDs on all attributes"); + if (object_type == SAI_OBJECT_TYPE_SWITCH && (api == SAI_COMMON_API_CREATE || api == SAI_COMMON_API_SET)) + { + /* + * We don't need to clear those pointers on switch remove (evan last), + * since those pointers will reside inside attributes, also sairedis + * will internally check whether pointer is null or not, so we here + * will receive all notifications, but redis only those that were set. + */ - translate_vid_to_rid_list(object_type, attr_count, attr_list); - } + check_notifications_pointers(attr_count, attr_list); + } - // TODO use metadata utils - auto info = sai_metadata_get_object_type_info(object_type); + if (isInitViewMode()) + { + return processEventInInitViewMode(object_type, str_object_id, api, attr_count, attr_list); + } - sai_status_t status; + if (api != SAI_COMMON_API_GET) + { + /* + * TODO we can also call translate on get, if sairedis will clean + * buffer so then all OIDs will be NULL, and translation will also + * convert them to NULL. + */ - /* - * TODO use sai meta key deserialize - */ + SWSS_LOG_DEBUG("translating VID to RIDs on all attributes"); - if (info->isnonobjectid) - { - sai_object_meta_key_t meta_key; + translate_vid_to_rid_list(object_type, attr_count, attr_list); + } - meta_key.objecttype = object_type; + // TODO use metadata utils + auto info = sai_metadata_get_object_type_info(object_type); - switch (object_type) + /* + * TODO use sai meta key deserialize + */ + + if (info->isnonobjectid) { - case SAI_OBJECT_TYPE_FDB_ENTRY: - sai_deserialize_fdb_entry(str_object_id, meta_key.objectkey.key.fdb_entry); - break; + sai_object_meta_key_t meta_key; - case SAI_OBJECT_TYPE_NEIGHBOR_ENTRY: - sai_deserialize_neighbor_entry(str_object_id, meta_key.objectkey.key.neighbor_entry); - break; + meta_key.objecttype = object_type; - case SAI_OBJECT_TYPE_ROUTE_ENTRY: - sai_deserialize_route_entry(str_object_id, meta_key.objectkey.key.route_entry); - break; + switch (object_type) + { + case SAI_OBJECT_TYPE_FDB_ENTRY: + sai_deserialize_fdb_entry(str_object_id, meta_key.objectkey.key.fdb_entry); + break; - default: + case SAI_OBJECT_TYPE_NEIGHBOR_ENTRY: + sai_deserialize_neighbor_entry(str_object_id, meta_key.objectkey.key.neighbor_entry); + break; - SWSS_LOG_THROW("non object id %s is not supported yet, FIXME", info->objecttypename); - } + case SAI_OBJECT_TYPE_ROUTE_ENTRY: + sai_deserialize_route_entry(str_object_id, meta_key.objectkey.key.route_entry); + break; - status = handle_non_object_id(meta_key, api, attr_count, attr_list); - } - else - { - status = handle_generic(object_type, str_object_id, api, attr_count, attr_list); - } + default: - if (api == SAI_COMMON_API_GET) - { - if (status != SAI_STATUS_SUCCESS) + SWSS_LOG_THROW("non object id %s is not supported yet, FIXME", info->objecttypename); + } + + status = handle_non_object_id(meta_key, api, attr_count, attr_list); + } + else { - SWSS_LOG_DEBUG("get API for key: %s op: %s returned status: %s", - key.c_str(), - op.c_str(), - sai_serialize_status(status).c_str()); + status = handle_generic(object_type, str_object_id, api, attr_count, attr_list); } - /* - * Extracting switch is double work here, we can avoid this when we - * will use meta_key. - */ + if (api == SAI_COMMON_API_GET) + { + if (status != SAI_STATUS_SUCCESS) + { + SWSS_LOG_DEBUG("get API for key: %s op: %s returned status: %s", + key.c_str(), + op.c_str(), + sai_serialize_status(status).c_str()); + } - sai_object_id_t switch_vid = extractSwitchVid(object_type, str_object_id); + /* + * Extracting switch is double work here, we can avoid this when we + * will use meta_key. + */ - internal_syncd_get_send(object_type, str_object_id, switch_vid, status, attr_count, attr_list); - } - else if (status != SAI_STATUS_SUCCESS) - { - if (!info->isnonobjectid && api == SAI_COMMON_API_SET) + sai_object_id_t switch_vid = extractSwitchVid(object_type, str_object_id); + + internal_syncd_get_send(object_type, str_object_id, switch_vid, status, attr_count, attr_list); + } + else if (status != SAI_STATUS_SUCCESS) { - sai_object_id_t vid; - sai_deserialize_object_id(str_object_id, vid); + if (!info->isnonobjectid && api == SAI_COMMON_API_SET) + { + sai_object_id_t vid; + sai_deserialize_object_id(str_object_id, vid); - sai_object_id_t rid = translate_vid_to_rid(vid); + sai_object_id_t rid = translate_vid_to_rid(vid); - SWSS_LOG_ERROR("VID: %s RID: %s", - sai_serialize_object_id(vid).c_str(), - sai_serialize_object_id(rid).c_str()); - } + SWSS_LOG_ERROR("VID: %s RID: %s", + sai_serialize_object_id(vid).c_str(), + sai_serialize_object_id(rid).c_str()); + } - for (const auto &v: values) - { - SWSS_LOG_ERROR("attr: %s: %s", fvField(v).c_str(), fvValue(v).c_str()); - } + for (const auto &v: values) + { + SWSS_LOG_ERROR("attr: %s: %s", fvField(v).c_str(), fvValue(v).c_str()); + } - SWSS_LOG_THROW("failed to execute api: %s, key: %s, status: %s", - op.c_str(), - key.c_str(), - sai_serialize_status(status).c_str()); - } + SWSS_LOG_THROW("failed to execute api: %s, key: %s, status: %s", + op.c_str(), + key.c_str(), + sai_serialize_status(status).c_str()); + } + } while (!consumer.empty()); return status; }