mirror of
https://github.com/eclipse-openvehicle-api/openvehicle-api.git
synced 2026-04-15 09:38:16 +00:00
202 lines
7.4 KiB
C++
202 lines
7.4 KiB
C++
/********************************************************************************
|
|
* Copyright (c) 2025-2026 ZF Friedrichshafen AG
|
|
*
|
|
* This program and the accompanying materials are made available under the
|
|
* terms of the Apache License Version 2.0 which is available at
|
|
* https://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Contributors:
|
|
* Erik Verhoeven - initial API and implementation
|
|
********************************************************************************/
|
|
|
|
#include "dispatchservice.h"
|
|
|
|
|
|
CDispatchService::CDispatchService()
|
|
{
|
|
CreateDirectTransactionID();
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::CreateTxTrigger(uint32_t uiCycleTime, uint32_t uiDelayTime, uint32_t uiBehaviorFlags,
|
|
sdv::IInterfaceAccess* pTriggerCallback)
|
|
{
|
|
if (GetObjectState() != sdv::EObjectState::configuring) return nullptr;
|
|
|
|
// Check for parameter validity
|
|
if (!uiCycleTime && !(uiBehaviorFlags & static_cast<uint32_t>(sdv::core::ISignalTransmission::ETxTriggerBehavior::spontaneous)))
|
|
return nullptr;
|
|
if (!pTriggerCallback)
|
|
return nullptr;
|
|
sdv::core::ITxTriggerCallback* pCallback = pTriggerCallback->GetInterface<sdv::core::ITxTriggerCallback>();
|
|
if (!pCallback) return nullptr;
|
|
|
|
std::unique_ptr<CTrigger> ptrTrigger = std::make_unique<CTrigger>(*this, uiCycleTime, uiDelayTime, uiBehaviorFlags, pCallback);
|
|
// Ignore cppcheck warning; normally the returned pointer should always have a value at this stage (otherwise an
|
|
// exception was triggered).
|
|
// cppcheck-suppress knownConditionTrueFalse
|
|
if (!ptrTrigger)
|
|
return nullptr;
|
|
CTrigger* pObject = ptrTrigger.get();
|
|
// Ignore cppcheck warning; normally the returned pointer should always have a value at this stage (otherwise an
|
|
// exception was triggered).
|
|
// cppcheck-suppress knownConditionTrueFalse
|
|
if (!pObject)
|
|
return nullptr;
|
|
if (!ptrTrigger->IsValid()) return nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxTriggers);
|
|
return m_mapTriggers.emplace(pObject, std::move(ptrTrigger)).second ? pObject : nullptr;
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::RegisterTxSignal(/*in*/ const sdv::u8string& ssSignalName, /*in*/ sdv::any_t anyDefVal)
|
|
{
|
|
if (GetObjectState() != sdv::EObjectState::configuring) return nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
|
|
auto prSignal = m_mapTxSignals.try_emplace(ssSignalName, *this, ssSignalName, sdv::core::ESignalDirection::sigdir_tx, anyDefVal);
|
|
return prSignal.first->second.CreateConsumer();
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::RegisterRxSignal(/*in*/ const sdv::u8string& ssSignalName)
|
|
{
|
|
if (GetObjectState() != sdv::EObjectState::configuring) return nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
auto prSignal = m_mapRxSignals.try_emplace(ssSignalName, *this, ssSignalName, sdv::core::ESignalDirection::sigdir_rx);
|
|
return prSignal.first->second.CreateProvider();
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::RequestSignalPublisher(/*in*/ const sdv::u8string& ssSignalName)
|
|
{
|
|
if (GetObjectState() != sdv::EObjectState::configuring) return nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
auto itSignal = m_mapTxSignals.find(ssSignalName);
|
|
if (itSignal == m_mapTxSignals.end()) return nullptr;
|
|
if (itSignal->second.GetDirection() != sdv::core::ESignalDirection::sigdir_tx)
|
|
return nullptr;
|
|
return itSignal->second.CreateProvider();
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::AddSignalSubscription(/*in*/ const sdv::u8string& ssSignalName, /*in*/ IInterfaceAccess* pSubscriber)
|
|
{
|
|
if (GetObjectState() != sdv::EObjectState::configuring) return nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
auto itSignal = m_mapRxSignals.find(ssSignalName);
|
|
if (itSignal == m_mapRxSignals.end()) return 0;
|
|
if (itSignal->second.GetDirection() != sdv::core::ESignalDirection::sigdir_rx)
|
|
return 0ull;
|
|
return itSignal->second.CreateConsumer(pSubscriber);
|
|
}
|
|
|
|
sdv::sequence<sdv::core::SSignalRegistration> CDispatchService::GetRegisteredSignals() const
|
|
{
|
|
sdv::sequence<sdv::core::SSignalRegistration> seqRegistrations;
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
for (const auto& prSignal : m_mapRxSignals)
|
|
seqRegistrations.push_back({prSignal.first, prSignal.second.GetDirection()});
|
|
for (const auto& prSignal : m_mapTxSignals)
|
|
seqRegistrations.push_back({prSignal.first, prSignal.second.GetDirection()});
|
|
return seqRegistrations;
|
|
}
|
|
|
|
sdv::IInterfaceAccess* CDispatchService::CreateTransaction()
|
|
{
|
|
// NOTE: Transactions can take place at any time (not only when running). DO not restrict transactions to any
|
|
// operation mode.
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxTransactions);
|
|
auto itTransaction = m_lstTransactions.emplace(m_lstTransactions.end(), *this);
|
|
if (itTransaction == m_lstTransactions.end()) return nullptr;
|
|
itTransaction->SetIterator(itTransaction);
|
|
return &(*itTransaction);
|
|
}
|
|
|
|
uint64_t CDispatchService::GetNextTransactionID()
|
|
{
|
|
return m_uiTransactionCnt++;
|
|
}
|
|
|
|
void CDispatchService::CreateDirectTransactionID()
|
|
{
|
|
m_uiDirectTransactionID = GetNextTransactionID();
|
|
}
|
|
|
|
uint64_t CDispatchService::GetDirectTransactionID() const
|
|
{
|
|
return m_uiDirectTransactionID;
|
|
}
|
|
|
|
bool CDispatchService::OnInitialize()
|
|
{
|
|
m_scheduler.Start();
|
|
return true;
|
|
}
|
|
|
|
void CDispatchService::OnShutdown()
|
|
{
|
|
m_scheduler.Stop();
|
|
}
|
|
|
|
void CDispatchService::UnregisterSignal(/*in*/ const sdv::u8string& ssSignalName, sdv::core::ESignalDirection eDirection)
|
|
{
|
|
// NOTE: Normally the remove function should be called in the configuration mode. Since it doesn't give
|
|
// feedback and the associated caller might delete any receiving function, allow the removal to take place even
|
|
// when running.
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
|
|
// Remove the signal
|
|
if (eDirection == sdv::core::ESignalDirection::sigdir_rx)
|
|
m_mapRxSignals.erase(ssSignalName);
|
|
else
|
|
m_mapTxSignals.erase(ssSignalName);
|
|
}
|
|
|
|
CSignal* CDispatchService::FindSignal(const sdv::u8string& rssSignalName, sdv::core::ESignalDirection eDirection)
|
|
{
|
|
std::unique_lock<std::mutex> lock(m_mtxSignals);
|
|
|
|
if (eDirection == sdv::core::ESignalDirection::sigdir_rx)
|
|
{
|
|
auto itSignal = m_mapRxSignals.find(rssSignalName);
|
|
return itSignal == m_mapRxSignals.end() ? nullptr : &itSignal->second;
|
|
} else
|
|
{
|
|
auto itSignal = m_mapTxSignals.find(rssSignalName);
|
|
return itSignal == m_mapTxSignals.end() ? nullptr : &itSignal->second;
|
|
}
|
|
}
|
|
|
|
void CDispatchService::FinishTransaction(const CTransaction* pTransaction)
|
|
{
|
|
// NOTE: Transactions can take place at any time (not only when running). DO not restrict transactions to any
|
|
// operation mode.
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxTransactions);
|
|
|
|
// Delete the transaction
|
|
auto itTransaction = pTransaction->GetIterator();
|
|
m_lstTransactions.erase(itTransaction);
|
|
}
|
|
|
|
CScheduler& CDispatchService::GetScheduler()
|
|
{
|
|
return m_scheduler;
|
|
}
|
|
|
|
void CDispatchService::RemoveTxTrigger(CTrigger* pTrigger)
|
|
{
|
|
// NOTE: Normally the remove function should be called in the configuration mode. Since it doesn't give
|
|
// feedback and the associated caller might delete any receiving function, allow the removal to take place even
|
|
// when running.
|
|
|
|
std::unique_lock<std::mutex> lock(m_mtxTriggers);
|
|
m_scheduler.RemoveFromSchedule(pTrigger);
|
|
m_mapTriggers.erase(pTrigger);
|
|
}
|