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