nvmesensor: Remove MCTP transport implementation
This was implemented in terms of a series of patches against libmctp
that were never upstreamed (because they in-turn relied on kernel
patches that also were never upstreamed). Further, there are no
configurations enabling the nvme-mi-mctp option in openbmc/openbmc.
This is motivated by the need to refactor NVMeContext to support polling
all sensors inside the polling period on a given root bus.
Signed-off-by: Andrew Jeffery <andrew@aj.id.au>
Change-Id: Ib20b7f3ce218191a391d7b2d893f204fb97b7ab5
diff --git a/include/NVMeDevice.hpp b/include/NVMeDevice.hpp
deleted file mode 100644
index 5ebb0b3..0000000
--- a/include/NVMeDevice.hpp
+++ /dev/null
@@ -1,99 +0,0 @@
-#pragma once
-
-#include <stdint.h>
-#include <stdlib.h>
-
-// NVM Express Management Interface 1.0 section 3.2.1
-const uint8_t NVME_MI_MESSAGE_TYPE = 0x04;
-
-const uint8_t NVME_MI_MESSAGE_TYPE_MASK = 0x7F;
-
-// Indicates this is covered by an MCTP integrity check
-const uint8_t NVME_MI_MCTP_INTEGRITY_CHECK = (1 << 7);
-
-// Indicates whether this is a request or response
-const uint8_t NVME_MI_HDR_FLAG_ROR = (1 << 7);
-
-const uint8_t NVME_MI_HDR_FLAG_MSG_TYPE_MASK = 0x0F;
-const uint8_t NVME_MI_HDR_FLAG_MSG_TYPE_SHIFT = 3;
-
-const uint16_t NVME_MI_MSG_BUFFER_SIZE = 256;
-
-// Minimum length of health status poll response
-// NMH + Status + NVMe-MI Command Response Message (NCRESP)
-const uint8_t NVME_MI_HEALTH_STATUS_POLL_MSG_MIN = 8;
-
-enum NVME_MI_HDR_MESSAGE_TYPE
-{
- NVME_MI_HDR_MESSAGE_TYPE_CONTROL_PRIMITIVE = 0x00,
- NVME_MI_HDR_MESSAGE_TYPE_MI_COMMAND = 0x01,
- NVME_MI_HDR_MESSAGE_TYPE_MI_ADMIN_COMMAND = 0x02,
- NVME_MI_HDR_MESSAGE_TYPE_PCIE_COMMAND = 0x04,
-};
-
-enum NVME_MI_HDR_COMMAND_SLOT
-{
- NVME_MI_HDR_COMMAND_SLOT_0 = 0x00,
- NVME_MI_HDR_COMMAND_SLOT_1 = 0x01,
-};
-
-enum NVME_MI_HDR_STATUS
-{
- NVME_MI_HDR_STATUS_SUCCESS = 0x00,
- NVME_MI_HDR_STATUS_MORE_PROCESSING_REQUIRED = 0x01,
- NVME_MI_HDR_STATUS_INTERNAL_ERROR = 0x02,
- NVME_MI_HDR_STATUS_INVALID_COMMAND_OPCODE = 0x03,
- NVME_MI_HDR_STATUS_INVALID_PARAMETER = 0x04,
- NVME_MI_HDR_STATUS_INVALID_COMMAND_SIZE = 0x05,
- NVME_MI_HDR_STATUS_INVALID_COMMAND_INPUT_DATA_SIZE = 0x06,
- NVME_MI_HDR_STATUS_ACCESS_DENIED = 0x07,
- NVME_MI_HDR_STATUS_VPD_UPDATES_EXCEEDED = 0x20,
- NVME_MI_HDR_STATUS_PCIE_INACCESSIBLE = 0x21,
-};
-
-enum NVME_MI_OPCODE
-{
- NVME_MI_OPCODE_READ_MI_DATA = 0x00,
- NVME_MI_OPCODE_HEALTH_STATUS_POLL = 0x01,
- NVME_MI_OPCODE_CONTROLLER_HEALTH_STATUS_POLL = 0x02,
- NVME_MI_OPCODE_CONFIGURATION_GET = 0x03,
- NVME_MI_OPCODE_CONFIGURATION_SET = 0x04,
- NVME_MI_OPCODE_VPD_READ = 0x05,
- NVME_MI_OPCODE_VPD_WRITE = 0x06,
- NVME_MI_OPCODE_RESET = 0x07,
-};
-
-const uint8_t NVME_MI_MSG_REQUEST_HEADER_SIZE = 16;
-struct nvme_mi_msg_request_header
-{
- uint8_t message_type;
- uint8_t flags;
- uint8_t opcode;
- uint32_t dword0;
- uint32_t dword1;
-};
-
-struct nvme_mi_msg_request
-{
- struct nvme_mi_msg_request_header header;
- uint8_t request_data[128];
- size_t request_data_len;
-};
-
-const uint8_t NVME_MI_MSG_RESPONSE_HEADER_SIZE = 5;
-struct nvme_mi_msg_response_header
-{
- uint8_t message_type;
- uint8_t flags;
- // Reserved bytes 2:3
- uint8_t status;
-};
-
-struct nvme_mi_controller_health
-{
- uint8_t nvm_subsystem_status;
- uint8_t smart_warnings;
- uint8_t composite_temperature;
- uint8_t percent_used;
- uint16_t composite_controller_status;
-};
\ No newline at end of file
diff --git a/include/NVMeMCTPContext.hpp b/include/NVMeMCTPContext.hpp
deleted file mode 100644
index 8af3e8d..0000000
--- a/include/NVMeMCTPContext.hpp
+++ /dev/null
@@ -1,29 +0,0 @@
-#pragma once
-
-#include "NVMeContext.hpp"
-
-#include <boost/asio/ip/tcp.hpp>
-
-class NVMeMCTPContext : public NVMeContext
-{
- public:
- NVMeMCTPContext(boost::asio::io_service& io, int rootBus);
-
- ~NVMeMCTPContext() override;
-
- void pollNVMeDevices() override;
- void close() override;
- void readAndProcessNVMeSensor() override;
- void processResponse(void* msg, size_t len) override;
-
- private:
- boost::asio::ip::tcp::socket nvmeSlaveSocket;
- boost::asio::deadline_timer mctpResponseTimer;
-
- void readResponse();
-};
-
-namespace nvme_mctp
-{
-void init(void);
-}
diff --git a/meson_options.txt b/meson_options.txt
index ea0302d..f9d25e7 100644
--- a/meson_options.txt
+++ b/meson_options.txt
@@ -7,7 +7,6 @@
option('ipmb', type: 'feature', value: 'enabled', description: 'Enable IPMB sensor.',)
option('mcu', type: 'feature', value: 'enabled', description: 'Enable MCU sensor.',)
option('nvme', type: 'feature', value: 'enabled', description: 'Enable NVMe sensor.',)
-option('nvme-mi-mctp', type: 'feature', value: 'disabled', description: 'Manage NVMe devices via MCTP')
option('psu', type: 'feature', value: 'enabled', description: 'Enable PSU sensor.',)
option('external', type: 'feature', value: 'enabled', description: 'Enable External sensor.',)
option('tests', type: 'feature', value: 'enabled', description: 'Build tests.',)
diff --git a/src/NVMeMCTPContext.cpp b/src/NVMeMCTPContext.cpp
deleted file mode 100644
index 40bb8ac..0000000
--- a/src/NVMeMCTPContext.cpp
+++ /dev/null
@@ -1,432 +0,0 @@
-/*
-// Copyright (c) 2019 Intel Corporation
-//
-// 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 "NVMeMCTPContext.hpp"
-
-#include "NVMeDevice.hpp"
-
-#include <crc32c.h>
-#include <libmctp-smbus.h>
-#include <libmctp.h>
-
-#include <boost/container/flat_map.hpp>
-
-static constexpr bool debug = false;
-
-static void rxMessage(uint8_t eid, void* data, void* msg, size_t len);
-
-namespace nvmeMCTP
-{
-struct mctp_binding_smbus* smbus = mctp_smbus_init();
-struct mctp* mctp = mctp_init();
-
-static boost::container::flat_map<int, int> inFds;
-static boost::container::flat_map<int, int> outFds;
-
-int getInFd(int rootBus)
-{
- auto findBus = inFds.find(rootBus);
- if (findBus != inFds.end())
- {
- return findBus->second;
- }
- int fd = mctp_smbus_open_in_bus(smbus, rootBus);
- if (fd < 0)
- {
- std::cerr << "Error opening IN Bus " << rootBus << "\n";
- }
- inFds[rootBus] = fd;
- return fd;
-}
-
-int getOutFd(int bus)
-{
- auto findBus = outFds.find(bus);
- if (findBus != outFds.end())
- {
- return findBus->second;
- }
- int fd = mctp_smbus_open_out_bus(smbus, bus);
- if (fd < 0)
- {
- std::cerr << "Error opening Out Bus " << bus << "\n";
- }
- outFds[bus] = fd;
- return fd;
-}
-
-// we don't close the outFd as multiple sensors could be sharing the fd, we need
-// to close the inFd as it can only be used on 1 socket at a time
-void closeInFd(int rootBus)
-{
- auto findFd = inFds.find(rootBus);
- if (findFd == inFds.end())
- {
- return;
- }
- close(findFd->second);
- inFds.erase(rootBus);
-}
-
-int getRootBus(int inFd)
-{
- // we assume that we won't have too many FDs, so looping is OK
- for (const auto [root, fd] : inFds)
- {
- if (fd == inFd)
- {
- return root;
- }
- }
-
- return -1;
-}
-
-void init()
-{
- if (mctp == nullptr || smbus == nullptr)
- {
- throw std::runtime_error("Unable to init mctp");
- }
- mctp_smbus_register_bus(smbus, nvmeMCTP::mctp, 0);
- mctp_set_rx_all(mctp, rxMessage, nullptr);
-}
-
-} // namespace nvmeMCTP
-
-static void rxMessage(uint8_t eid, void*, void* msg, size_t len)
-{
- int inFd = mctp_smbus_get_in_fd(nvmeMCTP::smbus);
- int rootBus = nvmeMCTP::getRootBus(inFd);
-
- NVMEMap& nvmeMap = getNVMEMap();
- auto findMap = nvmeMap.find(rootBus);
- if (findMap == nvmeMap.end())
- {
- std::cerr << "Unable to lookup root bus " << rootBus << "\n";
- return;
- }
-
- if (debug)
- {
- std::cout << "Eid from the received messaged: " << eid << "\n";
- }
-
- findMap->second->processResponse(msg, len);
-}
-
-static int verifyIntegrity(uint8_t* msg, size_t len)
-{
- uint32_t msgIntegrity = {0};
- if (len < NVME_MI_MSG_RESPONSE_HEADER_SIZE + sizeof(msgIntegrity))
- {
- std::cerr << "Not enough bytes for nvme header and trailer\n";
- return -1;
- }
-
- msgIntegrity = (msg[len - 4]) + (msg[len - 3] << 8) + (msg[len - 2] << 16) +
- (msg[len - 1] << 24);
-
- uint32_t calculateIntegrity = crc32c(msg, len - sizeof(msgIntegrity));
- if (msgIntegrity != calculateIntegrity)
- {
- std::cerr << "CRC mismatch. Got=" << msgIntegrity
- << " Expected=" << calculateIntegrity << "\n";
- return -1;
- }
- return 0;
-}
-
-static double getTemperatureReading(int8_t reading)
-{
-
- if (reading == static_cast<int8_t>(0x80) ||
- reading == static_cast<int8_t>(0x81))
- {
- // 0x80 = No temperature data or temperature data is more the 5 s
- // old 0x81 = Temperature sensor failure
- return std::numeric_limits<double>::quiet_NaN();
- }
-
- return reading;
-}
-
-void NVMeMCTPContext::processResponse(void* msg, size_t len)
-{
- struct nvme_mi_msg_response_header header
- {};
-
- if (msg == nullptr)
- {
- std::cerr << "Bad message received\n";
- return;
- }
-
- if (len <= 0)
- {
- std::cerr << "Received message not long enough\n";
- return;
- }
-
- uint8_t* messageData = static_cast<uint8_t*>(msg);
-
- if ((*messageData & NVME_MI_MESSAGE_TYPE_MASK) != NVME_MI_MESSAGE_TYPE)
- {
- std::cerr << "Got unknown type message_type="
- << (*messageData & NVME_MI_MESSAGE_TYPE_MASK) << "\n";
- return;
- }
-
- if (len < NVME_MI_MSG_RESPONSE_HEADER_SIZE + sizeof(uint32_t))
- {
- std::cerr << "Not enough bytes for NVMe header and trailer\n";
- return;
- }
-
- if (verifyIntegrity(messageData, len) != 0)
- {
- std::cerr << "Verification of message integrity failed\n";
- return;
- }
-
- header.message_type = messageData[0];
- header.flags = messageData[1];
- header.status = messageData[4];
-
- if (header.status == NVME_MI_HDR_STATUS_MORE_PROCESSING_REQUIRED)
- {
- return;
- }
-
- if (header.status != NVME_MI_HDR_STATUS_SUCCESS)
- {
- std::cerr << "Command failed with status= " << header.status << "\n";
- return;
- }
-
- messageData += NVME_MI_MSG_RESPONSE_HEADER_SIZE;
- size_t messageLength =
- len - NVME_MI_MSG_RESPONSE_HEADER_SIZE - sizeof(uint32_t);
- if (((header.flags >> NVME_MI_HDR_FLAG_MSG_TYPE_SHIFT) &
- NVME_MI_HDR_FLAG_MSG_TYPE_MASK) != NVME_MI_HDR_MESSAGE_TYPE_MI_COMMAND)
- {
- std::cerr << "Not MI type comamnd\n";
- return;
- }
-
- if (messageLength < NVME_MI_HEALTH_STATUS_POLL_MSG_MIN)
- {
- std::cerr << "Got improperly sized health status poll\n";
- return;
- }
-
- std::shared_ptr<NVMeSensor> sensorInfo = sensors.front();
- if (debug)
- {
- std::cout << "Temperature Reading: "
- << getTemperatureReading(messageData[5])
- << " Celsius for device " << sensorInfo->name << "\n";
- }
-
- double value = getTemperatureReading(messageData[5]);
- if (!std::isfinite(value))
- {
- sensorInfo->markAvailable(false);
- sensorInfo->incrementError();
- }
- else
- {
- sensorInfo->updateValue(value);
- }
-
- if (debug)
- {
- std::cout << "Cancelling the timer now\n";
- }
-
- // move to back of scan queue
- sensors.pop_front();
- sensors.emplace_back(sensorInfo);
-
- mctpResponseTimer.cancel();
-}
-
-static int nvmeMessageTransmit(mctp& mctp, nvme_mi_msg_request& req)
-{
- std::array<uint8_t, NVME_MI_MSG_BUFFER_SIZE> messageBuf = {};
-
- req.header.flags |= NVME_MI_HDR_MESSAGE_TYPE_MI_COMMAND
- << NVME_MI_HDR_FLAG_MSG_TYPE_SHIFT;
- req.header.message_type =
- NVME_MI_MESSAGE_TYPE | NVME_MI_MCTP_INTEGRITY_CHECK;
-
- uint32_t integrity = 0;
- size_t msgSize = NVME_MI_MSG_REQUEST_HEADER_SIZE + req.request_data_len +
- sizeof(integrity);
-
- if (sizeof(messageBuf) < msgSize)
- {
- return EXIT_FAILURE;
- }
-
- messageBuf[0] = req.header.message_type;
- messageBuf[1] = req.header.flags;
- // Reserved bytes 2-3
-
- messageBuf[4] = req.header.opcode;
- // reserved bytes 5-7
- messageBuf[8] = req.header.dword0 & 0xff;
- messageBuf[9] = (req.header.dword0 >> 8) & 0xff;
- messageBuf[10] = (req.header.dword0 >> 16) & 0xff;
- messageBuf[11] = (req.header.dword0 >> 24) & 0xff;
-
- messageBuf[12] = req.header.dword1 & 0xff;
- messageBuf[13] = (req.header.dword1 >> 8) & 0xff;
- messageBuf[14] = (req.header.dword1 >> 16) & 0xff;
- messageBuf[15] = (req.header.dword1 >> 24) & 0xff;
-
- std::copy_n(req.request_data, req.request_data_len,
- messageBuf.data() +
- static_cast<uint8_t>(NVME_MI_MSG_REQUEST_HEADER_SIZE));
-
- msgSize = NVME_MI_MSG_REQUEST_HEADER_SIZE + req.request_data_len;
- integrity = crc32c(messageBuf.data(),
- NVME_MI_MSG_REQUEST_HEADER_SIZE + req.request_data_len);
- messageBuf[msgSize] = integrity & 0xff;
- messageBuf[msgSize + 1] = (integrity >> 8) & 0xff;
- messageBuf[msgSize + 2] = (integrity >> 16) & 0xff;
- messageBuf[msgSize + 3] = (integrity >> 24) & 0xff;
- msgSize += sizeof(integrity);
-
- return mctp_message_tx(&mctp, 0, messageBuf.data(), msgSize);
-}
-
-void NVMeMCTPContext::readResponse()
-{
- nvmeSlaveSocket.async_wait(
- boost::asio::ip::tcp::socket::wait_error,
- [this](const boost::system::error_code errorCode) {
- if (errorCode)
- {
- return;
- }
-
- mctp_smbus_set_in_fd(nvmeMCTP::smbus, nvmeMCTP::getInFd(rootBus));
-
- // through libmctp this will invoke rxMessage
- mctp_smbus_read(nvmeMCTP::smbus);
- });
-}
-
-void NVMeMCTPContext::readAndProcessNVMeSensor()
-{
- struct nvme_mi_msg_request requestMsg = {};
- requestMsg.header.opcode = NVME_MI_OPCODE_HEALTH_STATUS_POLL;
- requestMsg.header.dword0 = 0;
- requestMsg.header.dword1 = 0;
-
- int mctpResponseTimeout = 1;
-
- if (sensors.empty())
- {
- return;
- }
-
- std::shared_ptr<NVMeSensor>& sensor = sensors.front();
-
- // setup the timeout timer
- mctpResponseTimer.expires_from_now(
- boost::posix_time::seconds(mctpResponseTimeout));
-
- mctpResponseTimer.async_wait(
- [sensor, this](const boost::system::error_code errorCode) {
- if (errorCode)
- {
- // timer cancelled successfully
- return;
- }
-
- sensor->incrementError();
-
- // cycle it back
- sensors.pop_front();
- sensors.emplace_back(sensor);
-
- nvmeSlaveSocket.cancel();
- });
-
- readResponse();
-
- if (debug)
- {
- std::cout << "Sending message to read data from Drive on bus: "
- << sensor->bus << " , rootBus: " << rootBus
- << " device: " << sensor->name << "\n";
- }
-
- mctp_smbus_set_out_fd(nvmeMCTP::smbus, nvmeMCTP::getOutFd(sensor->bus));
- int rc = nvmeMessageTransmit(*nvmeMCTP::mctp, requestMsg);
-
- if (rc != 0)
- {
- std::cerr << "Error sending request message to NVMe device\n";
- }
-}
-
-NVMeMCTPContext::NVMeMCTPContext(boost::asio::io_service& io, int rootBus) :
- NVMeContext::NVMeContext(io, rootBus), nvmeSlaveSocket(io),
- mctpResponseTimer(io)
-{
- nvmeSlaveSocket.assign(boost::asio::ip::tcp::v4(),
- nvmeMCTP::getInFd(rootBus));
-}
-
-void NVMeMCTPContext::pollNVMeDevices()
-{
- scanTimer.expires_from_now(boost::posix_time::seconds(1));
- scanTimer.async_wait(
- [self{shared_from_this()}](const boost::system::error_code errorCode) {
- if (errorCode == boost::asio::error::operation_aborted)
- {
- return; // we're being canceled
- }
- else if (errorCode)
- {
- std::cerr << "Error:" << errorCode.message() << "\n";
- return;
- }
- else
- {
- self->readAndProcessNVMeSensor();
- }
-
- self->pollNVMeDevices();
- });
-}
-
-void NVMeMCTPContext::close()
-{
- this->NVMeContext::close();
-
- mctpResponseTimer.cancel();
- nvmeSlaveSocket.cancel();
- nvmeMCTP::closeInFd(rootBus);
-}
-
-NVMeMCTPContext::~NVMeMCTPContext()
-{
- close();
-}
diff --git a/src/NVMeSensorMain.cpp b/src/NVMeSensorMain.cpp
index 7e32815..d5e3026 100644
--- a/src/NVMeSensorMain.cpp
+++ b/src/NVMeSensorMain.cpp
@@ -16,7 +16,6 @@
#include <NVMeBasicContext.hpp>
#include <NVMeContext.hpp>
-#include <NVMeMCTPContext.hpp>
#include <NVMeSensor.hpp>
#include <boost/asio/deadline_timer.hpp>
@@ -101,11 +100,7 @@
}
std::shared_ptr<NVMeContext> context =
-#if HAVE_NVME_MI_MCTP
- std::make_shared<NVMeMCTPContext>(io, rootBus);
-#else
std::make_shared<NVMeBasicContext>(io, rootBus);
-#endif
map[rootBus] = context;
return context;
@@ -235,9 +230,6 @@
auto systemBus = std::make_shared<sdbusplus::asio::connection>(io);
systemBus->request_name("xyz.openbmc_project.NVMeSensor");
sdbusplus::asio::object_server objectServer(systemBus);
-#if HAVE_NVME_MI_MCTP
- nvmeMCTP::init();
-#endif
io.post([&]() { createSensors(io, objectServer, systemBus); });
diff --git a/src/meson.build b/src/meson.build
index d2535e2..0fe5f56 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -142,14 +142,6 @@
nvme_deps = [ default_deps, i2c, thresholds_dep, utils_dep, threads ]
- mi_mctp = get_option('nvme-mi-mctp')
- if mi_mctp.enabled()
- nvme_srcs += files('NVMeMCTPContext.cpp')
- nvme_deps += meson.get_compiler('cpp').find_library('libmctp')
- conf_data = configuration_data()
- conf_data.set('HAVE_NVME_MI_MCTP', true)
- endif
-
executable(
'nvmesensor',
sources: nvme_srcs,