xref: /openbmc/openpower-proc-control/targeting.cpp (revision f78d90420967100233cc9c2a5e0834e9789e2637)
12c05aa76SMatt Spinler /**
2e84b4ddbSPatrick Venture  * Copyright (C) 2017 IBM Corporation
32c05aa76SMatt Spinler  *
42c05aa76SMatt Spinler  * Licensed under the Apache License, Version 2.0 (the "License");
52c05aa76SMatt Spinler  * you may not use this file except in compliance with the License.
62c05aa76SMatt Spinler  * You may obtain a copy of the License at
72c05aa76SMatt Spinler  *
82c05aa76SMatt Spinler  *     http://www.apache.org/licenses/LICENSE-2.0
92c05aa76SMatt Spinler  *
102c05aa76SMatt Spinler  * Unless required by applicable law or agreed to in writing, software
112c05aa76SMatt Spinler  * distributed under the License is distributed on an "AS IS" BASIS,
122c05aa76SMatt Spinler  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
132c05aa76SMatt Spinler  * See the License for the specific language governing permissions and
142c05aa76SMatt Spinler  * limitations under the License.
152c05aa76SMatt Spinler  */
168316b77bSEdward A. James 
17*f78d9042SPatrick Venture #include "targeting.hpp"
18*f78d9042SPatrick Venture 
198316b77bSEdward A. James #include <endian.h>
20*f78d9042SPatrick Venture 
212c05aa76SMatt Spinler #include <experimental/filesystem>
22a231ceb4SMatt Spinler #include <phosphor-logging/elog-errors.hpp>
23*f78d9042SPatrick Venture #include <phosphor-logging/elog.hpp>
242c05aa76SMatt Spinler #include <phosphor-logging/log.hpp>
252c05aa76SMatt Spinler #include <regex>
26a231ceb4SMatt Spinler #include <xyz/openbmc_project/Common/File/error.hpp>
2718b0786aSDhruvaraj Subhashchandran 
282c05aa76SMatt Spinler namespace openpower
292c05aa76SMatt Spinler {
302c05aa76SMatt Spinler namespace targeting
312c05aa76SMatt Spinler {
322c05aa76SMatt Spinler 
332c05aa76SMatt Spinler using namespace phosphor::logging;
342c05aa76SMatt Spinler namespace fs = std::experimental::filesystem;
35a231ceb4SMatt Spinler namespace file_error = sdbusplus::xyz::openbmc_project::Common::File::Error;
362c05aa76SMatt Spinler 
37c3bffed7SMatt Spinler int Target::getCFAMFD()
38c3bffed7SMatt Spinler {
39c3bffed7SMatt Spinler     if (cfamFD.get() == nullptr)
40c3bffed7SMatt Spinler     {
41*f78d9042SPatrick Venture         cfamFD =
42*f78d9042SPatrick Venture             std::make_unique<openpower::util::FileDescriptor>(getCFAMPath());
43c3bffed7SMatt Spinler     }
44c3bffed7SMatt Spinler 
45c3bffed7SMatt Spinler     return cfamFD->get();
46c3bffed7SMatt Spinler }
47c3bffed7SMatt Spinler 
48be407166SMichael Tritz std::unique_ptr<Target>& Targeting::getTarget(size_t pos)
49be407166SMichael Tritz {
50*f78d9042SPatrick Venture     auto search = [pos](const auto& t) { return t->getPos() == pos; };
51be407166SMichael Tritz 
52be407166SMichael Tritz     auto target = find_if(targets.begin(), targets.end(), search);
53be407166SMichael Tritz     if (target == targets.end())
54be407166SMichael Tritz     {
55be407166SMichael Tritz         throw std::runtime_error("Target not found: " + std::to_string(pos));
56be407166SMichael Tritz     }
57be407166SMichael Tritz     else
58be407166SMichael Tritz     {
59be407166SMichael Tritz         return *target;
60be407166SMichael Tritz     }
61be407166SMichael Tritz }
62be407166SMichael Tritz 
638316b77bSEdward A. James static uint32_t noEndianSwap(uint32_t data)
648316b77bSEdward A. James {
658316b77bSEdward A. James     return data;
668316b77bSEdward A. James }
678316b77bSEdward A. James 
688316b77bSEdward A. James static uint32_t endianSwap(uint32_t data)
698316b77bSEdward A. James {
708316b77bSEdward A. James     return htobe32(data);
718316b77bSEdward A. James }
728316b77bSEdward A. James 
732c05aa76SMatt Spinler Targeting::Targeting(const std::string& fsiMasterDev,
742c05aa76SMatt Spinler                      const std::string& fsiSlaveDir) :
752c05aa76SMatt Spinler     fsiMasterPath(fsiMasterDev),
762c05aa76SMatt Spinler     fsiSlaveBasePath(fsiSlaveDir)
772c05aa76SMatt Spinler {
788316b77bSEdward A. James     swap_endian_t swapper = endianSwap;
798316b77bSEdward A. James     std::regex exp{"fsi1/slave@([0-9]{2}):00", std::regex::extended};
808316b77bSEdward A. James 
818316b77bSEdward A. James     if (!fs::exists(fsiMasterPath))
828316b77bSEdward A. James     {
838316b77bSEdward A. James         std::regex expOld{"hub@00/slave@([0-9]{2}):00", std::regex::extended};
848316b77bSEdward A. James 
858316b77bSEdward A. James         // Fall back to old (4.7) path
868316b77bSEdward A. James         exp = expOld;
878316b77bSEdward A. James         fsiMasterPath = fsiMasterDevPathOld;
888316b77bSEdward A. James         fsiSlaveBasePath = fsiSlaveBaseDirOld;
898316b77bSEdward A. James 
908316b77bSEdward A. James         // And don't swap the endianness of CFAM data
918316b77bSEdward A. James         swapper = noEndianSwap;
928316b77bSEdward A. James     }
938316b77bSEdward A. James 
942c05aa76SMatt Spinler     // Always create P0, the FSI master.
958316b77bSEdward A. James     targets.push_back(std::make_unique<Target>(0, fsiMasterPath, swapper));
9618b0786aSDhruvaraj Subhashchandran     try
9718b0786aSDhruvaraj Subhashchandran     {
982c05aa76SMatt Spinler         // Find the the remaining P9s dynamically based on which files show up
992c05aa76SMatt Spinler         for (auto& file : fs::directory_iterator(fsiSlaveBasePath))
1002c05aa76SMatt Spinler         {
1012c05aa76SMatt Spinler             std::smatch match;
1022c05aa76SMatt Spinler             std::string path = file.path();
1032c05aa76SMatt Spinler             if (std::regex_search(path, match, exp))
1042c05aa76SMatt Spinler             {
1052c05aa76SMatt Spinler                 auto pos = atoi(match[1].str().c_str());
1062c05aa76SMatt Spinler                 if (pos == 0)
1072c05aa76SMatt Spinler                 {
1082c05aa76SMatt Spinler                     log<level::ERR>("Unexpected FSI slave device name found",
109fabe92e8SMatt Spinler                                     entry("DEVICE_NAME=%s", path.c_str()));
1102c05aa76SMatt Spinler                     continue;
1112c05aa76SMatt Spinler                 }
1122c05aa76SMatt Spinler 
1132c05aa76SMatt Spinler                 path += "/raw";
1142c05aa76SMatt Spinler 
1158316b77bSEdward A. James                 targets.push_back(std::make_unique<Target>(pos, path, swapper));
1162c05aa76SMatt Spinler             }
1172c05aa76SMatt Spinler         }
11818b0786aSDhruvaraj Subhashchandran     }
11918b0786aSDhruvaraj Subhashchandran     catch (fs::filesystem_error& e)
12018b0786aSDhruvaraj Subhashchandran     {
121a231ceb4SMatt Spinler         using metadata = xyz::openbmc_project::Common::File::Open;
122a231ceb4SMatt Spinler 
123*f78d9042SPatrick Venture         elog<file_error::Open>(metadata::ERRNO(e.code().value()),
124a231ceb4SMatt Spinler                                metadata::PATH(e.path1().c_str()));
12518b0786aSDhruvaraj Subhashchandran     }
1262c05aa76SMatt Spinler 
1272c05aa76SMatt Spinler     auto sortTargets = [](const std::unique_ptr<Target>& left,
128*f78d9042SPatrick Venture                           const std::unique_ptr<Target>& right) {
1292c05aa76SMatt Spinler         return left->getPos() < right->getPos();
1302c05aa76SMatt Spinler     };
1312c05aa76SMatt Spinler     std::sort(targets.begin(), targets.end(), sortTargets);
1322c05aa76SMatt Spinler }
1332c05aa76SMatt Spinler 
134*f78d9042SPatrick Venture } // namespace targeting
135*f78d9042SPatrick Venture } // namespace openpower
136