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