From 297d5855ca3ac2e3fea329bd938cf4763fac583b Mon Sep 17 00:00:00 2001 From: Samuel O'Brien Date: Tue, 23 Jun 2020 15:09:55 -0500 Subject: utils: Expose CHDR Parsing API This commit introduces a new public api in uhd::utils which allows serializing and deserializing chdr packets. As far as testing, this commit adds the chdr_parse_test test. It uses a wireshark trace located in rfnoc_packets_*.cpp as well as hand coded packets from hardcoded_packets.cpp to test the serialization and deserialization process Signed-off-by: Samuel O'Brien --- host/lib/rfnoc/chdr_types.cpp | 124 ++++++++++++++++++++++++++ host/lib/utils/CMakeLists.txt | 2 + host/lib/utils/chdr/CMakeLists.txt | 9 ++ host/lib/utils/chdr/chdr_packet.cpp | 168 ++++++++++++++++++++++++++++++++++++ 4 files changed, 303 insertions(+) create mode 100644 host/lib/utils/chdr/CMakeLists.txt create mode 100644 host/lib/utils/chdr/chdr_packet.cpp (limited to 'host/lib') diff --git a/host/lib/rfnoc/chdr_types.cpp b/host/lib/rfnoc/chdr_types.cpp index 3cb68ff9a..3978c8694 100644 --- a/host/lib/rfnoc/chdr_types.cpp +++ b/host/lib/rfnoc/chdr_types.cpp @@ -9,6 +9,7 @@ #include #include #include +#include using namespace uhd; using namespace uhd::rfnoc::chdr; @@ -141,6 +142,17 @@ void ctrl_payload::deserialize(const uint64_t* buff, UHD_ASSERT_THROW(ptr <= max_size_bytes); } +size_t ctrl_payload::get_length() const +{ + size_t length = 1; + if (this->has_timestamp()) { + length += 1; + } + size_t operations = 1 + this->data_vtr.size(); + length += operations / 2 + operations % 2; + return length; +} + bool ctrl_payload::operator==(const ctrl_payload& rhs) const { return (dst_port == rhs.dst_port) && (src_port == rhs.src_port) @@ -241,6 +253,11 @@ void strs_payload::deserialize(const uint64_t* buff, status_info = get_field_u64(word3, STATUS_INFO_OFFSET, STATUS_INFO_WIDTH); } +size_t strs_payload::get_length() const +{ + return 4; +} + bool strs_payload::operator==(const strs_payload& rhs) const { return (src_epid == rhs.src_epid) && (status == rhs.status) @@ -310,6 +327,11 @@ void strc_payload::deserialize(const uint64_t* buff, num_bytes = conv_byte_order(buff[1]); } +size_t strc_payload::get_length() const +{ + return 2; +} + bool strc_payload::operator==(const strc_payload& rhs) const { return (src_epid == rhs.src_epid) && (op_code == rhs.op_code) @@ -328,6 +350,74 @@ const std::string strc_payload::to_string() const // CHDR Management Payload //---------------------------------------------------- +const std::string mgmt_op_t::to_string() const +{ + std::stringstream stream; + switch (get_op_code()) { + case mgmt_op_t::MGMT_OP_NOP: + stream << "NOP"; + break; + case mgmt_op_t::MGMT_OP_ADVERTISE: + stream << "ADVERTISE"; + break; + case mgmt_op_t::MGMT_OP_SEL_DEST: + stream << "SEL_DEST"; + break; + case mgmt_op_t::MGMT_OP_RETURN: + stream << "RETURN"; + break; + case mgmt_op_t::MGMT_OP_INFO_REQ: + stream << "INFO_REQ"; + break; + case mgmt_op_t::MGMT_OP_INFO_RESP: + stream << "INFO_RESP"; + break; + case mgmt_op_t::MGMT_OP_CFG_WR_REQ: + stream << "CFG_WR_REQ"; + break; + case mgmt_op_t::MGMT_OP_CFG_RD_REQ: + stream << "CFG_RD_REQ"; + break; + case mgmt_op_t::MGMT_OP_CFG_RD_RESP: + stream << "CFG_RD_RESP"; + break; + default: + UHD_THROW_INVALID_CODE_PATH(); + } + stream << ": "; + switch (get_op_code()) { + case mgmt_op_t::MGMT_OP_SEL_DEST: { + mgmt_op_t::sel_dest_payload payload = static_cast(get_op_payload()); + stream << "dest:" << payload.dest; + break; + } + case mgmt_op_t::MGMT_OP_CFG_WR_REQ: + case mgmt_op_t::MGMT_OP_CFG_RD_REQ: + case mgmt_op_t::MGMT_OP_CFG_RD_RESP: { + mgmt_op_t::cfg_payload payload = static_cast(get_op_payload()); + stream << str( + boost::format("addr:0x%08x, data:0x%08x") % payload.addr % payload.data); + break; + } + case mgmt_op_t::MGMT_OP_INFO_REQ: + case mgmt_op_t::MGMT_OP_INFO_RESP: { + mgmt_op_t::node_info_payload payload = + static_cast(get_op_payload()); + stream << "device_id:" << payload.device_id + << ", node_type:" << payload.node_type + << ", node_inst:" << payload.node_inst + << ", ext_info:" << payload.ext_info; + break; + } + default: { + stream << "-"; + break; + } + } + stream << "\n"; + return stream.str(); +} + //! Serialize this hop into a list of 64-bit words size_t mgmt_hop_t::serialize(std::vector& target, const std::function& conv_byte_order, @@ -369,6 +459,21 @@ void mgmt_hop_t::deserialize(std::list& src, } while (ops_remaining > 0); } +const std::string mgmt_hop_t::to_string() const +{ + std::stringstream stream; + for (size_t op_index = 0; op_index < get_num_ops(); op_index++) { + if (op_index == 0) { + stream << " -> "; + } else { + stream << " "; + } + const mgmt_op_t& op = get_op(op_index); + stream << op.to_string(); + } + return stream.str(); +} + void mgmt_payload::populate_header(chdr_header& header) const { header.set_pkt_type(PKT_TYPE_MGMT); @@ -440,6 +545,15 @@ void mgmt_payload::deserialize(const uint64_t* buff, } } +size_t mgmt_payload::get_length() const +{ + size_t length = 1 + _padding_size; /* header */ + for (const auto& hop : this->_hops) { + length += hop.get_num_ops() + _padding_size; + } + return length; +} + const std::string mgmt_payload::to_string() const { return str(boost::format( @@ -447,6 +561,16 @@ const std::string mgmt_payload::to_string() const % _src_epid % int(_chdr_w) % _protover % _hops.size()); } +const std::string mgmt_payload::hops_to_string() const +{ + std::stringstream stream; + for (size_t hop_index = 0; hop_index < get_num_hops(); hop_index++) { + const mgmt_hop_t& hop = get_hop(hop_index); + stream << hop.to_string(); + } + return stream.str(); +} + bool mgmt_payload::operator==(const mgmt_payload& rhs) const { diff --git a/host/lib/utils/CMakeLists.txt b/host/lib/utils/CMakeLists.txt index aa06e2c45..7e95d8c28 100644 --- a/host/lib/utils/CMakeLists.txt +++ b/host/lib/utils/CMakeLists.txt @@ -9,6 +9,8 @@ # This file included, use CMake directory variables ######################################################################## +INCLUDE_SUBDIRECTORY(chdr) + ######################################################################## # Setup defines for process scheduling ######################################################################## diff --git a/host/lib/utils/chdr/CMakeLists.txt b/host/lib/utils/chdr/CMakeLists.txt new file mode 100644 index 000000000..04f1180d4 --- /dev/null +++ b/host/lib/utils/chdr/CMakeLists.txt @@ -0,0 +1,9 @@ +# +# Copyright 2020 Ettus Research, a National Instruments Brand +# +# SPDX-License-Identifier: GPL-3.0-or-later +# + +LIBUHD_APPEND_SOURCES( + ${CMAKE_CURRENT_SOURCE_DIR}/chdr_packet.cpp +) diff --git a/host/lib/utils/chdr/chdr_packet.cpp b/host/lib/utils/chdr/chdr_packet.cpp new file mode 100644 index 000000000..6a0e3f8bb --- /dev/null +++ b/host/lib/utils/chdr/chdr_packet.cpp @@ -0,0 +1,168 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include +#include +#include + +namespace chdr_rfnoc = uhd::rfnoc::chdr; +namespace chdr_util = uhd::utils::chdr; +using namespace uhd; + +chdr_util::chdr_packet::chdr_packet(uhd::rfnoc::chdr_w_t chdr_w, + chdr_rfnoc::chdr_header header, + std::vector payload, + boost::optional timestamp, + std::vector mdata) + : _chdr_w(chdr_w) + , _header(header) + , _payload(std::move(payload)) + , _timestamp(timestamp) + , _mdata(std::move(mdata)) +{ + set_header_lengths(); +} + +inline static uint64_t u64_to_host(uhd::endianness_t endianness, uint64_t word) +{ + return (endianness == ENDIANNESS_BIG) ? uhd::ntohx(word) + : uhd::wtohx(word); +} + +inline static uint64_t u64_from_host(uhd::endianness_t endianness, uint64_t word) +{ + return (endianness == ENDIANNESS_BIG) ? uhd::htonx(word) + : uhd::htowx(word); +} + +chdr_rfnoc::chdr_header chdr_util::chdr_packet::get_header() const +{ + return this->_header; +} + +void chdr_util::chdr_packet::set_header(chdr_rfnoc::chdr_header header) +{ + _header = header; + set_header_lengths(); +} + +boost::optional chdr_util::chdr_packet::get_timestamp() const +{ + return this->_timestamp; +} + +void chdr_util::chdr_packet::set_timestamp(boost::optional timestamp) +{ + _timestamp = timestamp; + set_header_lengths(); +} + +std::vector chdr_util::chdr_packet::serialize_to_byte_vector( + endianness_t endianness) const +{ + std::vector buffer(get_packet_len(), 0); + serialize(buffer.begin(), buffer.end(), endianness); + return buffer; +} + +void chdr_util::chdr_packet::serialize_ptr( + endianness_t endianness, void* start, void* end) const +{ + size_t len = static_cast(end) - static_cast(start); + UHD_ASSERT_THROW(get_packet_len() <= len); + chdr_rfnoc::chdr_packet_factory factory(_chdr_w, endianness); + chdr_rfnoc::chdr_packet_writer::uptr packet_writer = + factory.make_generic(std::numeric_limits::max()); + chdr_rfnoc::chdr_header header = _header; + packet_writer->refresh(start, header, (_timestamp ? *_timestamp : 0)); + + uint64_t* mdata_ptr = static_cast(packet_writer->get_mdata_ptr()); + auto mdata_src_begin = this->_mdata.begin(); + auto mdata_src_end = this->_mdata.end(); + std::transform( + mdata_src_begin, mdata_src_end, mdata_ptr, [endianness](uint64_t value) { + return u64_from_host(endianness, value); + }); + + uint8_t* payload_ptr = static_cast(packet_writer->get_payload_ptr()); + std::copy(_payload.begin(), _payload.end(), payload_ptr); +} + +chdr_util::chdr_packet chdr_util::chdr_packet::deserialize_ptr( + uhd::rfnoc::chdr_w_t chdr_w, + endianness_t endianness, + const void* start, + const void* end) +{ + chdr_rfnoc::chdr_packet_factory factory(chdr_w, endianness); + chdr_rfnoc::chdr_packet_writer::uptr packet_writer = + factory.make_generic(std::numeric_limits::max()); + packet_writer->refresh(start); + + chdr_rfnoc::chdr_header header = packet_writer->get_chdr_header(); + boost::optional timestamp = packet_writer->get_timestamp(); + + const size_t mdata_size_words = packet_writer->get_mdata_size() / 8; + const uint64_t* mdata_src_begin = + static_cast(packet_writer->get_mdata_const_ptr()); + const uint64_t* mdata_src_end = mdata_src_begin + mdata_size_words; + std::vector mdata(mdata_size_words, 0); + uint64_t* mdata_ptr = mdata.data(); + UHD_ASSERT_THROW(mdata_src_end < static_cast(end)); + std::transform( + mdata_src_begin, mdata_src_end, mdata_ptr, [endianness](uint64_t value) { + return u64_from_host(endianness, value); + }); + + size_t payload_size = packet_writer->get_payload_size(); + const uint8_t* payload_begin = + static_cast(packet_writer->get_payload_const_ptr()); + const uint8_t* payload_end = payload_begin + payload_size; + std::vector payload(payload_size, 0); + UHD_ASSERT_THROW(payload_end <= static_cast(end)); + std::copy(payload_begin, payload_end, payload.begin()); + + return chdr_util::chdr_packet(chdr_w, header, payload, timestamp, mdata); +} + +void chdr_util::chdr_packet::set_metadata(std::vector metadata) +{ + _mdata = std::move(metadata); + set_header_lengths(); +} + +const std::vector& chdr_util::chdr_packet::get_metadata() const +{ + return _mdata; +} + +size_t chdr_util::chdr_packet::get_packet_len() const +{ + size_t chdr_w_bytes = uhd::rfnoc::chdr_w_to_bits(_chdr_w) / 8; + // The timestamp and header take up 2 chdr_w lengths only when CHDR_W = 64 bits and + // there is a timestamp (RFNoC Specifcation section 2.2.1) + return _payload.size() * sizeof(uint8_t) + _mdata.size() * sizeof(uint64_t) + + ((_timestamp != boost::none && _chdr_w == uhd::rfnoc::CHDR_W_64) ? 2 : 1) + * chdr_w_bytes; +} + +const std::vector& chdr_util::chdr_packet::get_payload_bytes() const +{ + return _payload; +} + +void chdr_util::chdr_packet::set_payload_bytes(std::vector bytes) +{ + this->_payload = std::move(bytes); + set_header_lengths(); +} + +//! Return a string representation of this object +const std::string chdr_util::chdr_packet::to_string() const +{ + return str(boost::format("chdr_packet{chdr_w:%u}\n%s") + % uhd::rfnoc::chdr_w_to_bits(_chdr_w) % _header.to_string()); +} -- cgit v1.2.3