F´ Flight Software - C/C++ Documentation
devel
A framework for building embedded system applications to NASA flight quality standards.
|
Subtopologies are topologies for smaller chunks of behavior in F Prime. It allows for grouping bits of topology architecture that fit together, to then be imported into a base deployment's topology. The use case for this is seen when working with shareable components, specifically in the form of libraries.
There are two ways that subtopologies can be created in F Prime. The first (native) way to do so is by hand-writing subtopologies. This method is less complex, but more tedious and restrictive on the capabilities of subtopologies. This document covers hand-written subtopologies.
The second method of creating subtopologies is through the Subtopology Autocoder (also described in a later section of this document). This method is more complex, however expands upon the feature set and capabilities of subtopologies. A document that details the process for creating a subtopology with it is available at the repo for the tool.
Contents
A subtopology is a folder that contains a topology.cpp
file, which includes a topology.fpp
file, which is a unified topology file including your wiring, component instances, and init specifications. As opposed to having separate files for component instances and the topology itself, the single file encapsulates both. A topologydefs.hpp
defines a struct that is used when instantiating your subtopology. A CMakeLists.txt
file links your topology.cpp
file to the build for your project.
Thus, the required structure for your subtopology should be:
All files will be discussed in more detail in later sections of this guide. Additionally, note that there are optional files that can be included in your subtopology to extend its capability.
docs
folder to document your subtopology. Simple markdown files (sdd
for subtopology design document) work well in this case..fpp
files can also be included in your subtopology. For example, a config.fpp
may prove to be quite useful.intentionally-empty.cpp
is a necessary file to include. It will be discussed why this is included later.Note that with the latest release of
fprime-tools
, you can runfprime-util new --subtopology
to generate the subtopology structure.
[!NOTE] Remember, this document covers hand-written subtopologies. See a later section for information about the Subtopology Autocoder tool, the other way of creating subtopologies.
This section will provide an overview to the contents of the individual files that make up a subtopology. We will use the names of the required files structure as shown in the above section. However, note that the file names shown are not required to be duplicated within your own custom subtopologies.
For this section, it would be helpful to come up with an example scenario where the subtopology is used, such that it can be better understood. Let's imagine that we would like to create a component called RNG
that will write a random integer to some telemetry channel when it is hooked up to a 1Hz clock (rate group). The RNG
component will have an input port called run
, that will be the entry point from the clock (rate group). The RNG
component will be an active component, and is under the MyLibrary
namespace.
As mentioned, this is a unified .fpp
file. It not only includes the instances instance declarations for components that you'd like your subtopology to use, but also the definitions for the instances.
Based on our example, we may have the following instance declarations:
and thus the following instance definitions:
and also the following wiring:
Thus, your unified fpp file, and so MySubtopology.fpp
could look like:
"Phases" are a way to be able to write C++ based configuration functions on a component instance-basis. They aim to replace a topology.cpp/hpp pair of files with the single, unified fpp file. For subtopologies, we enforce the usage of phases to maintain consistency and simplicity in the subtopology.
In a standard cpp/hpp configuration model, your topology would be initialized using the following three functions:
In the phase pattern, these three (as well as others, described in detail here) are tied as "init specifiers" to fpp component instances. So, instead of configuring all components in one place, each instance defines its own init specification.
This looks like the following example, and can be done inline within your unified fpp file:
As aforementioned, we enforce the utilization of phases for subtopologies. Notably, not shown here is that phases can still take in the TopologyState state
variable. TopologyState
is that struct, which can include any variables you would like a developer to be able to modify. This provides dynamic-ness to your subtopology.
We're now going to interject information about MySubtopologyTopologyDefs.hpp
, because it is more relevant to the current topic.
In our example, we may want to allow the user to customize the initial seed for our random number generator:
This struct definition is included within MySubtopologyDefs.hpp
. In addition to this, we may notice that we need to include global definitions in our Defs.hpp
file; however depending on your subtopology, you may find this part optional. An example of how these definitions may be written is here:
Notice that we wrap PingEntries
in a namespace called GlobalDefs
. This is important, as it allows us to use the namespace GlobalDefs
across multiple topologies, and then link them all together in the main deployment. This will become more clear in the next section.
This brings us to a unique naming scheme for variable names for subtopologies. On the development-end of the subtopology, we see names like configureTopology
. However, on the build end when these subtopologies are folded into a bigger project, these functions are added to namespaces via their names. Inherently this makes sense, so that we don't confuse a function, or especially a component instance, with each other. So:
fpp | cpp syntax |
---|---|
MySubtopology.rateGroup | MySubtopology_rateGroup |
The last step is to include our CMake-specific files. This includes CMakeLists.txt
. There is a very repeatable structure, and inherently just notifies the compiler that our topology exists when we build a deployment.
In CMakeLists.txt
:
As you can see, we have reached the point where we need to use intentionally-empty.cpp
. Remember the syntax renaming from earlier, where MySubtopology -> rateGroup
becomes MySubtopology_rateGroup...
? Well, if we run fprime-util build
, the main deployment that uses our subtopology as well as our subtopology will build and create that syntax. In this case, we get a namespace error, where we try to use MySubtopology_rateGroup
in a place where it's "not defined". The empty file tells CMake that when our subtopology is built, return the empty file, so there is only a single place where the syntax is defined.
At this point, we're ready to integrate our subtopology into the topology of our main deployment (you can think of this being our deliverable, with MySubtopology
being a portion of it). We will assume, given our example, that you have the RNG component developed alongside the topology. Additionally, we assume that you have created an F Prime project and an associated deployment for it. Let's call this deployment "MainDeployment", and the project "MainProject".
First step is to ensure that our subtopology is linked to our project; within project.cmake
add:
Then, let's cd
into MainDeployment/Top/
. We have some files here we need to configure to use our subtopology. Before anything let's go to CMakeLists.txt
to include your own subtopology's CMake file:
Head over to MainDeplomentTopologyDefs.hpp
. We want to not only include our subtopology's definitions header, but also modify PingEntires
to use GlobalDefs::PingEntries
. At the end of namespace MainDeployment
, include:
Then modify the current PingEntries
namespace call to be surrounded by GlobalDefs:
Then, we want to tell our MainDeployment's topology to import and use our unified topology file from our subtopology. While we're here, we should also hook up the CycleOut
port of our main deployment's rate group driver to the CycleIn
port of our subtopology's rate group. Ensure that the rate group driver has an appropriate output port array size. We head over to the topology.fpp
file, and include:
At this point, we want to now call our subtopology's configure/start/teardown functions within the corresponding functions of our MainDeployment
topology. So, in MainDeploymentTopology.cpp
:
Lastly, since our RNG component has some telemetry, we need to include (or ignore) these channels within the Packets.xml
file in this folder. As with any other component that is added to a deployment, you use the same syntax with the name of the instance followed by the name of the telemetry channel. For example:
Now go ahead and run and build your deployment, and you should see that you have a built deployment that uses a subtopology.
As you may notice, the current implementation of subtopologies lacks in a few areas of simplicity and especially reusability. A couple of these issues are tabulated here:
st
within a main topology main
. For example, st
could define the topology for managing a single temperature sensor, but I would like to implement $n$ number of those sensors. At the moment, to do this one would need to duplicate the entire subtopology and make proper modifications to instances, the TopologyDefs file, and more.st
being the topology for managing a single temperature sensor. It may be the case that st
only implements the software behavior, as is reasonable: the developer of the subtopology probably cannot write hardware interface drivers for every platform possible. So, the user would provide the proper driver to st
, which is again reasonable. However, to accomplish this one would need to modify the contents of a subtopology, changing instance definitions and possibly the connection graphs as well. Such a task could become monstrous if, say st
now implements the hub pattern.Thus, an autocoder tool dubbed the "Subtopology AC Tool" has been developed to be able to provide features like instantiation, local components, and formal subtopology interfaces. The tool itself provides examples of the syntax required to use these features, as well as a design methodology and a worked example with diagrams using a similar context to the one in this document.
We recommend that subtopologies are built around the syntax of this tool, as it is on the road map to introduce the patterns in this tool into native FPP syntax.
This how-to guide has walked through the development of a subtopology. Deployments can include multiple different subtopologies, and thus this feature truly paves the way for making F Prime more accessible to quick prototyping.