bmcweb的console如何设置成多路sol??

bmcweb里面obmc_console.hpp的源码如下所示:

#pragma once
#### include "app.hpp"
#include "async_resp.hpp"
#include "websocket.hpp"
#include <sys/socket.h>
 
#include <boost/asio/local/stream_protocol.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/system/error_code.hpp>
 
#include <array>
#include <memory>
#include <string>
#include <string_view>
 
namespace crow
{
namespace obmc_console
{
 
// Update this value each time we add new console route.
static constexpr const uint maxSessions = 32;
 
class ConsoleHandler : public std::enable_shared_from_this<ConsoleHandler>
{
  public:
    ConsoleHandler(boost::asio::io_context& ioc,
                   crow::websocket::Connection& connIn) :
        hostSocket(ioc), conn(connIn)
    {}
 
    ~ConsoleHandler() = default;
 
    ConsoleHandler(const ConsoleHandler&) = delete;
    ConsoleHandler(ConsoleHandler&&) = delete;
    ConsoleHandler& operator=(const ConsoleHandler&) = delete;
    ConsoleHandler& operator=(ConsoleHandler&&) = delete;
 
    void doWrite()
    {
        if (doingWrite)
        {
            BMCWEB_LOG_DEBUG("Already writing.  Bailing out");
            return;
        }
 
        if (inputBuffer.empty())
        {
            BMCWEB_LOG_DEBUG("Outbuffer empty.  Bailing out");
            return;
        }
 
        doingWrite = true;
        hostSocket.async_write_some(
            boost::asio::buffer(inputBuffer.data(), inputBuffer.size()),
            [weak(weak_from_this())](const boost::beast::error_code& ec,
                                     std::size_t bytesWritten) {
                std::shared_ptr<ConsoleHandler> self = weak.lock();
                if (self == nullptr)
                {
                    return;
                }
 
                self->doingWrite = false;
                self->inputBuffer.erase(0, bytesWritten);
                if (ec == boost::asio::error::eof)
                {
                    self->conn.close("Error in reading to host port");
                    return;
                }
                if (ec)
                {
                    BMCWEB_LOG_ERROR("Error in host serial write {}",
                                     ec.message());
                    return;
                }
                self->doWrite();
            });
    }
 
    static void afterSendEx(const std::weak_ptr<ConsoleHandler>& weak)
    {
        std::shared_ptr<ConsoleHandler> self = weak.lock();
        if (self == nullptr)
        {
            return;
        }
        self->doRead();
    }
 
    void doRead()
    {
        BMCWEB_LOG_DEBUG("Reading from socket");
        hostSocket.async_read_some(
            boost::asio::buffer(outputBuffer),
            [this, weakSelf(weak_from_this())](
                const boost::system::error_code& ec, std::size_t bytesRead) {
                BMCWEB_LOG_DEBUG("read done.  Read {} bytes", bytesRead);
                std::shared_ptr<ConsoleHandler> self = weakSelf.lock();
                if (self == nullptr)
                {
                    return;
                }
                if (ec)
                {
                    BMCWEB_LOG_ERROR("Couldn't read from host serial port: {}",
                                     ec.message());
                    conn.close("Error connecting to host port");
                    return;
                }
                std::string_view payload(outputBuffer.data(), bytesRead);
                self->conn.sendEx(
                    crow::websocket::MessageType::Binary, payload,
                    std::bind_front(afterSendEx, weak_from_this()));
            });
    }
 
    bool connect(int fd)
    {
        boost::system::error_code ec;
        boost::asio::local::stream_protocol proto;
        hostSocket.assign(proto, fd, ec);
        if (ec)
        {
            BMCWEB_LOG_ERROR(
                "Failed to assign the DBUS socket Socket assign error: {}",
                ec.message());
            return false;
        }
        conn.resumeRead();
        doWrite();
        doRead();
        return true;
    }
 
