aboutsummaryrefslogtreecommitdiffstats
path: root/host
diff options
context:
space:
mode:
Diffstat (limited to 'host')
-rw-r--r--host/lib/include/uhdlib/utils/eeprom_utils.hpp80
-rw-r--r--host/lib/usrp/x300/x300_mb_eeprom.cpp26
2 files changed, 104 insertions, 2 deletions
diff --git a/host/lib/include/uhdlib/utils/eeprom_utils.hpp b/host/lib/include/uhdlib/utils/eeprom_utils.hpp
index 28deca790..3071fd745 100644
--- a/host/lib/include/uhdlib/utils/eeprom_utils.hpp
+++ b/host/lib/include/uhdlib/utils/eeprom_utils.hpp
@@ -5,8 +5,10 @@
//
#include <uhd/types/byte_vector.hpp>
+#include <uhd/types/dict.hpp>
#include <uhd/types/mac_addr.hpp>
#include <boost/asio/ip/address_v4.hpp>
+#include <uhd/utils/log.hpp>
#include <string>
#include <vector>
@@ -18,3 +20,81 @@ uhd::byte_vector_t string_to_uint16_bytes(const std::string &num_str);
//! convert a byte vector read from eeprom to a string
std::string uint16_bytes_to_string(const uhd::byte_vector_t &bytes);
+
+/*!
+ * Check for duplicate values within a given set of keys. Assumes the desire
+ * is to replace current EEPROM contents with new EEPROM contents and checks to
+ * see if the resulting contents will contain duplicates. Useful error
+ * messages are logged describing any duplicates.
+ *
+ * <field_type> must provide to_string() and from_string() functions
+ *
+ * \param error_label Label to put on error messages
+ * \param new_eeprom New EEPROM contents
+ * \param curr_eeprom Current EEPROM contents
+ * \param category Category label for the type of values being checked
+ * \param keys Keys to examine for duplicate values
+ * \return true if duplicates are found, false if not
+ */
+template <typename field_type>
+bool check_for_duplicates(
+ const std::string& error_label,
+ const uhd::dict<std::string, std::string>& new_eeprom,
+ const uhd::dict<std::string, std::string>& curr_eeprom,
+ const std::string& category,
+ const std::vector<std::string>& keys
+) {
+ bool has_duplicates = false;
+ for (size_t i = 0; i < keys.size(); i++)
+ {
+ bool found_duplicate = false;
+ auto key = keys[i];
+
+ if (not new_eeprom.has_key(key))
+ {
+ continue;
+ }
+
+ auto value = field_type::from_string(new_eeprom[key]).to_string();
+
+ // Check other values in new_eeprom for duplicate
+ // Starting at key index i+1 so the same duplicate is not found twice
+ for (size_t j = i+1; j < keys.size(); j++)
+ {
+ auto other_key = keys[j];
+ if (not new_eeprom.has_key(other_key))
+ {
+ continue;
+ }
+ auto other_value = field_type::from_string(new_eeprom[other_key]).to_string();
+ if (value == other_value)
+ {
+ // Value is a duplicate of another supplied value
+ UHD_LOG_ERROR(error_label, "Duplicate " << category << " "
+ << new_eeprom[key] << " is supplied for both " << key
+ << " and " << other_key);
+ found_duplicate = true;
+ }
+ }
+ // Check all keys in curr_eeprom for duplicate value
+ for (auto other_key: keys)
+ {
+ // Skip any keys in new_eeprom
+ if (new_eeprom.has_key(other_key))
+ {
+ continue;
+ }
+
+ if (value == curr_eeprom[other_key])
+ {
+ // Value is duplicate of one in the EEPROM
+ UHD_LOG_ERROR(error_label, "Duplicate " << category << " "
+ << new_eeprom[key] << " is already in use for "
+ << other_key);
+ found_duplicate = true;
+ }
+ }
+ has_duplicates |= found_duplicate;
+ }
+ return has_duplicates;
+}
diff --git a/host/lib/usrp/x300/x300_mb_eeprom.cpp b/host/lib/usrp/x300/x300_mb_eeprom.cpp
index b8fb59538..7bff3d782 100644
--- a/host/lib/usrp/x300/x300_mb_eeprom.cpp
+++ b/host/lib/usrp/x300/x300_mb_eeprom.cpp
@@ -130,11 +130,34 @@ mboard_eeprom_t x300_impl::get_mb_eeprom(uhd::i2c_iface::sptr iface)
return mb_eeprom;
}
-
void x300_impl::set_mb_eeprom(
i2c_iface::sptr iface,
const mboard_eeprom_t &mb_eeprom
) {
+ const mboard_eeprom_t curr_eeprom = get_mb_eeprom(iface);
+
+ // Check for duplicate MAC and IP addresses
+ const std::vector<std::string> mac_keys{
+ "mac-addr0",
+ "mac-addr1"
+ };
+ const std::vector<std::string> ip_keys{
+ "ip-addr0",
+ "ip-addr1",
+ "ip-addr2",
+ "ip-addr3"
+ };
+
+ //make sure there are no duplicate values
+ if (check_for_duplicates<uhd::mac_addr_t>(
+ "X300", mb_eeprom, curr_eeprom,"MAC address", mac_keys) or
+ check_for_duplicates<boost::asio::ip::address_v4>(
+ "X300", mb_eeprom, curr_eeprom, "IP address", ip_keys))
+ {
+ throw uhd::value_error(
+ "Duplicate values not permitted - write to EEPROM aborted");
+ }
+
//parse the revision number
if (mb_eeprom.has_key("revision")) iface->write_eeprom(
X300_EEPROM_ADDR, offsetof(x300_eeprom_map, revision),
@@ -195,4 +218,3 @@ void x300_impl::set_mb_eeprom(
string_to_bytes(mb_eeprom["name"], NAME_MAX_LEN)
);
}
-