#include #include #include #include #include #include "occ_pass_through.hpp" #include "occ_finder.hpp" namespace open_power { namespace occ { namespace pass_through { void run() { auto bus = sdbusplus::bus::new_default(); sdbusplus::server::manager::manager objManager(bus, OCC_PASS_THROUGH_ROOT); std::vector> objects; auto occs = open_power::occ::finder::get(); for (const auto& occ : occs) { auto occPassThrough = object(occ); objects.emplace_back( std::make_unique(bus, occPassThrough.c_str())); } bus.request_name(OCC_PASS_THROUGH_BUSNAME); while (true) { bus.process_discard(); bus.wait(); } } PassThrough::PassThrough( sdbusplus::bus::bus& bus, const char* path) : Iface(bus, path), path(path), fd(openDevice()) { // Nothing to do. } int PassThrough::openDevice() { // Device instance number starts from 1. devicePath.append(std::to_string((this->path.back() - '0') + 1)); int fd = open(devicePath.c_str(), O_RDWR | O_NONBLOCK); if (fd < 0) { // This is for completion. This is getting replaced by elog // in the next commit throw std::runtime_error("Error opening " + devicePath); } return fd; } std::vector PassThrough::send(std::vector command) { using namespace phosphor::logging; std::vector response {}; // Amester packs data in 4 bytes ssize_t size = command.size() * sizeof(int32_t); auto rc = write((fd)(), command.data(), size); if (rc < 0 || (rc != size)) { log("Error writing to OCC"); // In the next commit, it will have exceptions. return response; } // Now read the response. This would be the content of occ-sram while(1) { errno = 0; int32_t data {}; auto len = read((fd)(), &data, sizeof(data)); if (len > 0) { response.emplace_back(data); } else if (len < 0 && errno == EAGAIN) { // We may have data coming still continue; } else if (len == 0) { // We have read all that we can. break; } else { // Will have exception in the next commit. log("Error reading from OCC"); break; } } return response; } } // namespace pass_through } // namespace occ } // namespace open_power