mirror of
https://github.com/nasa/fprime.git
synced 2025-12-10 17:47:10 -06:00
* manual packet gen * Added tlmPkt to deployment * Packet files * Fixes to packet gen script * format python, updated packet files * spelling fixes * integrating packet autocoder * final updates for telemetry packetization * formatting telemetry packetizer cpde * moving tlm packetizer configuration file * resetting stock telemetry channelizer * missed packetizer setup call * fixing autocoding miss-match with Svc::TlmChan Co-authored-by: M Starch <LeStarch@googlemail.com>
189 lines
7.9 KiB
C++
189 lines
7.9 KiB
C++
// ======================================================================
|
||
// \title Topology.cpp
|
||
// \author mstarch
|
||
// \brief cpp file containing the topology instantiation code
|
||
//
|
||
// \copyright
|
||
// Copyright 2009-2022, by the California Institute of Technology.
|
||
// ALL RIGHTS RESERVED. United States Government Sponsorship
|
||
// acknowledged.
|
||
// ======================================================================
|
||
// Provides access to autocoded functions
|
||
#include <Ref/Top/RefTopologyAc.hpp>
|
||
#include <Ref/Top/RefPacketsAc.hpp>
|
||
|
||
// Necessary project-specified types
|
||
#include <Fw/Types/MallocAllocator.hpp>
|
||
#include <Os/Log.hpp>
|
||
#include <Svc/FramingProtocol/FprimeProtocol.hpp>
|
||
|
||
// Used for 1Hz synthetic cycling
|
||
#include <Os/Mutex.hpp>
|
||
|
||
// Allows easy reference to objects in FPP/autocoder required namespaces
|
||
using namespace Ref;
|
||
|
||
// Instantiate a system logger that will handle Fw::Logger::logMsg calls
|
||
Os::Log logger;
|
||
|
||
// The reference topology uses a malloc-based allocator for components that need to allocate memory during the
|
||
// initialization phase.
|
||
Fw::MallocAllocator mallocator;
|
||
|
||
// The reference topology uses the F´ packet protocol when communicating with the ground and therefore uses the F´
|
||
// framing and deframing implementations.
|
||
Svc::FprimeFraming framing;
|
||
Svc::FprimeDeframing deframing;
|
||
|
||
// The reference topology divides the incoming clock signal (1Hz) into sub-signals: 1Hz, 1/2Hz, and 1/4Hz
|
||
NATIVE_INT_TYPE rateGroupDivisors[Svc::RateGroupDriver::DIVIDER_SIZE] = {1, 2, 4};
|
||
|
||
// Rate groups may supply a context token to each of the attached children whose purpose is set by the project. The
|
||
// reference topology sets each token to zero as these contexts are unused in this project.
|
||
NATIVE_INT_TYPE rateGroup1Context[Svc::ActiveRateGroup::CONNECTION_COUNT_MAX] = {};
|
||
NATIVE_INT_TYPE rateGroup2Context[Svc::ActiveRateGroup::CONNECTION_COUNT_MAX] = {};
|
||
NATIVE_INT_TYPE rateGroup3Context[Svc::ActiveRateGroup::CONNECTION_COUNT_MAX] = {};
|
||
|
||
// A number of constants are needed for construction of the topology. These are specified here.
|
||
enum TopologyConstants {
|
||
CMD_SEQ_BUFFER_SIZE = 5 * 1024,
|
||
FILE_DOWNLINK_TIMEOUT = 1000,
|
||
FILE_DOWNLINK_COOLDOWN = 1000,
|
||
FILE_DOWNLINK_CYCLE_TIME = 1000,
|
||
FILE_DOWNLINK_FILE_QUEUE_DEPTH = 10,
|
||
HEALTH_WATCHDOG_CODE = 0x123,
|
||
COMM_PRIORITY = 100,
|
||
UPLINK_BUFFER_MANAGER_STORE_SIZE = 3000,
|
||
UPLINK_BUFFER_MANAGER_QUEUE_SIZE = 30,
|
||
UPLINK_BUFFER_MANAGER_ID = 200
|
||
};
|
||
|
||
// Ping entries are autocoded, however; this code is not properly exported. Thus, it is copied here.
|
||
Svc::Health::PingEntry pingEntries[] = {
|
||
{PingEntries::blockDrv::WARN, PingEntries::blockDrv::FATAL, "blockDrv"},
|
||
{PingEntries::tlmSend::WARN, PingEntries::tlmSend::FATAL, "chanTlm"},
|
||
{PingEntries::cmdDisp::WARN, PingEntries::cmdDisp::FATAL, "cmdDisp"},
|
||
{PingEntries::cmdSeq::WARN, PingEntries::cmdSeq::FATAL, "cmdSeq"},
|
||
{PingEntries::eventLogger::WARN, PingEntries::eventLogger::FATAL, "eventLogger"},
|
||
{PingEntries::fileDownlink::WARN, PingEntries::fileDownlink::FATAL, "fileDownlink"},
|
||
{PingEntries::fileManager::WARN, PingEntries::fileManager::FATAL, "fileManager"},
|
||
{PingEntries::fileUplink::WARN, PingEntries::fileUplink::FATAL, "fileUplink"},
|
||
{PingEntries::pingRcvr::WARN, PingEntries::pingRcvr::FATAL, "pingRcvr"},
|
||
{PingEntries::prmDb::WARN, PingEntries::prmDb::FATAL, "prmDb"},
|
||
{PingEntries::rateGroup1Comp::WARN, PingEntries::rateGroup1Comp::FATAL, "rateGroup1Comp"},
|
||
{PingEntries::rateGroup2Comp::WARN, PingEntries::rateGroup2Comp::FATAL, "rateGroup2Comp"},
|
||
{PingEntries::rateGroup3Comp::WARN, PingEntries::rateGroup3Comp::FATAL, "rateGroup3Comp"},
|
||
};
|
||
|
||
/**
|
||
* \brief configure/setup components in project-specific way
|
||
*
|
||
* This is a *helper* function which configures/sets up each component requiring project specific input. This includes
|
||
* allocating resources, passing-in arguments, etc. This function may be inlined into the topology setup function if
|
||
* desired, but is extracted here for clarity.
|
||
*/
|
||
void configureTopology() {
|
||
// Command sequencer needs to allocate memory to hold contents of command sequences
|
||
cmdSeq.allocateBuffer(0, mallocator, CMD_SEQ_BUFFER_SIZE);
|
||
|
||
// Rate group driver needs a divisor list
|
||
rateGroupDriverComp.configure(rateGroupDivisors, FW_NUM_ARRAY_ELEMENTS(rateGroupDivisors));
|
||
|
||
// Rate groups require context arrays. Empty for Reference example.
|
||
rateGroup1Comp.configure(rateGroup1Context, FW_NUM_ARRAY_ELEMENTS(rateGroup1Context));
|
||
rateGroup2Comp.configure(rateGroup2Context, FW_NUM_ARRAY_ELEMENTS(rateGroup2Context));
|
||
rateGroup3Comp.configure(rateGroup3Context, FW_NUM_ARRAY_ELEMENTS(rateGroup3Context));
|
||
|
||
// File downlink requires some project-derived properties.
|
||
fileDownlink.configure(FILE_DOWNLINK_TIMEOUT, FILE_DOWNLINK_COOLDOWN, FILE_DOWNLINK_CYCLE_TIME,
|
||
FILE_DOWNLINK_FILE_QUEUE_DEPTH);
|
||
|
||
// Parameter database is configured with a database file name, and that file must be initially read.
|
||
prmDb.configure("PrmDb.dat");
|
||
prmDb.readParamFile();
|
||
|
||
// Health is supplied a set of ping entires.
|
||
health.setPingEntries(pingEntries, FW_NUM_ARRAY_ELEMENTS(pingEntries), HEALTH_WATCHDOG_CODE);
|
||
|
||
// Buffer managers need a configured set of buckets and an allocator used to allocate memory for those buckets.
|
||
Svc::BufferManager::BufferBins upBuffMgrBins;
|
||
memset(&upBuffMgrBins, 0, sizeof(upBuffMgrBins));
|
||
upBuffMgrBins.bins[0].bufferSize = UPLINK_BUFFER_MANAGER_STORE_SIZE;
|
||
upBuffMgrBins.bins[0].numBuffers = UPLINK_BUFFER_MANAGER_QUEUE_SIZE;
|
||
fileUplinkBufferManager.setup(UPLINK_BUFFER_MANAGER_ID, 0, mallocator, upBuffMgrBins);
|
||
|
||
// Framer and Deframer components need to be passed a protocol handler
|
||
downlink.setup(framing);
|
||
uplink.setup(deframing);
|
||
|
||
// Note: Uncomment when using Svc:TlmPacketizer
|
||
//tlmSend.setPacketList(RefPacketsPkts, RefPacketsIgnore, 1);
|
||
}
|
||
|
||
// Public functions for use in main program are namespaced with deployment name Ref
|
||
namespace Ref {
|
||
void setupTopology(const TopologyState& state) {
|
||
// Autocoded initialization. Function provided by autocoder.
|
||
initComponents(state);
|
||
// Autocoded id setup. Function provided by autocoder.
|
||
setBaseIds();
|
||
// Autocoded connection wiring. Function provided by autocoder.
|
||
connectComponents();
|
||
// Autocoded command registration. Function provided by autocoder.
|
||
regCommands();
|
||
// Project-specific component configuration. Function provided above. May be inlined, if desired.
|
||
configureTopology();
|
||
// Autocoded parameter loading. Function provided by autocoder.
|
||
loadParameters();
|
||
// Autocoded task kick-off (active components). Function provided by autocoder.
|
||
startTasks(state);
|
||
// Initialize socket client communication if and only if there is a valid specification
|
||
if (state.hostname != nullptr && state.port != 0) {
|
||
Os::TaskString name("ReceiveTask");
|
||
// Uplink is configured for receive so a socket task is started
|
||
comm.configure(state.hostname, state.port);
|
||
comm.startSocketTask(name, true, COMM_PRIORITY, Default::STACK_SIZE);
|
||
}
|
||
}
|
||
|
||
// Variables used for cycle simulation
|
||
Os::Mutex cycleLock;
|
||
volatile bool cycleFlag = true;
|
||
|
||
void startSimulatedCycle(U32 milliseconds) {
|
||
cycleLock.lock();
|
||
bool cycling = cycleFlag;
|
||
cycleLock.unLock();
|
||
|
||
// Main loop
|
||
while (cycling) {
|
||
Ref::blockDrv.callIsr();
|
||
Os::Task::delay(milliseconds);
|
||
|
||
cycleLock.lock();
|
||
cycling = cycleFlag;
|
||
cycleLock.unLock();
|
||
}
|
||
}
|
||
|
||
void stopSimulatedCycle() {
|
||
cycleLock.lock();
|
||
cycleFlag = false;
|
||
cycleLock.unLock();
|
||
}
|
||
|
||
void teardownTopology(const TopologyState& state) {
|
||
// Autocoded (active component) task clean-up. Functions provided by topology autocoder.
|
||
stopTasks(state);
|
||
freeThreads(state);
|
||
|
||
// Other task clean-up.
|
||
comm.stopSocketTask();
|
||
(void)comm.joinSocketTask(nullptr);
|
||
|
||
// Resource deallocation
|
||
cmdSeq.deallocateBuffer(mallocator);
|
||
fileUplinkBufferManager.cleanup();
|
||
}
|
||
}; // namespace Ref
|