blob: b3b9bb7fc9eccc86777b69dcbe8fdf26c5b2b410 [file] [log] [blame]
Gunnar Mills57d9c502018-09-14 14:42:34 -05001#include "config.h"
2
Patrick Venture189d44e2018-07-09 12:30:59 -07003#include "ethernet_interface.hpp"
4
Ratan Gupta497c0c92017-08-22 19:15:59 +05305#include "config_parser.hpp"
Ratan Gupta4f1c18b2017-05-25 12:59:35 +05306#include "network_manager.hpp"
William A. Kennington III2e09d272022-10-14 17:15:00 -07007#include "system_queries.hpp"
William A. Kennington III95530ec2022-08-19 01:44:39 -07008#include "util.hpp"
Ratan Gupta91a99cc2017-04-14 16:32:09 +05309
William A. Kennington IIIa520a392022-08-08 12:17:34 -070010#include <fmt/compile.h>
William A. Kennington III26275a32021-07-13 20:32:42 -070011#include <fmt/format.h>
William A. Kennington IIId7946a72019-04-19 14:24:09 -070012#include <linux/rtnetlink.h>
William A. Kennington IIIfd862be2022-10-09 18:40:55 -070013#include <net/if.h>
William A. Kennington III9c441fd2023-02-24 13:40:01 -080014#include <net/if_arp.h>
Ratan Gupta91a99cc2017-04-14 16:32:09 +053015
Patrick Venture189d44e2018-07-09 12:30:59 -070016#include <phosphor-logging/elog-errors.hpp>
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -070017#include <phosphor-logging/lg2.hpp>
William A. Kennington III12beaad2020-06-13 19:30:41 -070018#include <stdplus/raw.hpp>
William A. Kennington III69f45542022-09-24 23:28:14 -070019#include <stdplus/zstring.hpp>
Patrick Williams89d734b2023-05-10 07:50:25 -050020#include <xyz/openbmc_project/Common/error.hpp>
21
22#include <algorithm>
23#include <filesystem>
Ratan Gupta2b106532017-07-25 16:05:02 +053024#include <string>
William A. Kennington III26275a32021-07-13 20:32:42 -070025#include <unordered_map>
26#include <variant>
Ratan Gupta82549cc2017-04-21 08:45:23 +053027
Ratan Gupta91a99cc2017-04-14 16:32:09 +053028namespace phosphor
29{
30namespace network
31{
32
33using namespace phosphor::logging;
Ratan Gupta2b106532017-07-25 16:05:02 +053034using namespace sdbusplus::xyz::openbmc_project::Common::Error;
Manojkiran Edaacd6dd52019-10-15 15:00:51 +053035using NotAllowed = sdbusplus::xyz::openbmc_project::Common::Error::NotAllowed;
36using NotAllowedArgument = xyz::openbmc_project::Common::NotAllowed;
Nagaraju Goruganti66b974d2017-10-03 08:43:08 -050037using Argument = xyz::openbmc_project::Common::InvalidArgument;
William A. Kennington III991a8e82022-10-11 15:02:47 -070038using std::literals::string_view_literals::operator""sv;
Manojkiran Edaacd6dd52019-10-15 15:00:51 +053039constexpr auto RESOLVED_SERVICE = "org.freedesktop.resolve1";
40constexpr auto RESOLVED_INTERFACE = "org.freedesktop.resolve1.Link";
41constexpr auto PROPERTY_INTERFACE = "org.freedesktop.DBus.Properties";
42constexpr auto RESOLVED_SERVICE_PATH = "/org/freedesktop/resolve1/link/";
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -060043
44constexpr auto TIMESYNCD_SERVICE = "org.freedesktop.timesync1";
45constexpr auto TIMESYNCD_INTERFACE = "org.freedesktop.timesync1.Manager";
46constexpr auto TIMESYNCD_SERVICE_PATH = "/org/freedesktop/timesync1";
47
Manojkiran Edaacd6dd52019-10-15 15:00:51 +053048constexpr auto METHOD_GET = "Get";
Ratan Gupta2b106532017-07-25 16:05:02 +053049
William A. Kennington III2e09d272022-10-14 17:15:00 -070050template <typename Func>
51inline decltype(std::declval<Func>()())
52 ignoreError(std::string_view msg, stdplus::zstring_view intf,
53 decltype(std::declval<Func>()()) fallback, Func&& func) noexcept
William A. Kennington III5dad2aa2022-01-21 16:00:17 -080054{
William A. Kennington III4ee7a7e2022-10-11 16:37:22 -070055 try
56 {
William A. Kennington III2e09d272022-10-14 17:15:00 -070057 return func();
William A. Kennington III4ee7a7e2022-10-11 16:37:22 -070058 }
William A. Kennington III2e09d272022-10-14 17:15:00 -070059 catch (const std::exception& e)
William A. Kennington III4ee7a7e2022-10-11 16:37:22 -070060 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -070061 lg2::error("{MSG} failed on {NET_INTF}: {ERROR}", "MSG", msg,
62 "NET_INTF", intf, "ERROR", e);
William A. Kennington III4ee7a7e2022-10-11 16:37:22 -070063 }
William A. Kennington III2e09d272022-10-14 17:15:00 -070064 return fallback;
William A. Kennington III4ee7a7e2022-10-11 16:37:22 -070065}
William A. Kennington IIId298f932022-10-17 14:31:38 -070066
William A. Kennington IIId298f932022-10-17 14:31:38 -070067static std::string makeObjPath(std::string_view root, std::string_view intf)
68{
69 auto ret = fmt::format(FMT_COMPILE("{}/{}"), root, intf);
70 std::replace(ret.begin() + ret.size() - intf.size(), ret.end(), '.', '_');
71 return ret;
72}
73
William A. Kennington IIIbbf5e9e2023-02-01 14:58:38 -080074template <typename Addr>
75static bool validIntfIP(Addr a) noexcept
76{
77 return a.isUnicast() && !a.isLoopback();
78}
79
William A. Kennington III9ede1b72022-11-21 01:59:28 -080080EthernetInterface::EthernetInterface(stdplus::PinnedRef<sdbusplus::bus_t> bus,
81 stdplus::PinnedRef<Manager> manager,
William A. Kennington III13d665c2022-11-15 20:34:40 -080082 const AllIntfInfo& info,
William A. Kennington IIId298f932022-10-17 14:31:38 -070083 std::string_view objRoot,
William A. Kennington IIIa520a392022-08-08 12:17:34 -070084 const config::Parser& config,
William A. Kennington IIIb6452542022-11-15 18:09:12 -080085 bool enabled) :
William A. Kennington III13d665c2022-11-15 20:34:40 -080086 EthernetInterface(bus, manager, info, makeObjPath(objRoot, *info.intf.name),
William A. Kennington IIIb6452542022-11-15 18:09:12 -080087 config, enabled)
Patrick Williams89d734b2023-05-10 07:50:25 -050088{}
William A. Kennington IIId298f932022-10-17 14:31:38 -070089
William A. Kennington III9ede1b72022-11-21 01:59:28 -080090EthernetInterface::EthernetInterface(stdplus::PinnedRef<sdbusplus::bus_t> bus,
91 stdplus::PinnedRef<Manager> manager,
William A. Kennington III13d665c2022-11-15 20:34:40 -080092 const AllIntfInfo& info,
William A. Kennington IIId298f932022-10-17 14:31:38 -070093 std::string&& objPath,
94 const config::Parser& config,
William A. Kennington IIIb6452542022-11-15 18:09:12 -080095 bool enabled) :
96 Ifaces(bus, objPath.c_str(), Ifaces::action::defer_emit),
William A. Kennington III876927c2022-11-17 16:20:15 -080097 manager(manager), bus(bus), objPath(std::move(objPath))
Ratan Gupta91a99cc2017-04-14 16:32:09 +053098{
William A. Kennington IIId99e6db2022-11-15 20:39:45 -080099 interfaceName(*info.intf.name, true);
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700100 auto dhcpVal = getDHCPValue(config);
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800101 EthernetInterfaceIntf::dhcp4(dhcpVal.v4, true);
102 EthernetInterfaceIntf::dhcp6(dhcpVal.v6, true);
103 EthernetInterfaceIntf::ipv6AcceptRA(getIPv6AcceptRA(config), true);
104 EthernetInterfaceIntf::nicEnabled(enabled, true);
Ravi Tejaa5a09442020-07-17 00:57:33 -0500105
William A. Kennington IIIe21a5cf2022-08-09 12:19:14 -0700106 EthernetInterfaceIntf::ntpServers(
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800107 config.map.getValueStrings("Network", "NTP"), true);
Ratan Gupta613a0122020-04-24 15:18:53 +0530108
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800109 updateInfo(info.intf, true);
110
111 if (info.defgw4)
112 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700113 EthernetInterface::defaultGateway(stdplus::toStr(*info.defgw4), true);
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800114 }
115 if (info.defgw6)
116 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700117 EthernetInterface::defaultGateway6(stdplus::toStr(*info.defgw6), true);
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800118 }
119 emit_object_added();
William A. Kennington IIId298f932022-10-17 14:31:38 -0700120
William A. Kennington III13d665c2022-11-15 20:34:40 -0800121 if (info.intf.vlan_id)
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700122 {
William A. Kennington III13d665c2022-11-15 20:34:40 -0800123 if (!info.intf.parent_idx)
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700124 {
125 std::runtime_error("Missing parent link");
126 }
William A. Kennington III13d665c2022-11-15 20:34:40 -0800127 vlan.emplace(bus, this->objPath.c_str(), info.intf, *this);
128 }
William A. Kennington III13d665c2022-11-15 20:34:40 -0800129 for (const auto& [_, addr] : info.addrs)
130 {
131 addAddr(addr);
132 }
133 for (const auto& [_, neigh] : info.staticNeighs)
134 {
135 addStaticNeigh(neigh);
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700136 }
Ratan Gupta29b0e432017-05-25 12:51:40 +0530137}
138
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800139void EthernetInterface::updateInfo(const InterfaceInfo& info, bool skipSignal)
William A. Kennington IIId298f932022-10-17 14:31:38 -0700140{
William A. Kennington III876927c2022-11-17 16:20:15 -0800141 ifIdx = info.idx;
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800142 EthernetInterfaceIntf::linkUp(info.flags & IFF_RUNNING, skipSignal);
William A. Kennington IIId298f932022-10-17 14:31:38 -0700143 if (info.mac)
144 {
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700145 MacAddressIntf::macAddress(stdplus::toStr(*info.mac), skipSignal);
William A. Kennington IIId298f932022-10-17 14:31:38 -0700146 }
147 if (info.mtu)
148 {
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800149 EthernetInterfaceIntf::mtu(*info.mtu, skipSignal);
William A. Kennington IIId298f932022-10-17 14:31:38 -0700150 }
William A. Kennington III21539662022-11-15 14:53:11 -0800151 if (ifIdx > 0)
152 {
153 auto ethInfo = ignoreError("GetEthInfo", *info.name, {}, [&] {
154 return system::getEthInfo(*info.name);
155 });
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800156 EthernetInterfaceIntf::autoNeg(ethInfo.autoneg, skipSignal);
157 EthernetInterfaceIntf::speed(ethInfo.speed, skipSignal);
William A. Kennington III21539662022-11-15 14:53:11 -0800158 }
William A. Kennington IIId298f932022-10-17 14:31:38 -0700159}
160
Johnathan Mantey817012a2020-01-30 15:07:39 -0800161bool EthernetInterface::originIsManuallyAssigned(IP::AddressOrigin origin)
162{
163 return (
164#ifdef LINK_LOCAL_AUTOCONFIGURATION
Johnathan Mantey817012a2020-01-30 15:07:39 -0800165 (origin == IP::AddressOrigin::Static ||
166 origin == IP::AddressOrigin::LinkLocal)
Tim Lee721d8f752023-07-08 00:26:27 +0800167#else
168 (origin == IP::AddressOrigin::Static)
Johnathan Mantey817012a2020-01-30 15:07:39 -0800169#endif
170
171 );
172}
173
William A. Kennington IIId6f53402022-11-07 14:48:53 -0800174void EthernetInterface::addAddr(const AddressInfo& info)
175{
William A. Kennington IIId6f53402022-11-07 14:48:53 -0800176 IP::AddressOrigin origin = IP::AddressOrigin::Static;
Ravi Teja86c4f442023-07-19 22:26:39 -0500177 if (dhcpIsEnabled(info.ifaddr.getAddr()))
178 {
179 origin = IP::AddressOrigin::DHCP;
180 }
181
William A. Kennington IIId6f53402022-11-07 14:48:53 -0800182#ifdef LINK_LOCAL_AUTOCONFIGURATION
183 if (info.scope == RT_SCOPE_LINK)
184 {
185 origin = IP::AddressOrigin::LinkLocal;
186 }
187#endif
188
Ravi Tejac6201202023-03-28 21:29:58 -0500189 if ((info.scope == RT_SCOPE_UNIVERSE) && (info.flags & IFA_F_PERMANENT))
190 {
191 origin = IP::AddressOrigin::Static;
192 }
193 if ((info.scope == RT_SCOPE_UNIVERSE) &&
194 ((info.flags & IFA_F_NOPREFIXROUTE) &&
195 (info.flags & IFA_F_MANAGETEMPADDR)))
196 {
197 origin = IP::AddressOrigin::SLAAC;
198 }
199 else if ((info.scope == RT_SCOPE_UNIVERSE) &&
200 ((info.flags & IFA_F_NOPREFIXROUTE)))
201 {
202 origin = IP::AddressOrigin::DHCP;
203 }
204
William A. Kennington III77747f62022-11-07 23:11:15 -0800205 auto it = addrs.find(info.ifaddr);
206 if (it == addrs.end())
207 {
208 addrs.emplace(info.ifaddr, std::make_unique<IPAddress>(
209 bus, std::string_view(objPath), *this,
210 info.ifaddr, origin));
211 }
212 else
213 {
214 it->second->IPIfaces::origin(origin);
215 }
William A. Kennington IIId6f53402022-11-07 14:48:53 -0800216}
217
William A. Kennington IIIbc2502c2022-11-07 16:31:54 -0800218void EthernetInterface::addStaticNeigh(const NeighborInfo& info)
219{
William A. Kennington IIId3615142022-11-12 01:28:33 -0800220 if (!info.mac || !info.addr)
William A. Kennington IIIbc2502c2022-11-07 16:31:54 -0800221 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700222 lg2::error("Missing neighbor mac on {NET_INTF}", "NET_INTF",
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700223 interfaceName());
William A. Kennington IIIbc2502c2022-11-07 16:31:54 -0800224 return;
225 }
William A. Kennington III4e75acc2022-11-14 19:13:31 -0800226
227 if (auto it = staticNeighbors.find(*info.addr); it != staticNeighbors.end())
228 {
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700229 it->second->NeighborObj::macAddress(stdplus::toStr(*info.mac));
William A. Kennington III4e75acc2022-11-14 19:13:31 -0800230 }
231 else
232 {
233 staticNeighbors.emplace(*info.addr, std::make_unique<Neighbor>(
234 bus, std::string_view(objPath),
235 *this, *info.addr, *info.mac,
236 Neighbor::State::Permanent));
237 }
William A. Kennington IIIbc2502c2022-11-07 16:31:54 -0800238}
239
Patrick Williams6aef7692021-05-01 06:39:41 -0500240ObjectPath EthernetInterface::ip(IP::Protocol protType, std::string ipaddress,
William A. Kennington IIIe25f8b42022-10-11 14:43:28 -0700241 uint8_t prefixLength, std::string)
Ratan Gupta82549cc2017-04-21 08:45:23 +0530242{
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700243 std::optional<stdplus::InAnyAddr> addr;
William A. Kennington III59e5b912022-11-02 02:49:46 -0700244 try
Ratan Guptafc2c7242017-05-29 08:46:06 +0530245 {
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700246 switch (protType)
247 {
248 case IP::Protocol::IPv4:
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700249 addr.emplace(stdplus::fromStr<stdplus::In4Addr>(ipaddress));
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700250 break;
251 case IP::Protocol::IPv6:
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700252 addr.emplace(stdplus::fromStr<stdplus::In6Addr>(ipaddress));
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700253 break;
William A. Kennington III59e5b912022-11-02 02:49:46 -0700254 default:
255 throw std::logic_error("Exhausted protocols");
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700256 }
William A. Kennington III30f3ba22023-06-28 15:50:12 -0700257 if (!std::visit([](auto ip) { return validIntfIP(ip); }, *addr))
258 {
259 throw std::invalid_argument("not unicast");
260 }
Nagaraju Goruganti66b974d2017-10-03 08:43:08 -0500261 }
William A. Kennington III59e5b912022-11-02 02:49:46 -0700262 catch (const std::exception& e)
Nagaraju Goruganti66b974d2017-10-03 08:43:08 -0500263 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700264 lg2::error("Invalid IP {NET_IP}: {ERROR}", "NET_IP", ipaddress, "ERROR",
265 e);
Nagaraju Goruganti66b974d2017-10-03 08:43:08 -0500266 elog<InvalidArgument>(Argument::ARGUMENT_NAME("ipaddress"),
267 Argument::ARGUMENT_VALUE(ipaddress.c_str()));
268 }
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700269 std::optional<stdplus::SubnetAny> ifaddr;
William A. Kennington III59e5b912022-11-02 02:49:46 -0700270 try
Nagaraju Goruganti66b974d2017-10-03 08:43:08 -0500271 {
William A. Kennington III30f3ba22023-06-28 15:50:12 -0700272 if (prefixLength == 0)
273 {
274 throw std::invalid_argument("default route");
275 }
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700276 ifaddr.emplace(*addr, prefixLength);
William A. Kennington III59e5b912022-11-02 02:49:46 -0700277 }
278 catch (const std::exception& e)
279 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700280 lg2::error("Invalid prefix length {NET_PFX}: {ERROR}", "NET_PFX",
281 prefixLength, "ERROR", e);
Gunnar Mills57d9c502018-09-14 14:42:34 -0500282 elog<InvalidArgument>(
283 Argument::ARGUMENT_NAME("prefixLength"),
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700284 Argument::ARGUMENT_VALUE(stdplus::toStr(prefixLength).c_str()));
Ratan Guptafc2c7242017-05-29 08:46:06 +0530285 }
286
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700287 auto it = addrs.find(*ifaddr);
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800288 if (it == addrs.end())
289 {
290 it = std::get<0>(addrs.emplace(
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700291 *ifaddr,
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800292 std::make_unique<IPAddress>(bus, std::string_view(objPath), *this,
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700293 *ifaddr, IP::AddressOrigin::Static)));
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800294 }
295 else
296 {
William A. Kennington IIIadd08712022-11-17 16:31:19 -0800297 if (it->second->origin() == IP::AddressOrigin::Static)
298 {
299 return it->second->getObjPath();
300 }
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800301 it->second->IPIfaces::origin(IP::AddressOrigin::Static);
302 }
Ratan Gupta4f1c18b2017-05-25 12:59:35 +0530303
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700304 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800305 manager.get().reloadConfigs();
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700306
William A. Kennington III434a9432022-11-04 18:38:46 -0700307 return it->second->getObjPath();
Ratan Gupta82549cc2017-04-21 08:45:23 +0530308}
309
Patrick Williams6aef7692021-05-01 06:39:41 -0500310ObjectPath EthernetInterface::neighbor(std::string ipAddress,
311 std::string macAddress)
William A. Kennington III08505792019-01-30 16:00:04 -0800312{
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700313 std::optional<stdplus::InAnyAddr> addr;
William A. Kennington III434a9432022-11-04 18:38:46 -0700314 try
William A. Kennington III08505792019-01-30 16:00:04 -0800315 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700316 addr.emplace(stdplus::fromStr<stdplus::InAnyAddr>(ipAddress));
William A. Kennington III434a9432022-11-04 18:38:46 -0700317 }
318 catch (const std::exception& e)
319 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700320 lg2::error("Not a valid IP address {NET_IP}: {ERROR}", "NET_IP",
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700321 ipAddress, "ERROR", e);
Patrick Williams6aef7692021-05-01 06:39:41 -0500322 elog<InvalidArgument>(Argument::ARGUMENT_NAME("ipAddress"),
323 Argument::ARGUMENT_VALUE(ipAddress.c_str()));
William A. Kennington III08505792019-01-30 16:00:04 -0800324 }
William A. Kennington III434a9432022-11-04 18:38:46 -0700325
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700326 std::optional<stdplus::EtherAddr> lladdr;
William A. Kennington III434a9432022-11-04 18:38:46 -0700327 try
William A. Kennington III08505792019-01-30 16:00:04 -0800328 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700329 lladdr.emplace(stdplus::fromStr<stdplus::EtherAddr>(macAddress));
William A. Kennington III434a9432022-11-04 18:38:46 -0700330 }
331 catch (const std::exception& e)
332 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700333 lg2::error("Not a valid MAC address {NET_MAC}: {ERROR}", "NET_MAC",
334 macAddress, "ERROR", e);
Patrick Williams6aef7692021-05-01 06:39:41 -0500335 elog<InvalidArgument>(Argument::ARGUMENT_NAME("macAddress"),
336 Argument::ARGUMENT_VALUE(macAddress.c_str()));
William A. Kennington III08505792019-01-30 16:00:04 -0800337 }
338
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700339 auto it = staticNeighbors.find(*addr);
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800340 if (it == staticNeighbors.end())
341 {
342 it = std::get<0>(staticNeighbors.emplace(
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700343 *addr, std::make_unique<Neighbor>(bus, std::string_view(objPath),
344 *this, *addr, *lladdr,
345 Neighbor::State::Permanent)));
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800346 }
347 else
348 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700349 auto str = stdplus::toStr(*lladdr);
William A. Kennington IIIadd08712022-11-17 16:31:19 -0800350 if (it->second->macAddress() == str)
351 {
352 return it->second->getObjPath();
353 }
354 it->second->NeighborObj::macAddress(str);
William A. Kennington IIIa429c9d2022-11-15 18:00:45 -0800355 }
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700356
357 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800358 manager.get().reloadConfigs();
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700359
William A. Kennington III434a9432022-11-04 18:38:46 -0700360 return it->second->getObjPath();
William A. Kennington III08505792019-01-30 16:00:04 -0800361}
362
Patrick Williams6aef7692021-05-01 06:39:41 -0500363bool EthernetInterface::ipv6AcceptRA(bool value)
Johnathan Mantey5b023f52019-06-24 16:06:37 -0700364{
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700365 if (ipv6AcceptRA() != EthernetInterfaceIntf::ipv6AcceptRA(value))
Johnathan Mantey5b023f52019-06-24 16:06:37 -0700366 {
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700367 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800368 manager.get().reloadConfigs();
Johnathan Mantey5b023f52019-06-24 16:06:37 -0700369 }
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700370 return value;
371}
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700372
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700373bool EthernetInterface::dhcp4(bool value)
374{
375 if (dhcp4() != EthernetInterfaceIntf::dhcp4(value))
376 {
377 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800378 manager.get().reloadConfigs();
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700379 }
380 return value;
381}
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700382
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700383bool EthernetInterface::dhcp6(bool value)
384{
385 if (dhcp6() != EthernetInterfaceIntf::dhcp6(value))
386 {
387 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800388 manager.get().reloadConfigs();
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700389 }
Johnathan Mantey5b023f52019-06-24 16:06:37 -0700390 return value;
391}
392
Patrick Williams6aef7692021-05-01 06:39:41 -0500393EthernetInterface::DHCPConf EthernetInterface::dhcpEnabled(DHCPConf value)
Ratan Gupta87c13982017-06-15 09:27:27 +0530394{
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700395 auto old4 = EthernetInterfaceIntf::dhcp4();
396 auto new4 = EthernetInterfaceIntf::dhcp4(value == DHCPConf::v4 ||
397 value == DHCPConf::v4v6stateless ||
398 value == DHCPConf::both);
399 auto old6 = EthernetInterfaceIntf::dhcp6();
400 auto new6 = EthernetInterfaceIntf::dhcp6(value == DHCPConf::v6 ||
401 value == DHCPConf::both);
402 auto oldra = EthernetInterfaceIntf::ipv6AcceptRA();
403 auto newra = EthernetInterfaceIntf::ipv6AcceptRA(
404 value == DHCPConf::v6stateless || value == DHCPConf::v4v6stateless ||
405 value == DHCPConf::v6 || value == DHCPConf::both);
406
407 if (old4 != new4 || old6 != new6 || oldra != newra)
Ratan Gupta5978dd12017-07-25 13:47:13 +0530408 {
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700409 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800410 manager.get().reloadConfigs();
Ratan Gupta5978dd12017-07-25 13:47:13 +0530411 }
Ratan Gupta87c13982017-06-15 09:27:27 +0530412 return value;
413}
414
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700415EthernetInterface::DHCPConf EthernetInterface::dhcpEnabled() const
416{
417 if (dhcp6())
418 {
419 return dhcp4() ? DHCPConf::both : DHCPConf::v6;
420 }
421 else if (dhcp4())
422 {
423 return ipv6AcceptRA() ? DHCPConf::v4v6stateless : DHCPConf::v4;
424 }
425 return ipv6AcceptRA() ? DHCPConf::v6stateless : DHCPConf::none;
426}
427
Tejas Patil2c0fc562021-08-03 19:13:46 +0530428size_t EthernetInterface::mtu(size_t value)
429{
William A. Kennington III2e09d272022-10-14 17:15:00 -0700430 const size_t old = EthernetInterfaceIntf::mtu();
431 if (value == old)
Tejas Patil2c0fc562021-08-03 19:13:46 +0530432 {
433 return value;
434 }
William A. Kennington III2e09d272022-10-14 17:15:00 -0700435 const auto ifname = interfaceName();
William A. Kennington III3e471c52022-10-27 19:46:07 -0700436 return EthernetInterfaceIntf::mtu(ignoreError("SetMTU", ifname, old, [&] {
William A. Kennington III2e09d272022-10-14 17:15:00 -0700437 system::setMTU(ifname, value);
438 return value;
439 }));
Tejas Patil2c0fc562021-08-03 19:13:46 +0530440}
441
Patrick Williams6aef7692021-05-01 06:39:41 -0500442bool EthernetInterface::nicEnabled(bool value)
Johnathan Manteyd0679f92019-10-29 16:20:28 -0700443{
Patrick Williams6aef7692021-05-01 06:39:41 -0500444 if (value == EthernetInterfaceIntf::nicEnabled())
Johnathan Manteyd0679f92019-10-29 16:20:28 -0700445 {
446 return value;
447 }
448
William A. Kennington IIIc922d5e2022-01-21 01:08:31 -0800449 EthernetInterfaceIntf::nicEnabled(value);
Johnathan Manteyd0679f92019-10-29 16:20:28 -0700450 writeConfigurationFile();
William A. Kennington III329b5fb2021-11-09 17:19:30 -0800451 if (!value)
452 {
453 // We only need to bring down the interface, networkd will always bring
454 // up managed interfaces
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800455 manager.get().addReloadPreHook(
William A. Kennington III2e09d272022-10-14 17:15:00 -0700456 [ifname = interfaceName()]() { system::setNICUp(ifname, false); });
William A. Kennington III329b5fb2021-11-09 17:19:30 -0800457 }
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800458 manager.get().reloadConfigs();
Johnathan Manteyfaa72e52020-01-08 10:38:58 -0800459
460 return value;
461}
462
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530463ServerList EthernetInterface::staticNameServers(ServerList value)
464{
William A. Kennington IIIc9f672e2022-11-04 14:35:53 -0700465 for (auto& ip : value)
Manojkiran Eda5fb6c332019-08-21 16:37:29 +0530466 {
William A. Kennington IIIc9f672e2022-11-04 14:35:53 -0700467 try
Manojkiran Eda5fb6c332019-08-21 16:37:29 +0530468 {
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700469 ip = stdplus::toStr(stdplus::fromStr<stdplus::InAnyAddr>(ip));
William A. Kennington IIIc9f672e2022-11-04 14:35:53 -0700470 }
471 catch (const std::exception& e)
472 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700473 lg2::error("Not a valid IP address {NET_IP}: {ERROR}", "NET_IP", ip,
474 "ERROR", e);
William A. Kennington IIIc9f672e2022-11-04 14:35:53 -0700475 elog<InvalidArgument>(Argument::ARGUMENT_NAME("StaticNameserver"),
476 Argument::ARGUMENT_VALUE(ip.c_str()));
Manojkiran Eda5fb6c332019-08-21 16:37:29 +0530477 }
478 }
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700479
William A. Kennington IIIa7485302023-05-30 16:57:57 -0700480 value = EthernetInterfaceIntf::staticNameServers(std::move(value));
481
482 writeConfigurationFile();
483 manager.get().reloadConfigs();
484
485 return value;
Ratan Gupta6dec390f2017-08-20 15:28:12 +0530486}
487
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600488void EthernetInterface::loadNTPServers(const config::Parser& config)
489{
490 EthernetInterfaceIntf::ntpServers(getNTPServerFromTimeSyncd());
491 EthernetInterfaceIntf::staticNTPServers(
492 config.map.getValueStrings("Network", "NTP"));
493}
494
William A. Kennington IIIa520a392022-08-08 12:17:34 -0700495void EthernetInterface::loadNameServers(const config::Parser& config)
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530496{
497 EthernetInterfaceIntf::nameservers(getNameServerFromResolvd());
William A. Kennington IIIa520a392022-08-08 12:17:34 -0700498 EthernetInterfaceIntf::staticNameServers(
William A. Kennington III34bb3e22022-08-18 15:17:22 -0700499 config.map.getValueStrings("Network", "DNS"));
Ratan Gupta6dec390f2017-08-20 15:28:12 +0530500}
501
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600502ServerList EthernetInterface::getNTPServerFromTimeSyncd()
503{
504 ServerList servers; // Variable to capture the NTP Server IPs
Patrick Williams89d734b2023-05-10 07:50:25 -0500505 auto method = bus.get().new_method_call(TIMESYNCD_SERVICE,
506 TIMESYNCD_SERVICE_PATH,
507 PROPERTY_INTERFACE, METHOD_GET);
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600508
509 method.append(TIMESYNCD_INTERFACE, "LinkNTPServers");
510
511 try
512 {
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800513 auto reply = bus.get().call(method);
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600514 std::variant<ServerList> response;
515 reply.read(response);
516 servers = std::get<ServerList>(response);
517 }
518 catch (const sdbusplus::exception::SdBusError& e)
519 {
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700520 lg2::error("Failed to get NTP server information from "
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700521 "systemd-timesyncd: {ERROR}",
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700522 "ERROR", e);
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600523 }
524
525 return servers;
526}
527
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530528ServerList EthernetInterface::getNameServerFromResolvd()
529{
530 ServerList servers;
William A. Kennington III2e09d272022-10-14 17:15:00 -0700531 auto OBJ_PATH = fmt::format("{}{}", RESOLVED_SERVICE_PATH, ifIdx);
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530532
533 /*
534 The DNS property under org.freedesktop.resolve1.Link interface contains
535 an array containing all DNS servers currently used by resolved. It
536 contains similar information as the DNS server data written to
537 /run/systemd/resolve/resolv.conf.
538
539 Each structure in the array consists of a numeric network interface index,
540 an address family, and a byte array containing the DNS server address
541 (either 4 bytes in length for IPv4 or 16 bytes in lengths for IPv6).
542 The array contains DNS servers configured system-wide, including those
543 possibly read from a foreign /etc/resolv.conf or the DNS= setting in
544 /etc/systemd/resolved.conf, as well as per-interface DNS server
545 information either retrieved from systemd-networkd or configured by
546 external software via SetLinkDNS().
547 */
548
549 using type = std::vector<std::tuple<int32_t, std::vector<uint8_t>>>;
550 std::variant<type> name; // Variable to capture the DNS property
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800551 auto method = bus.get().new_method_call(RESOLVED_SERVICE, OBJ_PATH.c_str(),
552 PROPERTY_INTERFACE, METHOD_GET);
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530553
554 method.append(RESOLVED_INTERFACE, "DNS");
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530555
556 try
557 {
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800558 auto reply = bus.get().call(method);
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530559 reply.read(name);
560 }
Patrick Williamsc38b0712022-07-22 19:26:54 -0500561 catch (const sdbusplus::exception_t& e)
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530562 {
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700563 lg2::error(
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700564 "Failed to get DNS information from systemd-resolved: {ERROR}",
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700565 "ERROR", e);
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530566 }
567 auto tupleVector = std::get_if<type>(&name);
568 for (auto i = tupleVector->begin(); i != tupleVector->end(); ++i)
569 {
Alexander Filippov983da552021-02-08 15:26:54 +0300570 int addressFamily = std::get<0>(*i);
571 std::vector<uint8_t>& ipaddress = std::get<1>(*i);
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700572 servers.push_back(stdplus::toStr(
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -0700573 addrFromBuf(addressFamily, stdplus::raw::asView<char>(ipaddress))));
Manojkiran Edaacd6dd52019-10-15 15:00:51 +0530574 }
575 return servers;
576}
577
William A. Kennington IIId298f932022-10-17 14:31:38 -0700578ObjectPath EthernetInterface::createVLAN(uint16_t id)
Ratan Gupta5978dd12017-07-25 13:47:13 +0530579{
William A. Kennington III9b2a20d2023-06-17 14:05:48 -0700580 auto idStr = stdplus::toStr(id);
581 auto intfName = fmt::format(FMT_COMPILE("{}.{}"), interfaceName(), idStr);
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800582 if (manager.get().interfaces.find(intfName) !=
583 manager.get().interfaces.end())
Jiaqing Zhao33b4eaa2022-04-12 23:11:40 +0800584 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700585 lg2::error("VLAN {NET_VLAN} already exists", "NET_VLAN", id);
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700586 elog<InvalidArgument>(Argument::ARGUMENT_NAME("VLANId"),
587 Argument::ARGUMENT_VALUE(idStr.c_str()));
Jiaqing Zhao33b4eaa2022-04-12 23:11:40 +0800588 }
589
William A. Kennington IIId298f932022-10-17 14:31:38 -0700590 auto objRoot = std::string_view(objPath).substr(0, objPath.rfind('/'));
William A. Kennington IIIfd862be2022-10-09 18:40:55 -0700591 auto macStr = MacAddressIntf::macAddress();
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700592 std::optional<stdplus::EtherAddr> mac;
William A. Kennington IIIfd862be2022-10-09 18:40:55 -0700593 if (!macStr.empty())
594 {
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700595 mac.emplace(stdplus::fromStr<stdplus::EtherAddr>(macStr));
William A. Kennington IIIfd862be2022-10-09 18:40:55 -0700596 }
William A. Kennington III13d665c2022-11-15 20:34:40 -0800597 auto info = AllIntfInfo{InterfaceInfo{
William A. Kennington III9c441fd2023-02-24 13:40:01 -0800598 .type = ARPHRD_ETHER,
William A. Kennington IIIfd862be2022-10-09 18:40:55 -0700599 .idx = 0, // TODO: Query the correct value after creation
600 .flags = 0,
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700601 .name = intfName,
William A. Kennington IIIfd862be2022-10-09 18:40:55 -0700602 .mac = std::move(mac),
603 .mtu = mtu(),
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700604 .parent_idx = ifIdx,
605 .vlan_id = id,
William A. Kennington III13d665c2022-11-15 20:34:40 -0800606 }};
Ratan Gupta5978dd12017-07-25 13:47:13 +0530607
Patrick Williams6aef7692021-05-01 06:39:41 -0500608 // Pass the parents nicEnabled property, so that the child
Manojkiran Edaca8b91b2020-05-28 09:28:42 +0530609 // VLAN interface can inherit.
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700610 auto vlanIntf = std::make_unique<EthernetInterface>(
William A. Kennington IIIb6452542022-11-15 18:09:12 -0800611 bus, manager, info, objRoot, config::Parser(), nicEnabled());
William A. Kennington IIId298f932022-10-17 14:31:38 -0700612 ObjectPath ret = vlanIntf->objPath;
Ratan Gupta5978dd12017-07-25 13:47:13 +0530613
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800614 manager.get().interfaces.emplace(intfName, std::move(vlanIntf));
Ratan Gupta5978dd12017-07-25 13:47:13 +0530615
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700616 // write the device file for the vlan interface.
617 config::Parser config;
618 auto& netdev = config.map["NetDev"].emplace_back();
619 netdev["Name"].emplace_back(intfName);
620 netdev["Kind"].emplace_back("vlan");
621 config.map["VLAN"].emplace_back()["Id"].emplace_back(std::move(idStr));
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800622 config.writeFile(
623 config::pathForIntfDev(manager.get().getConfDir(), intfName));
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700624
625 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800626 manager.get().reloadConfigs();
William A. Kennington IIIf4b4ff82019-04-09 19:06:52 -0700627
William A. Kennington III7b90bc82022-11-17 14:55:12 -0800628 return ret;
Ratan Gupta5978dd12017-07-25 13:47:13 +0530629}
Ratan Gupta2b106532017-07-25 16:05:02 +0530630
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600631ServerList EthernetInterface::staticNTPServers(ServerList value)
Ratan Gupta497c0c92017-08-22 19:15:59 +0530632{
William A. Kennington IIIa7485302023-05-30 16:57:57 -0700633 value = EthernetInterfaceIntf::staticNTPServers(std::move(value));
Ratan Gupta497c0c92017-08-22 19:15:59 +0530634
William A. Kennington IIIa7485302023-05-30 16:57:57 -0700635 writeConfigurationFile();
636 manager.get().reloadConfigs();
637
638 return value;
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600639}
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700640
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600641ServerList EthernetInterface::ntpServers(ServerList /*servers*/)
642{
643 elog<NotAllowed>(NotAllowedArgument::REASON("ReadOnly Property"));
Ratan Gupta497c0c92017-08-22 19:15:59 +0530644}
Ratan Gupta2b106532017-07-25 16:05:02 +0530645// Need to merge the below function with the code which writes the
646// config file during factory reset.
647// TODO openbmc/openbmc#1751
648
649void EthernetInterface::writeConfigurationFile()
650{
William A. Kennington III95a49a22022-08-18 17:50:05 -0700651 config::Parser config;
652 config.map["Match"].emplace_back()["Name"].emplace_back(interfaceName());
Ratan Gupta2b106532017-07-25 16:05:02 +0530653 {
William A. Kennington III95a49a22022-08-18 17:50:05 -0700654 auto& link = config.map["Link"].emplace_back();
Johnathan Mantey609c12d2022-02-03 09:23:09 -0800655#ifdef PERSIST_MAC
William A. Kennington III95a49a22022-08-18 17:50:05 -0700656 auto mac = MacAddressIntf::macAddress();
657 if (!mac.empty())
658 {
659 link["MACAddress"].emplace_back(mac);
660 }
Johnathan Mantey609c12d2022-02-03 09:23:09 -0800661#endif
William A. Kennington III95a49a22022-08-18 17:50:05 -0700662 if (!EthernetInterfaceIntf::nicEnabled())
663 {
664 link["Unmanaged"].emplace_back("yes");
665 }
Johnathan Manteyd0679f92019-10-29 16:20:28 -0700666 }
William A. Kennington III95a49a22022-08-18 17:50:05 -0700667 {
668 auto& network = config.map["Network"].emplace_back();
669 auto& lla = network["LinkLocalAddressing"];
Oskar Senftad21fc22018-07-26 16:32:23 -0400670#ifdef LINK_LOCAL_AUTOCONFIGURATION
William A. Kennington III95a49a22022-08-18 17:50:05 -0700671 lla.emplace_back("yes");
Oskar Senftad21fc22018-07-26 16:32:23 -0400672#else
William A. Kennington III95a49a22022-08-18 17:50:05 -0700673 lla.emplace_back("no");
Oskar Senftad21fc22018-07-26 16:32:23 -0400674#endif
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700675 network["IPv6AcceptRA"].emplace_back(ipv6AcceptRA() ? "true" : "false");
676 network["DHCP"].emplace_back(dhcp4() ? (dhcp6() ? "true" : "ipv4")
677 : (dhcp6() ? "ipv6" : "false"));
Nagaraju Goruganti210420a2018-03-07 09:22:28 -0600678 {
William A. Kennington III95a49a22022-08-18 17:50:05 -0700679 auto& vlans = network["VLAN"];
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800680 for (const auto& [_, intf] : manager.get().interfaces)
William A. Kennington III95a49a22022-08-18 17:50:05 -0700681 {
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700682 if (intf->vlan && intf->vlan->parentIdx == ifIdx)
683 {
684 vlans.emplace_back(intf->interfaceName());
685 }
William A. Kennington III95a49a22022-08-18 17:50:05 -0700686 }
687 }
688 {
689 auto& ntps = network["NTP"];
Asmitha Karunanithi003b8b72022-01-06 04:17:59 -0600690 for (const auto& ntp : EthernetInterfaceIntf::staticNTPServers())
William A. Kennington III95a49a22022-08-18 17:50:05 -0700691 {
692 ntps.emplace_back(ntp);
693 }
694 }
695 {
696 auto& dnss = network["DNS"];
697 for (const auto& dns : EthernetInterfaceIntf::staticNameServers())
698 {
699 dnss.emplace_back(dns);
700 }
701 }
702 {
703 auto& address = network["Address"];
William A. Kennington III59e5b912022-11-02 02:49:46 -0700704 for (const auto& addr : addrs)
William A. Kennington III95a49a22022-08-18 17:50:05 -0700705 {
William A. Kennington III59e5b912022-11-02 02:49:46 -0700706 if (originIsManuallyAssigned(addr.second->origin()))
William A. Kennington III95a49a22022-08-18 17:50:05 -0700707 {
708 address.emplace_back(
709 fmt::format("{}/{}", addr.second->address(),
710 addr.second->prefixLength()));
711 }
712 }
713 }
714 {
715 auto& gateways = network["Gateway"];
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700716 if (!dhcp4())
William A. Kennington III95a49a22022-08-18 17:50:05 -0700717 {
718 auto gateway = EthernetInterfaceIntf::defaultGateway();
719 if (!gateway.empty())
720 {
721 gateways.emplace_back(gateway);
722 }
723 }
Ratan Gupta2b106532017-07-25 16:05:02 +0530724
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700725 if (!dhcp6())
William A. Kennington III95a49a22022-08-18 17:50:05 -0700726 {
727 auto gateway6 = EthernetInterfaceIntf::defaultGateway6();
728 if (!gateway6.empty())
729 {
730 gateways.emplace_back(gateway6);
731 }
732 }
Nagaraju Goruganti210420a2018-03-07 09:22:28 -0600733 }
Johnathan Mantey817012a2020-01-30 15:07:39 -0800734 }
William A. Kennington III95a49a22022-08-18 17:50:05 -0700735 config.map["IPv6AcceptRA"].emplace_back()["DHCPv6Client"].emplace_back(
William A. Kennington III8060c0d2022-08-18 19:19:34 -0700736 dhcp6() ? "true" : "false");
Ravi Tejaa5a09442020-07-17 00:57:33 -0500737 {
William A. Kennington III95a49a22022-08-18 17:50:05 -0700738 auto& neighbors = config.map["Neighbor"];
739 for (const auto& sneighbor : staticNeighbors)
Lei YUcb2d4082021-08-12 15:26:49 +0800740 {
William A. Kennington III95a49a22022-08-18 17:50:05 -0700741 auto& neighbor = neighbors.emplace_back();
742 neighbor["Address"].emplace_back(sneighbor.second->ipAddress());
743 neighbor["MACAddress"].emplace_back(sneighbor.second->macAddress());
Lei YUcb2d4082021-08-12 15:26:49 +0800744 }
Ravi Tejaa5a09442020-07-17 00:57:33 -0500745 }
Ravi Tejaa5a09442020-07-17 00:57:33 -0500746 {
William A. Kennington III95a49a22022-08-18 17:50:05 -0700747 auto& dhcp = config.map["DHCP"].emplace_back();
748 dhcp["ClientIdentifier"].emplace_back("mac");
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800749 const auto& conf = manager.get().getDHCPConf();
William A. Kennington III5b179382022-11-15 15:23:26 -0800750 auto dns_enabled = conf.dnsEnabled() ? "true" : "false";
751 dhcp["UseDNS"].emplace_back(dns_enabled);
752 dhcp["UseDomains"].emplace_back(dns_enabled);
753 dhcp["UseNTP"].emplace_back(conf.ntpEnabled() ? "true" : "false");
754 dhcp["UseHostname"].emplace_back(conf.hostNameEnabled() ? "true"
755 : "false");
756 dhcp["SendHostname"].emplace_back(conf.sendHostNameEnabled() ? "true"
757 : "false");
Ravi Tejaa5a09442020-07-17 00:57:33 -0500758 }
Patrick Williams89d734b2023-05-10 07:50:25 -0500759 auto path = config::pathForIntfConf(manager.get().getConfDir(),
760 interfaceName());
William A. Kennington III95a49a22022-08-18 17:50:05 -0700761 config.writeFile(path);
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700762 lg2::info("Wrote networkd file: {CFG_FILE}", "CFG_FILE", path);
Ratan Gupta2b106532017-07-25 16:05:02 +0530763}
764
Jiaqing Zhao69cfa312022-02-18 16:52:55 +0800765std::string EthernetInterface::macAddress([[maybe_unused]] std::string value)
Ratan Guptabd303b12017-08-18 17:10:07 +0530766{
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700767 if (vlan)
768 {
Jagpal Singh Gillf78a4152023-04-17 21:17:56 -0700769 lg2::error("Tried to set MAC address on VLAN");
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700770 elog<InternalFailure>();
771 }
Jiaqing Zhao69cfa312022-02-18 16:52:55 +0800772#ifdef PERSIST_MAC
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700773 stdplus::EtherAddr newMAC;
Asmitha Karunanithi86f659e2021-01-05 00:16:03 -0600774 try
775 {
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700776 newMAC = stdplus::fromStr<stdplus::EtherAddr>(value);
Asmitha Karunanithi86f659e2021-01-05 00:16:03 -0600777 }
Patrick Williams5758db32021-10-06 12:29:22 -0500778 catch (const std::invalid_argument&)
Asmitha Karunanithi86f659e2021-01-05 00:16:03 -0600779 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700780 lg2::error("MAC Address {NET_MAC} is not valid", "NET_MAC", value);
Asmitha Karunanithi86f659e2021-01-05 00:16:03 -0600781 elog<InvalidArgument>(Argument::ARGUMENT_NAME("MACAddress"),
782 Argument::ARGUMENT_VALUE(value.c_str()));
783 }
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700784 if (!newMAC.isUnicast())
Ratan Guptabd303b12017-08-18 17:10:07 +0530785 {
William A. Kennington III1d25ca42023-05-30 16:55:28 -0700786 lg2::error("MAC Address {NET_MAC} is not valid", "NET_MAC", value);
Gunnar Mills90480c42018-06-19 16:02:17 -0500787 elog<InvalidArgument>(Argument::ARGUMENT_NAME("MACAddress"),
788 Argument::ARGUMENT_VALUE(value.c_str()));
Ratan Guptabd303b12017-08-18 17:10:07 +0530789 }
790
Alexander Filippov76b2aa32020-07-10 13:28:55 +0300791 auto interface = interfaceName();
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700792 auto validMAC = stdplus::toStr(newMAC);
Alexander Filippov76b2aa32020-07-10 13:28:55 +0300793
William A. Kennington III1137a972019-04-20 20:49:58 -0700794 // We don't need to update the system if the address is unchanged
William A. Kennington IIIb7d6a1a2023-06-17 02:00:53 -0700795 auto oldMAC =
796 stdplus::fromStr<stdplus::EtherAddr>(MacAddressIntf::macAddress());
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -0700797 if (newMAC != oldMAC)
Ratan Guptabd303b12017-08-18 17:10:07 +0530798 {
William A. Kennington III1137a972019-04-20 20:49:58 -0700799 // Update everything that depends on the MAC value
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800800 for (const auto& [_, intf] : manager.get().interfaces)
Ratan Guptabd303b12017-08-18 17:10:07 +0530801 {
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700802 if (intf->vlan && intf->vlan->parentIdx == ifIdx)
803 {
804 intf->MacAddressIntf::macAddress(validMAC);
805 }
Ratan Guptabd303b12017-08-18 17:10:07 +0530806 }
Patrick Williams6aef7692021-05-01 06:39:41 -0500807 MacAddressIntf::macAddress(validMAC);
Ratan Guptabd303b12017-08-18 17:10:07 +0530808
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700809 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800810 manager.get().addReloadPreHook([interface]() {
William A. Kennington III6bfdf3e2021-11-09 17:15:12 -0800811 // The MAC and LLADDRs will only update if the NIC is already down
William A. Kennington III2e09d272022-10-14 17:15:00 -0700812 system::setNICUp(interface, false);
William A. Kennington III6bfdf3e2021-11-09 17:15:12 -0800813 });
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800814 manager.get().reloadConfigs();
Ratan Gupta677ae122017-09-18 16:28:50 +0530815 }
William A. Kennington III1137a972019-04-20 20:49:58 -0700816
Alexander Filippov76b2aa32020-07-10 13:28:55 +0300817#ifdef HAVE_UBOOT_ENV
818 // Ensure that the valid address is stored in the u-boot-env
William A. Kennington III69f45542022-09-24 23:28:14 -0700819 auto envVar = interfaceToUbootEthAddr(interface);
Alexander Filippov76b2aa32020-07-10 13:28:55 +0300820 if (envVar)
821 {
Asmitha Karunanithi33bc9a92020-08-13 08:48:33 -0500822 // Trimming MAC addresses that are out of range. eg: AA:FF:FF:FF:FF:100;
823 // and those having more than 6 bytes. eg: AA:AA:AA:AA:AA:AA:BB
824 execute("/sbin/fw_setenv", "fw_setenv", envVar->c_str(),
825 validMAC.c_str());
Alexander Filippov76b2aa32020-07-10 13:28:55 +0300826 }
827#endif // HAVE_UBOOT_ENV
828
William A. Kennington III1137a972019-04-20 20:49:58 -0700829 return value;
Jiaqing Zhao69cfa312022-02-18 16:52:55 +0800830#else
831 elog<NotAllowed>(
832 NotAllowedArgument::REASON("Writing MAC address is not allowed"));
833#endif // PERSIST_MAC
Ratan Guptabd303b12017-08-18 17:10:07 +0530834}
835
Ratan Guptae9c9b812017-09-22 17:15:37 +0530836void EthernetInterface::deleteAll()
837{
Ratan Guptae9c9b812017-09-22 17:15:37 +0530838 // clear all the ip on the interface
839 addrs.clear();
William A. Kennington IIIbd649af2021-10-08 17:55:13 -0700840
841 writeConfigurationFile();
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800842 manager.get().reloadConfigs();
Ratan Guptae9c9b812017-09-22 17:15:37 +0530843}
844
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800845template <typename Addr>
846static void normalizeGateway(std::string& gw)
Ravi Tejaa5a09442020-07-17 00:57:33 -0500847{
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800848 if (gw.empty())
849 {
850 return;
851 }
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700852 try
Ravi Tejaa5a09442020-07-17 00:57:33 -0500853 {
William A. Kennington III9ca5c8e2022-12-14 14:40:52 -0800854 auto ip = stdplus::fromStr<Addr>(gw);
855 if (ip == Addr{})
856 {
857 gw.clear();
858 return;
859 }
William A. Kennington IIIbbf5e9e2023-02-01 14:58:38 -0800860 if (!validIntfIP(ip))
861 {
862 throw std::invalid_argument("Invalid unicast");
863 }
William A. Kennington III9ca5c8e2022-12-14 14:40:52 -0800864 gw = stdplus::toStr(ip);
Ravi Tejaa5a09442020-07-17 00:57:33 -0500865 }
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700866 catch (const std::exception& e)
Ravi Tejaa5a09442020-07-17 00:57:33 -0500867 {
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800868 lg2::error("Invalid GW `{NET_GW}`: {ERROR}", "NET_GW", gw, "ERROR", e);
Ravi Tejaa5a09442020-07-17 00:57:33 -0500869 elog<InvalidArgument>(Argument::ARGUMENT_NAME("GATEWAY"),
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800870 Argument::ARGUMENT_VALUE(gw.c_str()));
Ravi Tejaa5a09442020-07-17 00:57:33 -0500871 }
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800872}
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700873
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800874std::string EthernetInterface::defaultGateway(std::string gateway)
875{
876 normalizeGateway<stdplus::In4Addr>(gateway);
877 if (gateway != defaultGateway())
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700878 {
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800879 gateway = EthernetInterfaceIntf::defaultGateway(std::move(gateway));
880 manager.get().reloadConfigs();
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700881 }
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700882 return gateway;
Ravi Tejaa5a09442020-07-17 00:57:33 -0500883}
884
885std::string EthernetInterface::defaultGateway6(std::string gateway)
886{
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800887 normalizeGateway<stdplus::In6Addr>(gateway);
888 if (gateway != defaultGateway6())
Ravi Tejaa5a09442020-07-17 00:57:33 -0500889 {
William A. Kennington IIId4ff7312023-02-03 14:19:17 -0800890 gateway = EthernetInterfaceIntf::defaultGateway6(std::move(gateway));
891 manager.get().reloadConfigs();
Ravi Tejaa5a09442020-07-17 00:57:33 -0500892 }
William A. Kennington IIIf6f0cf32022-11-04 14:28:42 -0700893 return gateway;
Ravi Tejaa5a09442020-07-17 00:57:33 -0500894}
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700895
896EthernetInterface::VlanProperties::VlanProperties(
897 sdbusplus::bus_t& bus, stdplus::const_zstring objPath,
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800898 const InterfaceInfo& info, stdplus::PinnedRef<EthernetInterface> eth) :
William A. Kennington IIIb6452542022-11-15 18:09:12 -0800899 VlanIfaces(bus, objPath.c_str(), VlanIfaces::action::defer_emit),
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700900 parentIdx(*info.parent_idx), eth(eth)
901{
William A. Kennington IIId99e6db2022-11-15 20:39:45 -0800902 VlanIntf::id(*info.vlan_id, true);
903 emit_object_added();
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700904}
905
906void EthernetInterface::VlanProperties::delete_()
907{
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800908 auto intf = eth.get().interfaceName();
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700909
910 // Remove all configs for the current interface
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800911 const auto& confDir = eth.get().manager.get().getConfDir();
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700912 std::error_code ec;
913 std::filesystem::remove(config::pathForIntfConf(confDir, intf), ec);
914 std::filesystem::remove(config::pathForIntfDev(confDir, intf), ec);
915
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800916 if (eth.get().ifIdx > 0)
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800917 {
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800918 eth.get().manager.get().interfacesByIdx.erase(eth.get().ifIdx);
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800919 }
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800920 auto it = eth.get().manager.get().interfaces.find(intf);
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800921 auto obj = std::move(it->second);
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800922 eth.get().manager.get().interfaces.erase(it);
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800923
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700924 // Write an updated parent interface since it has a VLAN entry
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800925 for (const auto& [_, intf] : eth.get().manager.get().interfaces)
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700926 {
927 if (intf->ifIdx == parentIdx)
928 {
929 intf->writeConfigurationFile();
930 }
931 }
932
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800933 if (eth.get().ifIdx > 0)
William A. Kennington III67b09da2022-10-31 14:09:53 -0700934 {
William A. Kennington III6d217512022-11-17 16:18:01 -0800935 // We need to forcibly delete the interface as systemd does not
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800936 eth.get().manager.get().addReloadPostHook(
937 [idx = eth.get().ifIdx]() { system::deleteIntf(idx); });
William A. Kennington III6d217512022-11-17 16:18:01 -0800938
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800939 // Ignore the interface so the reload doesn't re-query it
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800940 eth.get().manager.get().ignoredIntf.emplace(eth.get().ifIdx);
William A. Kennington III67b09da2022-10-31 14:09:53 -0700941 }
William A. Kennington III93f5c6d2022-11-17 16:23:44 -0800942
William A. Kennington III9ede1b72022-11-21 01:59:28 -0800943 eth.get().manager.get().reloadConfigs();
William A. Kennington III09f3a4a2022-10-25 02:59:16 -0700944}
945
Gunnar Mills57d9c502018-09-14 14:42:34 -0500946} // namespace network
947} // namespace phosphor