#include "config.h" #include "network_manager.hpp" #include "ipaddress.hpp" #include "network_config.hpp" #include "types.hpp" #include "util.hpp" #include #include #include #include #include #include #include #include #include #include #include constexpr char SYSTEMD_BUSNAME[] = "org.freedesktop.systemd1"; constexpr char SYSTEMD_PATH[] = "/org/freedesktop/systemd1"; constexpr char SYSTEMD_INTERFACE[] = "org.freedesktop.systemd1.Manager"; namespace phosphor { namespace network { extern std::unique_ptr refreshObjectTimer; extern std::unique_ptr restartTimer; using namespace phosphor::logging; using namespace sdbusplus::xyz::openbmc_project::Common::Error; Manager::Manager(sdbusplus::bus::bus& bus, const char* objPath, const std::string& path) : details::VLANCreateIface(bus, objPath, true), bus(bus), objectPath(objPath) { fs::path confDir(path); setConfDir(confDir); } bool Manager::createDefaultNetworkFiles(bool force) { auto isCreated = false; try { // Directory would have created before with // setConfDir function. if (force) { // Factory Reset case // we need to forcefully write the files // so delete the existing ones. if (fs::is_directory(confDir)) { for (const auto& file : fs::directory_iterator(confDir)) { fs::remove(file.path()); } } } auto interfaceStrList = getInterfaces(); for (const auto& interface : interfaceStrList) { // if the interface has '.' in the name, it means that this is a // VLAN - don't create the network file. if (interface.find(".") != std::string::npos) { continue; } auto fileName = systemd::config::networkFilePrefix + interface + systemd::config::networkFileSuffix; fs::path filePath = confDir; filePath /= fileName; // create the interface specific network file // if not exist or we forcefully wants to write // the network file. if (force || !fs::is_regular_file(filePath.string())) { bmc::writeDHCPDefault(filePath.string(), interface); log("Created the default network file.", entry("INTERFACE=%s", interface.c_str())); isCreated = true; } } } catch (std::exception& e) { log("Unable to create the default network file"); } return isCreated; } void Manager::setConfDir(const fs::path& dir) { confDir = dir; if (!fs::exists(confDir)) { if (!fs::create_directories(confDir)) { log("Unable to create the network conf dir", entry("DIR=%s", confDir.c_str())); elog(); } } } void Manager::createInterfaces() { // clear all the interfaces first interfaces.clear(); auto interfaceStrList = getInterfaces(); for (auto& interface : interfaceStrList) { fs::path objPath = objectPath; auto index = interface.find("."); // interface can be of vlan type or normal ethernet interface. // vlan interface looks like "interface.vlanid",so here by looking // at the interface name we decide that we need // to create the vlaninterface or normal physical interface. if (index != std::string::npos) { // it is vlan interface auto interfaceName = interface.substr(0, index); auto vlanid = interface.substr(index + 1); uint32_t vlanInt = std::stoul(vlanid); interfaces[interfaceName]->loadVLAN(vlanInt); continue; } // normal ethernet interface objPath /= interface; auto dhcp = getDHCPValue(confDir, interface); auto intf = std::make_shared( bus, objPath.string(), dhcp, *this); intf->createIPAddressObjects(); this->interfaces.emplace( std::make_pair(std::move(interface), std::move(intf))); } } void Manager::createChildObjects() { // creates the ethernet interface dbus object. createInterfaces(); systemConf.reset(nullptr); dhcpConf.reset(nullptr); fs::path objPath = objectPath; objPath /= "config"; // create the system conf object. systemConf = std::make_unique( bus, objPath.string(), *this); // create the dhcp conf object. objPath /= "dhcp"; dhcpConf = std::make_unique( bus, objPath.string(), *this); } void Manager::vLAN(IntfName interfaceName, uint32_t id) { interfaces[interfaceName]->createVLAN(id); } void Manager::reset() { if (!createDefaultNetworkFiles(true)) { log("Network Factory Reset failed."); return; // TODO: openbmc/openbmc#1721 - Log ResetFailed error here. } log("Network Factory Reset done."); } // Need to merge the below function with the code which writes the // config file during factory reset. // TODO openbmc/openbmc#1751 void Manager::writeToConfigurationFile() { // write all the static ip address in the systemd-network conf file for (const auto& intf : interfaces) { intf.second->writeConfigurationFile(); } restartTimers(); } void Manager::restartTimers() { using namespace std::chrono; if (refreshObjectTimer && restartTimer) { restartTimer->restartOnce(restartTimeout); refreshObjectTimer->restartOnce(refreshTimeout); } } void Manager::restartSystemdUnit(const std::string& unit) { try { auto method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH, SYSTEMD_INTERFACE, "ResetFailedUnit"); method.append(unit); bus.call_noreply(method); } catch (const sdbusplus::exception::SdBusError& ex) { log("Failed to reset failed unit", entry("UNIT=%s", unit.c_str()), entry("ERR=%s", ex.what())); elog(); } try { auto method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH, SYSTEMD_INTERFACE, "RestartUnit"); method.append(unit.c_str(), "replace"); bus.call_noreply(method); } catch (const sdbusplus::exception::SdBusError& ex) { log("Failed to restart nslcd service", entry("ERR=%s", ex.what())); elog(); } } } // namespace network } // namespace phosphor