    boost::asio::local::stream_protocol::socket hostSocket;
    std::array<char, 4096> outputBuffer{};
    std::string inputBuffer;
    bool doingWrite = false;
    crow::websocket::Connection& conn;
};
 
using ObmcConsoleMap = boost::container::flat_map<
    crow::websocket::Connection*, std::shared_ptr<ConsoleHandler>, std::less<>,
    std::vector<std::pair<crow::websocket::Connection*,
                          std::shared_ptr<ConsoleHandler>>>>;
 
inline ObmcConsoleMap& getConsoleHandlerMap()
{
    static ObmcConsoleMap map;
    return map;
}
 
// Remove connection from the connection map and if connection map is empty
// then remove the handler from handlers map.
inline void onClose(crow::websocket::Connection& conn, const std::string& err)
{
    BMCWEB_LOG_INFO("Closing websocket. Reason: {}", err);
    auto iter = getConsoleHandlerMap().find(&conn);
    if (iter == getConsoleHandlerMap().end())
    {
        BMCWEB_LOG_CRITICAL("Unable to find connection {}", logPtr(&conn));
        return;
    }
    BMCWEB_LOG_DEBUG("Remove connection {} from obmc console", logPtr(&conn));
    // Removed last connection so remove the path
    getConsoleHandlerMap().erase(iter);
}
 
inline void connectConsoleSocket(crow::websocket::Connection& conn,
                                 const boost::system::error_code& ec,
                                 const sdbusplus::message::unix_fd& unixfd)
{
    if (ec)
    {
        BMCWEB_LOG_ERROR(
            "Failed to call console Connect() method DBUS error: {}",
            ec.message());
        conn.close("Failed to connect");
        return;
    }
 
    // Look up the handler
    auto iter = getConsoleHandlerMap().find(&conn);
    if (iter == getConsoleHandlerMap().end())
    {
        BMCWEB_LOG_ERROR("Connection was already closed");
        return;
    }
 
    int fd = dup(unixfd);
    if (fd == -1)
    {
        BMCWEB_LOG_ERROR("Failed to dup the DBUS unixfd error: {}",
                         strerror(errno));
        conn.close("Internal error");
        return;
    }
    BMCWEB_LOG_DEBUG("Console duped FD: {}", fd);
    if (!iter->second->connect(fd))
    {
        close(fd);
        conn.close("Internal Error");
    }
}
 
inline void processConsoleObject(
    crow::websocket::Connection& conn, const std::string& consoleObjPath,
    const boost::system::error_code& ec,
    const ::dbus::utility::MapperGetObject& objInfo)
{
    // Look up the handler
    auto iter = getConsoleHandlerMap().find(&conn);
    if (iter == getConsoleHandlerMap().end())
    {
        BMCWEB_LOG_ERROR("Connection was already closed");
        return;
    }
    if (ec)
    {
        BMCWEB_LOG_WARNING("getDbusObject() for consoles failed. DBUS error:{}",
                           ec.message());
        conn.close("getDbusObject() for consoles failed.");
        return;
    }
 
    const auto valueIface = objInfo.begin();
    if (valueIface == objInfo.end())
    {
        BMCWEB_LOG_WARNING("getDbusObject() returned unexpected size: {}",
                           objInfo.size());
        conn.close("getDbusObject() returned unexpected size");
        return;
    }
 
    const std::string& consoleService = valueIface->first;
    BMCWEB_LOG_DEBUG("Looking up unixFD for Service {} Path {}", consoleService,
                     consoleObjPath);
    // Call Connect() method to get the unix FD
    crow::connections::systemBus->async_method_call(
        [&conn](const boost::system::error_code& ec1,
                const sdbusplus::message::unix_fd& unixfd) {
            connectConsoleSocket(conn, ec1, unixfd);
        },
        consoleService, consoleObjPath, "xyz.openbmc_project.Console.Access",
        "Connect");
}
 
// Query consoles from DBUS and find the matching to the
// rules string.
inline void onOpen(crow::websocket::Connection& conn)
{
    std::string consoleLeaf;
    BMCWEB_LOG_DEBUG("Connection {} opened", logPtr(&conn));
    if (getConsoleHandlerMap().size() >= maxSessions)
    {
        conn.close("Max sessions are already connected");
        return;
    }
 
    std::shared_ptr<ConsoleHandler> handler =
        std::make_shared<ConsoleHandler>(conn.getIoContext(), conn);
    getConsoleHandlerMap().emplace(&conn, handler);
 
    conn.deferRead();
 
    // Keep old path for backward compatibility
    if (conn.url().path() == "/console0")
    {
        consoleLeaf = "default";
    }
    else
    {
        // Get the console id from console router path and prepare the console
        // object path and console service.
        consoleLeaf = conn.url().segments().back();
    }
    std::string consolePath =
        sdbusplus::message::object_path("/xyz/openbmc_project/console") /
        consoleLeaf;
 
    BMCWEB_LOG_DEBUG("Console Object path = {} Request target = {}",
                     consolePath, conn.url().path());
 
    // mapper call lambda
    constexpr std::array<std::string_view, 1> interfaces = {
        "xyz.openbmc_project.Console.Access"};
 
    dbus::utility::getDbusObject(
        consolePath, interfaces,
        [&conn, consolePath](const boost::system::error_code& ec,
                             const ::dbus::utility::MapperGetObject& objInfo) {
            processConsoleObject(conn, consolePath, ec, objInfo);
        });
}
 
inline void onMessage(crow::websocket::Connection& conn,
                      const std::string& data, bool /*isBinary*/)
{
    auto handler = getConsoleHandlerMap().find(&conn);
    if (handler == getConsoleHandlerMap().end())
    {
        BMCWEB_LOG_CRITICAL("Unable to find connection {}", logPtr(&conn));
        return;
    }
    handler->second->inputBuffer += data;
    handler->second->doWrite();
}
 
inline void requestRoutes(App& app)
{
    BMCWEB_ROUTE(app, "/console0")
        .privileges({{"OpenBMCHostConsole"}})
        .websocket()
        .onopen(onOpen)
        .onclose(onClose)
        .onmessage(onMessage);
 
    BMCWEB_ROUTE(app, "/console/<str>")
        .privileges({{"OpenBMCHostConsole"}})
        .websocket()
        .onopen(onOpen)
        .onclose(onClose)
        .onmessage(onMessage);
}
} // namespace obmc_console
} // namespace crow

