xref: /openbmc/openpower-proc-control/targeting.cpp (revision e84b4ddb671d97cb4725ee2de18979c69e42b971)
12c05aa76SMatt Spinler /**
2*e84b4ddbSPatrick 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 
178316b77bSEdward A. James #include <endian.h>
182c05aa76SMatt Spinler #include <experimental/filesystem>
19a231ceb4SMatt Spinler #include <phosphor-logging/elog.hpp>
20a231ceb4SMatt Spinler #include <phosphor-logging/elog-errors.hpp>
212c05aa76SMatt Spinler #include <phosphor-logging/log.hpp>
222c05aa76SMatt Spinler #include <regex>
23a231ceb4SMatt Spinler #include <xyz/openbmc_project/Common/File/error.hpp>
242c05aa76SMatt Spinler #include "targeting.hpp"
252c05aa76SMatt Spinler 
2618b0786aSDhruvaraj Subhashchandran 
272c05aa76SMatt Spinler namespace openpower
282c05aa76SMatt Spinler {
292c05aa76SMatt Spinler namespace targeting
302c05aa76SMatt Spinler {
312c05aa76SMatt Spinler 
322c05aa76SMatt Spinler using namespace phosphor::logging;
332c05aa76SMatt Spinler namespace fs = std::experimental::filesystem;
34a231ceb4SMatt Spinler namespace file_error = sdbusplus::xyz::openbmc_project::Common::File::Error;
352c05aa76SMatt Spinler 
36c3bffed7SMatt Spinler int Target::getCFAMFD()
37c3bffed7SMatt Spinler {
38c3bffed7SMatt Spinler     if (cfamFD.get() == nullptr)
39c3bffed7SMatt Spinler     {
40c3bffed7SMatt Spinler         cfamFD = std::make_unique<
41c3bffed7SMatt Spinler             openpower::util::FileDescriptor>(getCFAMPath());
42c3bffed7SMatt Spinler     }
43c3bffed7SMatt Spinler 
44c3bffed7SMatt Spinler     return cfamFD->get();
45c3bffed7SMatt Spinler }
46c3bffed7SMatt Spinler 
47be407166SMichael Tritz std::unique_ptr<Target>& Targeting::getTarget(size_t pos)
48be407166SMichael Tritz {
49be407166SMichael Tritz     auto search = [pos](const auto& t)
50be407166SMichael Tritz     {
51be407166SMichael Tritz         return t->getPos() == pos;
52be407166SMichael Tritz     };
53be407166SMichael Tritz 
54be407166SMichael Tritz     auto target = find_if(targets.begin(), targets.end(), search);
55be407166SMichael Tritz     if (target == targets.end())
56be407166SMichael Tritz     {
57be407166SMichael Tritz         throw std::runtime_error("Target not found: " + std::to_string(pos));
58be407166SMichael Tritz     }
59be407166SMichael Tritz     else
60be407166SMichael Tritz     {
61be407166SMichael Tritz         return *target;
62be407166SMichael Tritz     }
63be407166SMichael Tritz }
64be407166SMichael Tritz 
65c3bffed7SMatt Spinler 
668316b77bSEdward A. James static uint32_t noEndianSwap(uint32_t data)
678316b77bSEdward A. James {
688316b77bSEdward A. James     return data;
698316b77bSEdward A. James }
708316b77bSEdward A. James 
718316b77bSEdward A. James static uint32_t endianSwap(uint32_t data)
728316b77bSEdward A. James {
738316b77bSEdward A. James     return htobe32(data);
748316b77bSEdward A. James }
758316b77bSEdward A. James 
762c05aa76SMatt Spinler Targeting::Targeting(const std::string& fsiMasterDev,
772c05aa76SMatt Spinler                      const std::string& fsiSlaveDir) :
782c05aa76SMatt Spinler     fsiMasterPath(fsiMasterDev),
792c05aa76SMatt Spinler     fsiSlaveBasePath(fsiSlaveDir)
802c05aa76SMatt Spinler {
818316b77bSEdward A. James     swap_endian_t swapper = endianSwap;
828316b77bSEdward A. James     std::regex exp{"fsi1/slave@([0-9]{2}):00", std::regex::extended};
838316b77bSEdward A. James 
848316b77bSEdward A. James     if (!fs::exists(fsiMasterPath))
858316b77bSEdward A. James     {
868316b77bSEdward A. James         std::regex expOld{"hub@00/slave@([0-9]{2}):00", std::regex::extended};
878316b77bSEdward A. James 
888316b77bSEdward A. James         //Fall back to old (4.7) path
898316b77bSEdward A. James         exp = expOld;
908316b77bSEdward A. James         fsiMasterPath = fsiMasterDevPathOld;
918316b77bSEdward A. James         fsiSlaveBasePath = fsiSlaveBaseDirOld;
928316b77bSEdward A. James 
938316b77bSEdward A. James         //And don't swap the endianness of CFAM data
948316b77bSEdward A. James         swapper = noEndianSwap;
958316b77bSEdward A. James     }
968316b77bSEdward A. James 
972c05aa76SMatt Spinler     //Always create P0, the FSI master.
988316b77bSEdward A. James     targets.push_back(std::make_unique<Target>(0, fsiMasterPath, swapper));
9918b0786aSDhruvaraj Subhashchandran     try
10018b0786aSDhruvaraj Subhashchandran     {
1012c05aa76SMatt Spinler         //Find the the remaining P9s dynamically based on which files show up
1022c05aa76SMatt Spinler         for (auto& file : fs::directory_iterator(fsiSlaveBasePath))
1032c05aa76SMatt Spinler         {
1042c05aa76SMatt Spinler             std::smatch match;
1052c05aa76SMatt Spinler             std::string path = file.path();
1062c05aa76SMatt Spinler             if (std::regex_search(path, match, exp))
1072c05aa76SMatt Spinler             {
1082c05aa76SMatt Spinler                 auto pos = atoi(match[1].str().c_str());
1092c05aa76SMatt Spinler                 if (pos == 0)
1102c05aa76SMatt Spinler                 {
1112c05aa76SMatt Spinler                     log<level::ERR>("Unexpected FSI slave device name found",
112fabe92e8SMatt Spinler                                     entry("DEVICE_NAME=%s", path.c_str()));
1132c05aa76SMatt Spinler                     continue;
1142c05aa76SMatt Spinler                 }
1152c05aa76SMatt Spinler 
1162c05aa76SMatt Spinler                 path += "/raw";
1172c05aa76SMatt Spinler 
1188316b77bSEdward A. James                 targets.push_back(std::make_unique<Target>(pos, path, swapper));
1192c05aa76SMatt Spinler             }
1202c05aa76SMatt Spinler         }
12118b0786aSDhruvaraj Subhashchandran     }
12218b0786aSDhruvaraj Subhashchandran     catch (fs::filesystem_error& e)
12318b0786aSDhruvaraj Subhashchandran     {
124a231ceb4SMatt Spinler         using metadata = xyz::openbmc_project::Common::File::Open;
125a231ceb4SMatt Spinler 
126a231ceb4SMatt Spinler         elog<file_error::Open>(
127a231ceb4SMatt Spinler                 metadata::ERRNO(e.code().value()),
128a231ceb4SMatt Spinler                 metadata::PATH(e.path1().c_str()));
12918b0786aSDhruvaraj Subhashchandran     }
1302c05aa76SMatt Spinler 
1312c05aa76SMatt Spinler     auto sortTargets = [](const std::unique_ptr<Target>& left,
1322c05aa76SMatt Spinler                           const std::unique_ptr<Target>& right)
1332c05aa76SMatt Spinler     {
1342c05aa76SMatt Spinler         return left->getPos() < right->getPos();
1352c05aa76SMatt Spinler     };
1362c05aa76SMatt Spinler     std::sort(targets.begin(), targets.end(), sortTargets);
1372c05aa76SMatt Spinler }
1382c05aa76SMatt Spinler 
1392c05aa76SMatt Spinler }
1402c05aa76SMatt Spinler }
141