| // Copyright 2021 Google LLC |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "ncsi_state_machine.h" |
| |
| #include "common_defs.h" |
| #include "platforms/nemora/portable/default_addresses.h" |
| #include "platforms/nemora/portable/ncsi_fsm.h" |
| |
| #include <arpa/inet.h> |
| #include <netinet/ether.h> |
| #include <unistd.h> |
| |
| #include <stdplus/print.hpp> |
| |
| #include <chrono> |
| #include <cstdint> |
| #include <cstdlib> |
| #include <cstring> |
| #include <string> |
| #include <thread> |
| #include <utility> |
| |
| #define ETHER_NCSI 0x88f8 |
| |
| // Only log messages a single time and drop all duplicates to prevent log spam. |
| // Having duplicate messages printed has historically not been helpful in |
| // debugging issues with this program. |
| static void do_log(std::string&& line) |
| { |
| constexpr auto line_dup_time = std::chrono::hours(1); |
| static std::chrono::steady_clock::time_point last_line_time; |
| static size_t line_rep_count = 0; |
| static std::string last_line; |
| |
| auto now = std::chrono::steady_clock::now(); |
| if (line != last_line || line_dup_time + last_line_time < now) |
| { |
| if (line_rep_count > 0) |
| { |
| stdplus::println(stderr, "... Repeated {} times ...", |
| line_rep_count); |
| } |
| stdplus::print(stderr, "{}", line); |
| last_line = std::move(line); |
| last_line_time = now; |
| line_rep_count = 0; |
| } |
| else |
| { |
| line_rep_count++; |
| } |
| } |
| |
| #define CPRINT(...) do_log(std::format(__VA_ARGS__)) |
| |
| #ifdef NCSID_VERBOSE_LOGGING |
| #define DEBUG_PRINTF printf |
| #else |
| #define DEBUG_PRINTF(...) |
| #endif |
| |
| namespace ncsi |
| { |
| |
| namespace |
| { |
| |
| const char kStateFormat[] = "l2_config=%d/%d l3l4_config=%d/%d test=%d/%d"; |
| // This assumes that the number of states is < 100, so each number |
| // in the format above does not take more than two characters to represent, |
| // thus %d (two characters) is substituted for the number which is also |
| // two characters max. |
| constexpr auto kStateFormatLen = sizeof(kStateFormat); |
| |
| void snprintf_state(char* buffer, int size, const ncsi_state_t* state) |
| { |
| (void)snprintf(buffer, size, kStateFormat, state->l2_config_state, |
| NCSI_STATE_L2_CONFIG_END, state->l3l4_config_state, |
| NCSI_STATE_L3L4_CONFIG_END, state->test_state, |
| NCSI_STATE_TEST_END); |
| } |
| |
| void print_state(const ncsi_state_t& state) |
| { |
| (void)state; |
| DEBUG_PRINTF(kStateFormat, state.l2_config_state, NCSI_STATE_L2_CONFIG_END, |
| state.l3l4_config_state, NCSI_STATE_L3L4_CONFIG_END, |
| state.test_state, NCSI_STATE_TEST_END); |
| DEBUG_PRINTF(" restart_delay_count=%d\n", state.restart_delay_count); |
| } |
| |
| const uint8_t echo_pattern[NCSI_OEM_ECHO_PATTERN_SIZE] = { |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, |
| 0xFF, 0xFF, 0xFF, 0xFF, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, |
| 0xA5, 0xA5, 0xA5, 0xA5, 0xA5, 0xA5, 0xA5, 0xA5, 0x12, 0x34, 0x56, 0x78, |
| 0x9A, 0xBC, 0xDE, 0xF0, 0xFE, 0xDC, 0xBA, 0x98, 0x76, 0x54, 0x32, 0x10}; |
| |
| } // namespace |
| |
| void StateMachine::reset() |
| { |
| std::memset(&ncsi_state_, 0, sizeof(ncsi_state_)); |
| ncsi_state_.restart_delay_count = NCSI_FSM_RESTART_DELAY_COUNT - 1; |
| network_debug_.ncsi.test.max_tries = MAX_TRIES; |
| // This needs to be initialized in the firmware. |
| network_debug_.ncsi.test.ch_under_test = 0; |
| network_debug_.ncsi.oem_filter_disable = false; |
| |
| network_debug_.ncsi.pending_stop = false; |
| network_debug_.ncsi.enabled = true; |
| network_debug_.ncsi.loopback = false; |
| } |
| |
| StateMachine::StateMachine() |
| { |
| reset(); |
| network_debug_.ncsi.pending_restart = true; |
| std::memcpy(network_debug_.ncsi.test.ping.tx, echo_pattern, |
| sizeof(echo_pattern)); |
| } |
| |
| StateMachine::~StateMachine() |
| { |
| CPRINT("[NCSI stopping]\n"); |
| } |
| |
| size_t StateMachine::poll_l2_config() |
| { |
| size_t len = 0; |
| mac_addr_t mac; |
| net_config_->get_mac_addr(&mac); |
| ncsi_response_type_t response_type = ncsi_fsm_poll_l2_config( |
| &ncsi_state_, &network_debug_, &ncsi_buf_, &mac); |
| |
| auto* response = reinterpret_cast<ncsi_simple_response_t*>(ncsi_buf_.data); |
| |
| if (response_type == NCSI_RESPONSE_ACK) |
| { |
| /* If the response is MAC response, some extra handling needed. */ |
| if ((NCSI_RESPONSE | NCSI_OEM_COMMAND) == |
| response->hdr.control_packet_type) |
| { |
| auto* oem_response = |
| reinterpret_cast<ncsi_oem_simple_response_t*>(ncsi_buf_.data); |
| if (oem_response->oem_header.oem_cmd == |
| NCSI_OEM_COMMAND_GET_HOST_MAC) |
| { |
| net_config_->set_mac_addr(mac); |
| } |
| } |
| } |
| else if (NCSI_RESPONSE_NONE == response_type) |
| { |
| /* Buffer is ready to be sent. */ |
| len = ncsi_buf_.len; |
| ncsi_buf_.len = 0; |
| } |
| else |
| { |
| report_ncsi_error(response_type); |
| } |
| |
| return len; |
| } |
| |
| size_t StateMachine::poll_simple(ncsi_simple_poll_f poll_func) |
| { |
| mac_addr_t mac; |
| net_config_->get_mac_addr(&mac); |
| const uint16_t rx_port = DEFAULT_ADDRESSES_RX_PORT; |
| |
| ncsi_response_type_t response_type = |
| poll_func(&ncsi_state_, &network_debug_, &ncsi_buf_, &mac, 0, rx_port); |
| |
| auto* response = reinterpret_cast<ncsi_simple_response_t*>(ncsi_buf_.data); |
| |
| size_t len = 0; |
| if (response_type == NCSI_RESPONSE_NONE) |
| { |
| /* Buffer is ready to be sent, or we are done. */ |
| len = ncsi_buf_.len; |
| ncsi_buf_.len = 0; |
| } |
| else if (response->hdr.control_packet_type == |
| (NCSI_RESPONSE | NCSI_GET_LINK_STATUS)) |
| { |
| auto status_response = |
| reinterpret_cast<ncsi_link_status_response_t*>(response); |
| bool new_link_up = ntohl(status_response->link_status.link_status) & |
| NCSI_LINK_STATUS_UP; |
| if (!link_up_ || new_link_up != *link_up_) |
| { |
| CPRINT("[NCSI link {}]\n", new_link_up ? "up" : "down"); |
| link_up_ = new_link_up; |
| } |
| } |
| else if (response->hdr.control_packet_type == |
| (NCSI_RESPONSE | NCSI_OEM_COMMAND)) |
| { |
| auto* oem_response = |
| reinterpret_cast<ncsi_oem_simple_response_t*>(ncsi_buf_.data); |
| if (oem_response->oem_header.oem_cmd == NCSI_OEM_COMMAND_GET_FILTER) |
| { |
| bool new_hostless = ncsi_fsm_is_nic_hostless(&ncsi_state_); |
| if (!hostless_ || new_hostless != *hostless_) |
| { |
| CPRINT("[NCSI nic {}]\n", |
| new_hostless ? "hostless" : "hostfull"); |
| net_config_->set_nic_hostless(new_hostless); |
| hostless_ = new_hostless; |
| } |
| } |
| } |
| else if (response_type != NCSI_RESPONSE_ACK) |
| { |
| report_ncsi_error(response_type); |
| } |
| |
| return len; |
| } |
| |
| void StateMachine::report_ncsi_error(ncsi_response_type_t response_type) |
| { |
| char state_string[kStateFormatLen]; |
| snprintf_state(state_string, sizeof(state_string), &ncsi_state_); |
| auto* response = |
| reinterpret_cast<const ncsi_simple_response_t*>(ncsi_buf_.data); |
| switch (response_type) |
| { |
| case NCSI_RESPONSE_UNDERSIZED: |
| if (!ncsi_buf_.len) |
| { |
| network_debug_.ncsi.rx_error.timeout_count++; |
| CPRINT("[NCSI timeout in state {}]\n", state_string); |
| } |
| else |
| { |
| network_debug_.ncsi.rx_error.undersized_count++; |
| CPRINT("[NCSI undersized response in state {}]\n", |
| state_string); |
| } |
| break; |
| case NCSI_RESPONSE_NACK: |
| network_debug_.ncsi.rx_error.nack_count++; |
| CPRINT( |
| "[NCSI nack in state {}. Response: {:#04x} Reason: {:#04x}]\n", |
| state_string, ntohs(response->response_code), |
| ntohs(response->reason_code)); |
| break; |
| case NCSI_RESPONSE_UNEXPECTED_TYPE: |
| network_debug_.ncsi.rx_error.unexpected_type_count++; |
| CPRINT( |
| "[NCSI unexpected response in state {}. Response type: {:#02x}]\n", |
| state_string, response->hdr.control_packet_type); |
| break; |
| case NCSI_RESPONSE_UNEXPECTED_SIZE: |
| { |
| int expected_size; |
| if ((NCSI_RESPONSE | NCSI_OEM_COMMAND) == |
| response->hdr.control_packet_type) |
| { |
| auto* oem_response = |
| reinterpret_cast<ncsi_oem_simple_response_t*>( |
| ncsi_buf_.data); |
| expected_size = ncsi_oem_get_response_size( |
| oem_response->oem_header.oem_cmd); |
| } |
| else |
| { |
| expected_size = ncsi_get_response_size( |
| response->hdr.control_packet_type & (~NCSI_RESPONSE)); |
| } |
| network_debug_.ncsi.rx_error.unexpected_size_count++; |
| CPRINT("[NCSI unexpected response size in state {}." |
| " Expected {}]\n", |
| state_string, expected_size); |
| } |
| break; |
| case NCSI_RESPONSE_OEM_FORMAT_ERROR: |
| network_debug_.ncsi.rx_error.unexpected_type_count++; |
| CPRINT("[NCSI OEM format error]\n"); |
| break; |
| case NCSI_RESPONSE_UNEXPECTED_PARAMS: |
| CPRINT("[NCSI OEM Filter MAC or TCP/IP Config Mismatch]\n"); |
| break; |
| default: |
| /* NCSI_RESPONSE_ACK and NCSI_RESPONSE_NONE are not errors and need |
| * not be handled here, so this branch is just to complete the |
| * switch. |
| */ |
| CPRINT("[NCSI OK]\n"); |
| break; |
| } |
| } |
| |
| int StateMachine::receive_ncsi() |
| { |
| int len; |
| do |
| { |
| // Return value of zero means timeout |
| len = sock_io_->recv(ncsi_buf_.data, sizeof(ncsi_buf_.data)); |
| if (len > 0) |
| { |
| auto* hdr = reinterpret_cast<struct ether_header*>(ncsi_buf_.data); |
| if (ETHER_NCSI == ntohs(hdr->ether_type)) |
| { |
| ncsi_buf_.len = len; |
| break; |
| } |
| } |
| |
| ncsi_buf_.len = 0; |
| } while (len > 0); |
| |
| return ncsi_buf_.len; |
| } |
| |
| void StateMachine::run_test_fsm(size_t* tx_len) |
| { |
| // Sleep and restart when test FSM finishes. |
| if (is_test_done()) |
| { |
| std::this_thread::sleep_for(std::chrono::seconds(retest_delay_s_)); |
| // Skip over busy wait in state machine - already waited. |
| ncsi_state_.retest_delay_count = NCSI_FSM_RESTART_DELAY_COUNT; |
| } |
| // until NCSI_STATE_TEST_END |
| *tx_len = poll_simple(ncsi_fsm_poll_test); |
| } |
| |
| void StateMachine::run(int max_rounds) |
| { |
| if (!net_config_ || !sock_io_) |
| { |
| CPRINT("StateMachine configuration incomplete: " |
| "net_config_: <{}>, sock_io_: <{}>", |
| reinterpret_cast<void*>(net_config_), |
| reinterpret_cast<void*>(sock_io_)); |
| return; |
| } |
| |
| bool infinite_loop = (max_rounds <= 0); |
| while (infinite_loop || --max_rounds >= 0) |
| { |
| receive_ncsi(); |
| |
| size_t tx_len = 0; |
| switch (ncsi_fsm_connection_state(&ncsi_state_, &network_debug_)) |
| { |
| case NCSI_CONNECTION_DOWN: |
| case NCSI_CONNECTION_LOOPBACK: |
| tx_len = poll_l2_config(); |
| break; |
| case NCSI_CONNECTION_UP: |
| if (!is_test_done() || ncsi_fsm_is_nic_hostless(&ncsi_state_)) |
| { |
| run_test_fsm(&tx_len); |
| } |
| else |
| { |
| // - Only start L3/L4 config when test is finished |
| // (it will last until success |
| // (i.e. NCSI_CONNECTION_UP_AND_CONFIGURED) or fail. |
| tx_len = poll_simple(ncsi_fsm_poll_l3l4_config); |
| } |
| break; |
| case NCSI_CONNECTION_UP_AND_CONFIGURED: |
| run_test_fsm(&tx_len); |
| break; |
| case NCSI_CONNECTION_DISABLED: |
| if (network_debug_.ncsi.pending_restart) |
| { |
| network_debug_.ncsi.enabled = true; |
| } |
| break; |
| default: |
| fail(); |
| } |
| |
| if (tx_len) |
| { |
| print_state(ncsi_state_); |
| |
| sock_io_->write(ncsi_buf_.data, tx_len); |
| } |
| } |
| } |
| |
| void StateMachine::clear_state() |
| { |
| // This implicitly resets: |
| // l2_config_state to NCSI_STATE_L2_CONFIG_BEGIN |
| // l3l4_config_state to NCSI_STATE_L3L4_CONFIG_BEGIN |
| // test_state to NCSI_STATE_TEST_BEGIN |
| std::memset(&ncsi_state_, 0, sizeof(ncsi_state_)); |
| } |
| |
| void StateMachine::fail() |
| { |
| network_debug_.ncsi.fail_count++; |
| clear_state(); |
| } |
| |
| void StateMachine::set_sockio(net::SockIO* sock_io) |
| { |
| sock_io_ = sock_io; |
| } |
| |
| void StateMachine::set_net_config(net::ConfigBase* net_config) |
| { |
| net_config_ = net_config; |
| } |
| |
| void StateMachine::set_retest_delay(unsigned delay) |
| { |
| retest_delay_s_ = delay; |
| } |
| |
| } // namespace ncsi |