| #include "config.h" |
| |
| #include "dns_updater.hpp" |
| #include "network_manager.hpp" |
| #include "rtnetlink_server.hpp" |
| #include "types.hpp" |
| #include "watch.hpp" |
| |
| #include <linux/netlink.h> |
| |
| #include <functional> |
| #include <memory> |
| #include <phosphor-logging/elog-errors.hpp> |
| #include <phosphor-logging/log.hpp> |
| #include <sdbusplus/bus.hpp> |
| #include <sdbusplus/server/manager.hpp> |
| #include <sdeventplus/event.hpp> |
| #include <xyz/openbmc_project/Common/error.hpp> |
| |
| using phosphor::logging::elog; |
| using phosphor::logging::entry; |
| using phosphor::logging::level; |
| using phosphor::logging::log; |
| using sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; |
| |
| namespace phosphor |
| { |
| namespace network |
| { |
| |
| std::unique_ptr<phosphor::network::Manager> manager = nullptr; |
| std::unique_ptr<Timer> refreshObjectTimer = nullptr; |
| std::unique_ptr<Timer> restartTimer = nullptr; |
| |
| /** @brief refresh the network objects. */ |
| void refreshObjects() |
| { |
| if (manager) |
| { |
| log<level::INFO>("Refreshing the objects."); |
| manager->createChildObjects(); |
| log<level::INFO>("Refreshing complete."); |
| } |
| } |
| |
| /** @brief restart the systemd networkd. */ |
| void restartNetwork() |
| { |
| if (manager) |
| { |
| manager->restartSystemdUnit("systemd-networkd.service"); |
| } |
| } |
| |
| void initializeTimers() |
| { |
| auto event = sdeventplus::Event::get_default(); |
| refreshObjectTimer = |
| std::make_unique<Timer>(event, std::bind(refreshObjects)); |
| restartTimer = std::make_unique<Timer>(event, std::bind(restartNetwork)); |
| } |
| |
| } // namespace network |
| } // namespace phosphor |
| |
| void createNetLinkSocket(phosphor::Descriptor& smartSock) |
| { |
| // RtnetLink socket |
| auto fd = socket(PF_NETLINK, SOCK_RAW | SOCK_NONBLOCK, NETLINK_ROUTE); |
| if (fd < 0) |
| { |
| log<level::ERR>("Unable to create the net link socket", |
| entry("ERRNO=%d", errno)); |
| elog<InternalFailure>(); |
| } |
| smartSock.set(fd); |
| } |
| |
| int main(int argc, char* argv[]) |
| { |
| phosphor::network::initializeTimers(); |
| |
| auto bus = sdbusplus::bus::new_default(); |
| |
| // Need sd_event to watch for OCC device errors |
| sd_event* event = nullptr; |
| auto r = sd_event_default(&event); |
| if (r < 0) |
| { |
| log<level::ERR>("Error creating a default sd_event handler"); |
| return r; |
| } |
| |
| phosphor::network::EventPtr eventPtr{event}; |
| event = nullptr; |
| |
| // Attach the bus to sd_event to service user requests |
| bus.attach_event(eventPtr.get(), SD_EVENT_PRIORITY_NORMAL); |
| |
| // Add sdbusplus Object Manager for the 'root' path of the network manager. |
| sdbusplus::server::manager::manager objManager(bus, OBJ_NETWORK); |
| bus.request_name(BUSNAME_NETWORK); |
| |
| phosphor::network::manager = std::make_unique<phosphor::network::Manager>( |
| bus, OBJ_NETWORK, NETWORK_CONF_DIR); |
| |
| // create the default network files if the network file |
| // is not there for any interface. |
| // Parameter false means don't create the network |
| // files forcefully. |
| if (phosphor::network::manager->createDefaultNetworkFiles(false)) |
| { |
| // if files created restart the network. |
| // don't need to call the create child objects as eventhandler |
| // will create it. |
| phosphor::network::restartNetwork(); |
| } |
| else |
| { |
| // this will add the additional fixes which is needed |
| // in the existing network file. |
| phosphor::network::manager->writeToConfigurationFile(); |
| // whenever the configuration file gets written it restart |
| // the network which creates the network objects |
| } |
| |
| // RtnetLink socket |
| phosphor::Descriptor smartSock; |
| createNetLinkSocket(smartSock); |
| |
| // RTNETLINK event handler |
| phosphor::network::rtnetlink::Server svr(eventPtr, smartSock); |
| |
| // DNS entry handler |
| phosphor::network::inotify::Watch watch( |
| eventPtr, DNS_ENTRY_FILE, |
| std::bind(&phosphor::network::dns::updater::processDNSEntries, |
| std::placeholders::_1)); |
| |
| // At this point, we have registered for the notifications for future |
| // events. However, if the file is already populated before this, then |
| // they won't ever get notified and thus we need to read once before |
| // waiting on change events |
| phosphor::network::dns::updater::processDNSEntries(DNS_ENTRY_FILE); |
| |
| sd_event_loop(eventPtr.get()); |
| } |