/* IBM_PROLOG_BEGIN_TAG */ /* This is an automatically generated prolog. */ /* */ /* $Source: src/usr/diag/attn/ipl/test/attnfakesys.C $ */ /* */ /* OpenPOWER HostBoot Project */ /* */ /* Contributors Listed Below - COPYRIGHT 2014,2016 */ /* [+] International Business Machines Corp. */ /* */ /* */ /* Licensed under the Apache License, Version 2.0 (the "License"); */ /* you may not use this file except in compliance with the License. */ /* You may obtain a copy of the License at */ /* */ /* http://www.apache.org/licenses/LICENSE-2.0 */ /* */ /* Unless required by applicable law or agreed to in writing, software */ /* distributed under the License is distributed on an "AS IS" BASIS, */ /* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or */ /* implied. See the License for the specific language governing */ /* permissions and limitations under the License. */ /* */ /* IBM_PROLOG_END_TAG */ /** * @file attnfakesys.C * * @brief HBATTN fake system class method definitions. */ #include "../../common/attnlist.H" #include "../../common/attntrace.H" #include "../../common/attntarget.H" #include "attnfakeelement.H" #include "attnfakesys.H" #include "attntest.H" #include #include using namespace std; using namespace PRDF; using namespace TARGETING; namespace ATTN { errlHndl_t FakeSystem::putScom( TargetHandle_t i_target, uint64_t i_address, uint64_t i_data) { errlHndl_t err = 0; mutex_lock(&iv_mutex); ATTN_TRACE("FakeSystem::putScom Target:%d Addr:%X", i_target, i_address); err = putReg(i_target, i_address, i_data); mutex_unlock(&iv_mutex); return err; } errlHndl_t FakeSystem::getScom( TargetHandle_t i_target, uint64_t i_address, uint64_t & o_data) { mutex_lock(&iv_mutex); ATTN_TRACE("FakeSystem::getScom Target:%d Addr:%X", i_target, i_address); o_data = getReg(i_target, i_address); mutex_unlock(&iv_mutex); return 0; } errlHndl_t FakeSystem::modifyScom( TargetHandle_t i_target, uint64_t i_address, uint64_t i_data, uint64_t & o_data, ScomOp i_op) { errlHndl_t err = 0; mutex_lock(&iv_mutex); o_data = iv_regs[i_target][i_address]; err = modifyReg(i_target, i_address, i_data, i_op); mutex_unlock(&iv_mutex); return err; } bool FakeSystem::wait(uint64_t i_maxWaitNs) { uint64_t count = 0; uint64_t previous = 0; uint64_t elapsedNs = 0; do { mutex_lock(&iv_mutex); previous = count; count = iv_attentions.size(); ATTN_TRACE("FakeSystem::wait attnCount:%d", count); mutex_unlock(&iv_mutex); if(!count) { break; } nanosleep(0, TEN_CTX_SWITCHES_NS); if(count == previous) { elapsedNs += TEN_CTX_SWITCHES_NS; } } while(elapsedNs < i_maxWaitNs); return count == 0; } void FakeSystem::wait() { mutex_lock(&iv_mutex); while(!iv_attentions.empty()) { ATTN_TRACE("FakeSystem::wait attnCount:%d", iv_attentions.size()); sync_cond_wait(&iv_cond, &iv_mutex); } mutex_unlock(&iv_mutex); } void FakeSystem::addReg(uint64_t i_address, FakeReg & i_element) { // register a new implemenation element mutex_lock(&iv_mutex); iv_regImpl.push_back(make_pair(i_address, &i_element)); mutex_unlock(&iv_mutex); } void FakeSystem::addSource( TYPE i_targetType, ATTENTION_VALUE_TYPE i_type, FakeSource & i_element) { // register a new implemenation element mutex_lock(&iv_mutex); iv_sources.push_back(make_pair(make_pair(i_targetType, i_type), &i_element)); mutex_unlock(&iv_mutex); } errlHndl_t FakeSystem::putReg( TargetHandle_t i_target, uint64_t i_address, uint64_t i_data) { uint64_t data = iv_regs[i_target][i_address]; putRegUnsafe(i_target, i_address, i_data); return makeRegCallbacks( i_target, i_address, i_data, data); } void FakeSystem::putRegUnsafe( TargetHandle_t i_target, uint64_t i_address, uint64_t i_data) { ATTN_TRACE("FakeSystem::putReg: tgt: %p, add: %016x, data: %016x", i_target, i_address, i_data); iv_regs[i_target][i_address] = i_data; } uint64_t FakeSystem::getReg( TargetHandle_t i_target, uint64_t i_address) { uint64_t data = iv_regs[i_target][i_address]; return data; } errlHndl_t FakeSystem::modifyReg( TargetHandle_t i_target, uint64_t i_address, uint64_t i_data, ScomOp i_op) { errlHndl_t err = 0; uint64_t data = iv_regs[i_target][i_address]; uint64_t changedData = i_op == SCOM_OR ? (data | i_data) : (data & i_data); bool changed = changedData != data; if(changed) { putRegUnsafe( i_target, i_address, changedData); err = makeRegCallbacks( i_target, i_address, changedData, data); } return err; } errlHndl_t FakeSystem::makeRegCallbacks( TargetHandle_t i_target, uint64_t i_address, uint64_t i_new, uint64_t i_old) { // inform each element of the update errlHndl_t err = 0; vector >::iterator it = iv_regImpl.begin(); while(!err && it != iv_regImpl.end()) { if(it->first == i_address || it->first == 0) { err = (*it->second).processPutReg( *this, i_target, i_address, i_new, i_old); } ++it; } return err; } errlHndl_t FakeSystem::makeAttnCallbacks( const AttnList & i_list, bool i_set) { errlHndl_t err = 0; AttnList::const_iterator ait = i_list.begin(); while(ait != i_list.end()) { vector::iterator it = iv_sources.begin(); TYPE type = getTargetService().getType(ait->targetHndl); AttnDataMap::iterator tit = iv_attentions.find(*ait); uint64_t count = tit == iv_attentions.end() ? 0 : tit->second; while(!err && it != iv_sources.end()) { Key & k = it->first; if(k.first == TYPE_NA || (k.first == type && k.second == ait->attnType)) { ATTN_TRACE("FakeSystem::makeAttnCallbacks i_set:%d", i_set); if(i_set) { err = (*it->second).processPutAttention( *this, *ait, count); } else { err = (*it->second).processClearAttention( *this, *ait, count); } } ++it; } ++ait; } return err; } errlHndl_t FakeSystem::putAttentions( const AttnList & i_list) { errlHndl_t err = 0; // inject the attentions and then // inform each element mutex_lock(&iv_mutex); AttnList::const_iterator ait = i_list.begin(); while(ait != i_list.end()) { ATTN_TRACE("FakeSystem::putAttention: tgt: %p, type: %d", ait->targetHndl, ait->attnType); iv_attentions[*ait]++; ++ait; } err = makeAttnCallbacks(i_list, true); mutex_unlock(&iv_mutex); return err; } void FakeSystem::clearAttentionUnsafe( const AttnData & i_attn) { ATTN_TRACE("FakeSystem::clearAttnUnsafe: tgt: %p, type: %d", i_attn.targetHndl, i_attn.attnType); // clear the attention and then // inform each element iv_attentions[i_attn]--; if(!iv_attentions[i_attn]) { iv_attentions.erase(i_attn); } } errlHndl_t FakeSystem::clearAttention( const PRDF::AttnData & i_attn) { errlHndl_t err = 0; mutex_lock(&iv_mutex); clearAttentionUnsafe(i_attn); bool done = iv_attentions.empty(); err = makeAttnCallbacks(AttnList(1, i_attn), false); mutex_unlock(&iv_mutex); if(done) { sync_cond_signal(&iv_cond); } return err; } errlHndl_t FakeSystem::clearAllAttentions( const PRDF::AttnData & i_attn) { errlHndl_t err = 0; mutex_lock(&iv_mutex); AttnDataMap::iterator it = iv_attentions.find(i_attn); uint64_t count = it == iv_attentions.end() ? 0 : it->second, cit = count; while(cit--) { clearAttentionUnsafe(i_attn); } bool done = iv_attentions.empty(); err = makeAttnCallbacks(AttnList(count, i_attn), false); mutex_unlock(&iv_mutex); if(done) { sync_cond_signal(&iv_cond); } return err; } uint64_t FakeSystem::count() { mutex_lock(&iv_mutex); uint64_t c = iv_attentions.size(); mutex_unlock(&iv_mutex); return c; } uint64_t FakeSystem::count(const AttnData & i_attention) { AttnDataMap::iterator it = iv_attentions.find(i_attention); uint64_t count = it == iv_attentions.end() ? 0 : it->second; return count; } void FakeSystem::dump() { AttnDataMap::iterator it = iv_attentions.begin(); while(it != iv_attentions.end()) { ATTN_TRACE("target: %p, type: %d, count: %d", it->first.targetHndl, it->first.attnType, it->second); ++it; } } FakeSystem::FakeSystem() { mutex_init(&iv_mutex); sync_cond_init(&iv_cond); } FakeSystem::~FakeSystem() { sync_cond_destroy(&iv_cond); mutex_destroy(&iv_mutex); } }