如果bmcweb和多路主机相连,即需要操作多个console,openbmc该如何配置对应的uart,保证不同的路由 BMCWEB_ROUTE(app, "/console/")选择不同的uart连接到指定的主机

阅读 988
avatarAI BotBETA

在OpenBMC环境中,当bmcweb需要支持多个控制台(console)会话,每个会话连接到不同的UART端口或不同的主机时,通常需要通过几个步骤来配置和扩展当前的系统。这里的关键在于如何根据请求的路由动态地连接到正确的UART端口或主机。

步骤 1: 扩展 DBUS 接口

首先,你需要确保你的DBUS服务(通常是在你的BMC固件中实现的)能够根据不同的请求(如基于路径的/console/serial0/console/serial1等)返回正确的UART端口或主机连接信息。这可能需要你修改或扩展现有的DBUS服务来支持更多的连接选项。

步骤 2: 修改 bmcweb 路由处理

在你的bmcweb代码中,你已经有了基本的路由处理(如/console0/console/<str>)。你需要扩展这些路由,以便它们可以根据URL的<str>部分来决定连接到哪个UART或主机。

  • 修改onOpen函数:确保在建立WebSocket连接时,根据请求的URL(如/console/serial0)来查询正确的DBUS服务并获取相应的连接信息(如UNIX文件描述符)。
  • 更新processConsoleObjectconnectConsoleSocket函数:这些函数需要根据从DBUS服务获取的信息来设置正确的UART连接。你可能需要传递额外的参数(如UART端口名称或主机ID)给这些函数。

步骤 3: 映射 URL 到 UART/主机

在你的BMC固件或配置中,你需要有一个方法来映射URL的<str>部分到具体的UART端口或主机。这可以通过配置文件、环境变量或直接在DBUS服务中硬编码来实现。

步骤 4: 测试和验证

完成上述更改后,你需要进行彻底的测试来确保:

  • 每个控制台路由都能正确连接到其指定的UART端口或主机。
  • 并发连接(如果支持)不会相互干扰。
  • 安全性(如权限检查)仍然得到妥善处理。

示例代码扩展

以下是一个简化的例子,展示了如何在onOpen函数中根据URL动态选择UART端口:

void onOpen(crow::websocket::Connection& conn)
{
    // ... (省略了之前的代码)

    std::string consoleLeaf = conn.url().segments().back();
    std::string uartPort = getUartPortFromConsoleLeaf(consoleLeaf); // 假设这个函数根据consoleLeaf返回UART端口名

    // 使用uartPort来配置你的连接
    // ... (省略了实际的连接配置代码)
}

std::string getUartPortFromConsoleLeaf(const std::string& leaf)
{
    // 这里应该有逻辑来根据leaf(如"serial0", "serial1"等)返回正确的UART端口名
    // 示例:硬编码返回
    if (leaf == "serial0")
    {
        return "/dev/ttyS0";
    }
    else if (leaf == "serial1")
    {
        return "/dev/ttyS1";
    }
    // ... (更多条件分支)
    return ""; // 默认或错误情况
}

注意:这个例子假设你有一个函数getUartPortFromConsoleLeaf,它根据consoleLeaf的值返回UART端口名。在实际应用中,这个函数可能需要与DBUS服务交互来获取端口名。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
logo
Microsoft
子站问答
访问
宣传栏