// // Copyright 2014-2016 Ettus Research LLC // Copyright 2018 Ettus Research, a National Instruments Company // // SPDX-License-Identifier: GPL-3.0-or-later // #include "common.h" #include "octoclock_impl.hpp" #include #include #include #include #include #include #include #include #include typedef boost::asio::ip::address_v4 ip_v4; using namespace uhd; using namespace uhd::usrp_clock; using namespace uhd::transport; /*********************************************************************** * Implementation **********************************************************************/ void octoclock_eeprom_t::_load() { uint8_t octoclock_data[udp_simple::mtu]; const octoclock_packet_t* pkt_in = reinterpret_cast(octoclock_data); const octoclock_fw_eeprom_t* eeprom_in = reinterpret_cast(pkt_in->data); auto pkt_out = make_octoclock_packet(); size_t len = 0; UHD_OCTOCLOCK_SEND_AND_RECV( xport, _proto_ver, SEND_EEPROM_CMD, pkt_out, len, octoclock_data); if (UHD_OCTOCLOCK_PACKET_MATCHES(SEND_EEPROM_ACK, pkt_out, pkt_in, len)) { // MAC address byte_vector_t mac_bytes(eeprom_in->mac_addr, eeprom_in->mac_addr + 6); (*this)["mac-addr"] = mac_addr_t::from_bytes(mac_bytes).to_string(); // IP address uint32_t ip_addr = uhd::htonx(eeprom_in->ip_addr); ip_v4::bytes_type ip_addr_bytes; memcpy(&ip_addr_bytes, &ip_addr, 4); (*this)["ip-addr"] = ip_v4(ip_addr_bytes).to_string(); // Default router uint32_t dr_addr = uhd::htonx(eeprom_in->dr_addr); ip_v4::bytes_type dr_addr_bytes; memcpy(&dr_addr_bytes, &dr_addr, 4); (*this)["gateway"] = ip_v4(dr_addr_bytes).to_string(); // Netmask uint32_t netmask = uhd::htonx(eeprom_in->netmask); ip_v4::bytes_type netmask_bytes; memcpy(&netmask_bytes, &netmask, 4); (*this)["netmask"] = ip_v4(netmask_bytes).to_string(); // Serial std::string raw_serial((char*)eeprom_in->serial, 10); byte_vector_t serial_bytes(raw_serial.begin(), raw_serial.end()); (*this)["serial"] = bytes_to_string(serial_bytes); // Name std::string raw_name((char*)eeprom_in->name, 10); byte_vector_t name_bytes(raw_name.begin(), raw_name.end()); (*this)["name"] = bytes_to_string(name_bytes); // Revision (*this)["revision"] = std::to_string(int(eeprom_in->revision)); } else throw uhd::runtime_error("Error loading OctoClock EEPROM."); } void octoclock_eeprom_t::_store() const { uint8_t octoclock_data[udp_simple::mtu]; const octoclock_packet_t* pkt_in = reinterpret_cast(octoclock_data); auto pkt_out = make_octoclock_packet(); pkt_out.len = sizeof(octoclock_fw_eeprom_t); size_t len = 0; octoclock_fw_eeprom_t* eeprom_out = reinterpret_cast(&pkt_out.data); memset(eeprom_out, 0xFF, sizeof(octoclock_fw_eeprom_t)); // MAC address if ((*this).has_key("mac-addr")) { byte_copy(mac_addr_t::from_string((*this)["mac-addr"]).to_bytes(), eeprom_out->mac_addr); } // IP address if ((*this).has_key("ip-addr")) { ip_v4::bytes_type ip_addr_bytes = ip_v4::from_string((*this)["ip-addr"]).to_bytes(); memcpy(&eeprom_out->ip_addr, &ip_addr_bytes, 4); eeprom_out->ip_addr = uhd::htonx(eeprom_out->ip_addr); } // Default router if ((*this).has_key("gateway")) { ip_v4::bytes_type dr_addr_bytes = ip_v4::from_string((*this)["gateway"]).to_bytes(); memcpy(&eeprom_out->dr_addr, &dr_addr_bytes, 4); eeprom_out->dr_addr = uhd::htonx(eeprom_out->dr_addr); } // Netmask if ((*this).has_key("netmask")) { ip_v4::bytes_type netmask_bytes = ip_v4::from_string((*this)["netmask"]).to_bytes(); memcpy(&eeprom_out->netmask, &netmask_bytes, 4); eeprom_out->netmask = uhd::htonx(eeprom_out->netmask); } // Serial if ((*this).has_key("serial")) { byte_copy(byte_vector_t((*this)["serial"].begin(), (*this)["serial"].end()), eeprom_out->serial); } // Name if ((*this).has_key("name")) { byte_copy(byte_vector_t((*this)["name"].begin(), (*this)["name"].end()), eeprom_out->name); } // Revision if ((*this).has_key("revision")) { eeprom_out->revision = (*this)["revision"][0] - '0'; } UHD_OCTOCLOCK_SEND_AND_RECV( xport, _proto_ver, BURN_EEPROM_CMD, pkt_out, len, octoclock_data); if (not UHD_OCTOCLOCK_PACKET_MATCHES(BURN_EEPROM_SUCCESS_ACK, pkt_out, pkt_in, len)) throw uhd::runtime_error("Error writing to OctoClock EEPROM."); } /*********************************************************************** * Implementation of OctoClock EEPROM **********************************************************************/ octoclock_eeprom_t::octoclock_eeprom_t(void) { /* NOP */ } octoclock_eeprom_t::octoclock_eeprom_t(udp_simple::sptr transport, uint32_t proto_ver) : xport(transport), _proto_ver(proto_ver) { _load(); } void octoclock_eeprom_t::commit() const { if (!xport) throw uhd::runtime_error("There is no set device communication."); _store(); }