blob: 3b632c3d737c0e64067876a0d51a8b24e1f16b6f [file] [log] [blame]
Brandon Kimdab96f12021-02-18 11:21:37 -08001// Copyright 2021 Google LLC
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
William A. Kennington III7d6fa422021-02-08 17:04:02 -080015#include "net_config.h"
16
17#include <fmt/format.h>
18#include <sys/types.h>
19#include <sys/wait.h>
20#include <unistd.h>
21
22#include <sdbusplus/bus.hpp>
William A. Kennington IIIa5c9d7a2022-05-11 13:55:22 -070023#include <stdplus/fd/create.hpp>
24#include <stdplus/fd/ops.hpp>
William A. Kennington III7d6fa422021-02-08 17:04:02 -080025#include <stdplus/util/string.hpp>
26
27#include <cstdio>
28#include <cstring>
William A. Kennington IIIa5c9d7a2022-05-11 13:55:22 -070029#include <filesystem>
William A. Kennington III7d6fa422021-02-08 17:04:02 -080030#include <utility>
31#include <variant>
32
33/* Most of the code for interacting with DBus is from
34 * phosphor-host-ipmid/utils.cpp
35 */
36
37namespace net
38{
39
40namespace
41{
42
43constexpr auto IFACE_ROOT = "/xyz/openbmc_project/network/";
44constexpr auto MAC_FORMAT = "%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx";
45// 2 chars for every byte + 5 colons + Null byte
46constexpr auto MAC_FORMAT_LENGTH = 6 * 2 + 5 + 1;
47constexpr auto MAC_INTERFACE = "xyz.openbmc_project.Network.MACAddress";
48constexpr auto NETWORK_SERVICE = "xyz.openbmc_project.Network";
49constexpr auto PROP_INTERFACE = "org.freedesktop.DBus.Properties";
50
51int parse_mac(const std::string& mac_addr, mac_addr_t* mac)
52{
53 int ret =
54 sscanf(mac_addr.c_str(), MAC_FORMAT, mac->octet, mac->octet + 1,
55 mac->octet + 2, mac->octet + 3, mac->octet + 4, mac->octet + 5);
56
57 return ret < 6 ? -1 : 0;
58}
59
60std::string format_mac(const mac_addr_t& mac)
61{
62 // 2 chars for every byte + 5 colons + Null byte
63 char mac_str[MAC_FORMAT_LENGTH];
64 snprintf(mac_str, sizeof(mac_str), MAC_FORMAT, mac.octet[0], mac.octet[1],
65 mac.octet[2], mac.octet[3], mac.octet[4], mac.octet[5]);
66
67 return std::string{mac_str};
68}
69
70} // namespace
71
72PhosphorConfig::PhosphorConfig(const std::string& iface_name) :
73 iface_name_{iface_name}, iface_path_{std::string(IFACE_ROOT) + iface_name},
74 shared_host_mac_(std::experimental::nullopt),
75 bus(sdbusplus::bus::new_default())
76{}
77
78sdbusplus::message::message
79 PhosphorConfig::new_networkd_call(sdbusplus::bus::bus* dbus, bool get) const
80{
81 auto networkd_call =
82 dbus->new_method_call(NETWORK_SERVICE, iface_path_.c_str(),
83 PROP_INTERFACE, get ? "Get" : "Set");
84
85 networkd_call.append(MAC_INTERFACE, "MACAddress");
86
87 return networkd_call;
88}
89
90int PhosphorConfig::get_mac_addr(mac_addr_t* mac)
91{
92 if (mac == nullptr)
93 {
94 fmt::print(stderr, "mac is nullptr\n");
95 return -1;
96 }
97
98 // Cache hit: we have stored host MAC.
99 if (shared_host_mac_)
100 {
101 *mac = shared_host_mac_.value();
102 }
103 else // Cache miss: read MAC over DBus, and store in cache.
104 {
105 std::string mac_string;
106 try
107 {
108 auto networkd_call = new_networkd_call(&bus, true);
109 auto reply = bus.call(networkd_call);
110 std::variant<std::string> result;
111 reply.read(result);
112 mac_string = std::get<std::string>(result);
113 }
114 catch (const sdbusplus::exception::SdBusError& ex)
115 {
116 fmt::print(stderr, "Failed to get MACAddress: {}\n", ex.what());
117 return -1;
118 }
119
120 if (parse_mac(mac_string, mac) < 0)
121 {
122 fmt::print(stderr, "Failed to parse MAC Address `{}`\n",
123 mac_string);
124 return -1;
125 }
126
127 shared_host_mac_ = *mac;
128 }
129
130 return 0;
131}
132
133int PhosphorConfig::set_mac_addr(const mac_addr_t& mac)
134{
135 auto networkd_call = new_networkd_call(&bus, false);
136 std::variant<std::string> mac_value(format_mac(mac));
137 networkd_call.append(mac_value);
138
139 try
140 {
William A. Kennington IIIa5c9d7a2022-05-11 13:55:22 -0700141 auto netdir = fmt::format("/run/systemd/network/00-bmc-{}.network.d",
142 iface_name_);
143 std::filesystem::create_directories(netdir);
144 auto netfile = fmt::format("{}/60-ncsi-mac.conf", netdir);
145 auto fd = stdplus::fd::open(
146 netfile,
147 stdplus::fd::OpenFlags(stdplus::fd::OpenAccess::WriteOnly)
148 .set(stdplus::fd::OpenFlag::Create),
149 0644);
150 auto contents = fmt::format("[Link]\nMACAddress={}\n",
151 std::get<std::string>(mac_value));
152 stdplus::fd::writeExact(fd, contents);
153 }
154 catch (const std::exception& ex)
155 {
156 fmt::print(stderr, "Failed to set MAC Addr `{}` writing file: {}\n",
157 std::get<std::string>(mac_value), ex.what());
158 return -1;
159 }
160
161 try
162 {
William A. Kennington III7d6fa422021-02-08 17:04:02 -0800163 auto reply = bus.call(networkd_call);
164 }
165 catch (const sdbusplus::exception::SdBusError& ex)
166 {
167 fmt::print(stderr, "Failed to set MAC Addr `{}`: {}\n",
168 std::get<std::string>(mac_value), ex.what());
169 return -1;
170 }
171
172 shared_host_mac_ = std::experimental::nullopt;
173 return 0;
174}
175
176int PhosphorConfig::set_nic_hostless(bool is_nic_hostless)
177{
178 // Ensure that we don't trigger the target multiple times. This is
179 // undesirable because it will cause any inactive services to re-trigger
180 // every time we run this code. Since the loop calling this executes this
181 // code every 1s, we don't want to keep re-executing services. A fresh
182 // start of the daemon will always trigger the service to ensure system
183 // consistency.
184 if (was_nic_hostless_ && is_nic_hostless == *was_nic_hostless_)
185 {
186 return 0;
187 }
188
189 static constexpr auto systemdService = "org.freedesktop.systemd1";
190 static constexpr auto systemdRoot = "/org/freedesktop/systemd1";
191 static constexpr auto systemdInterface = "org.freedesktop.systemd1.Manager";
192
193 auto method = bus.new_method_call(systemdService, systemdRoot,
194 systemdInterface, "StartUnit");
195 if (is_nic_hostless)
196 {
197 method.append(
198 stdplus::util::strCat("nic-hostless@", iface_name_, ".target"));
199 }
200 else
201 {
202 method.append(
203 stdplus::util::strCat("nic-hostful@", iface_name_, ".target"));
204 }
205
206 // Specify --job-mode (see systemctl(1) for detail).
207 method.append("replace");
208
209 try
210 {
211 bus.call_noreply(method);
212 was_nic_hostless_ = is_nic_hostless;
213 return 0;
214 }
215 catch (const sdbusplus::exception::SdBusError& ex)
216 {
217 fmt::print(stderr, "Failed to set systemd nic status: {}\n", ex.what());
218 return 1;
219 }
220}
221
222} // namespace net