blob: 403d766100a9a98d307c5347f64e81e1f90defec [file] [log] [blame]
Nan Zhou14fe6692021-06-08 16:35:44 -07001// 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 "nemora.hpp"
16
17#include "default_addresses.h"
18
19#include <netinet/in.h>
20#include <time.h>
21#include <unistd.h>
22
23#include <phosphor-logging/log.hpp>
24#include <sdbusplus/bus.hpp>
25#include <sdbusplus/exception.hpp>
26
27#include <cstdint>
28#include <cstring>
Nan Zhou14fe6692021-06-08 16:35:44 -070029#include <variant>
30
31using phosphor::logging::level;
32using phosphor::logging::log;
33using sdbusplus::exception::SdBusError;
34
35constexpr auto MAC_FORMAT = "%02hhx:%02hhx:%02hhx:%02hhx:%02hhx:%02hhx";
36constexpr auto MAC_INTERFACE = "xyz.openbmc_project.Network.MACAddress";
37constexpr auto NETWORK_INTERFACE = "xyz.openbmc_project.Network";
38constexpr auto PROP_INTERFACE = "org.freedesktop.DBus.Properties";
39constexpr auto IFACE_ROOT = "/xyz/openbmc_project/network/";
40
41bool Nemora::ParseMac(const std::string& mac_addr, MacAddr* mac)
42{
Patrick Williams2be45232023-05-10 07:51:22 -050043 int ret = sscanf(mac_addr.c_str(), MAC_FORMAT, mac->octet, mac->octet + 1,
44 mac->octet + 2, mac->octet + 3, mac->octet + 4,
45 mac->octet + 5);
Nan Zhou14fe6692021-06-08 16:35:44 -070046 return (ret == MAC_ADDR_SIZE);
47}
48
49bool Nemora::GetMacAddr(MacAddr* mac, const std::string& iface_path)
50{
51 if (mac == nullptr)
52 {
53 log<level::ERR>("Nemora::GetMacAddr MAC Address is nullptr");
54 return false;
55 }
56 auto dbus = sdbusplus::bus::new_default();
Patrick Williams59ac2c22022-07-22 19:26:57 -050057 sdbusplus::message_t reply;
Nan Zhou14fe6692021-06-08 16:35:44 -070058
59 try
60 {
61 auto networkd_call = dbus.new_method_call(
62 NETWORK_INTERFACE, iface_path.c_str(), PROP_INTERFACE, "Get");
63 networkd_call.append(MAC_INTERFACE, "MACAddress");
64
65 reply = dbus.call(networkd_call);
66 }
67 catch (const SdBusError& e)
68 {
69 log<level::ERR>(
70 "Nemora::GetMacAddr failed to call Network D-Bus interface");
71 return false;
72 }
73
74 std::variant<std::string> result;
75 reply.read(result);
76 auto mac_addr = std::get<std::string>(result);
77 if (!ParseMac(mac_addr, mac))
78 {
79 log<level::ERR>("Nemora::GetMacAddr Failed to parse MAC Address");
80 return false;
81 }
82 return true;
83}
84
85void Nemora::InitEventData()
86{
87 event_data_.type = NemoraDatagramType::NemoraEvent;
88
89 // UDP IPv4 addr for POST
90 event_data_.destination.sin_family = AF_INET;
91 event_data_.destination.sin_port = htons(DEFAULT_ADDRESSES_TARGET_PORT);
92
93 // UDP IPv6 addr for POST
94 event_data_.destination6.sin6_family = AF_INET6;
95 event_data_.destination6.sin6_port = htons(DEFAULT_ADDRESSES_TARGET_PORT);
96}
97
98Nemora::Nemora()
99{
100 InitEventData();
101}
102
103Nemora::Nemora(const std::string& iface_name, const in_addr ipv4,
104 const in6_addr ipv6) :
105 socketManager_(),
106 hostManager_(), iface_path_{std::string(IFACE_ROOT) + iface_name}
107{
108 InitEventData();
109 event_data_.destination.sin_addr = ipv4;
110 event_data_.destination6.sin6_addr = ipv6;
111}
112
113void Nemora::UdpPoll()
114{
115 auto postcodes = hostManager_.DrainPostcodes();
116
117 // Don't bother updating if there is no POST code
118 // EC supports a flag EC_NEMORA_UDP_CONFIG_MASK_PERIODIC to send
119 // periodic updates, which is non-POR for gBMC for now.
120 bool shouldBroadcast = !postcodes.empty();
121
122 UpdateEventData(std::move(postcodes));
123
124 log<level::INFO>("UpdateEventData gets called.");
125
126 if (shouldBroadcast)
127 {
128 log<level::INFO>("Should broadcast");
129 std::lock_guard<std::mutex> lock(event_data_mutex_);
130 socketManager_.SendDatagram(static_cast<NemoraDatagram*>(&event_data_));
131 }
132
133 sleep(20);
134}
135
136void Nemora::UpdateEventData(std::vector<uint64_t>&& postcodes)
137{
138 MacAddr mac;
139 GetMacAddr(&mac, iface_path_);
140
141 std::lock_guard<std::mutex> lock(event_data_mutex_);
142
143 memcpy(event_data_.mac, mac.octet, sizeof(MacAddr));
144
145 event_data_.postcodes = std::move(postcodes);
146 event_data_.sent_time_s = time(0);
147}