-
Notifications
You must be signed in to change notification settings - Fork 10
RPMsg
RPMsg is a communication protocol for heterogeneous inter-processor communication. These processors are often on the same SoC but isolated from each other via software (for instance, a board may boot up Linux with an arm processor while additional microcontrollers (remote processors) run an RTOS or some bare-metal application).
This page aims to establish an introduction to the inner workings of RPMsg and how to execute a basic test for it. Please note that this is by no means a definitive guide on any RPMsg implementation (eg Linux, OpenAMP, RPMsgLite, etc) and for a deeper understanding, it is encouraged to go through the respective project's dedicated documentation.
Considering one of ADI's DSP offerings as an example: the ADSP-SC598. This board contains 2 SHARC+ processing cores and 1 Arm Cortex A55, where the Arm processor is used to run Linux and is isolated, via software, from both the SHARC+ cores. All processors however share the memory provided to the system (L2 and L3).
In this scenario, we will aim to establish communication between a SHARC+ core and the cortex-A55. Both sides will require clients to handle RPMsg related activity. For the arm processor, since it is running Linux, the kernel includes support for RPMsg via a dedicated driver.
The latter, however, is currently assumed to be running bare-metal applications/RTOS only. Due to this, a lightweight, bare-metal library called rpmsg-lite is utilized. It implements a subset of features from Linux's RPMsg implementation to allow devices such as small MCUs to utilize this framework.
Although RPMsg-based communication is possible between any combination of processors (ARM-SHARC/SHARC-SHARC), the following is discussed with respect to ARM-SHARC interactions.
The communication takes place via shared memory, and hence, these processors share and know certain regions in the common memory space to search for the relevant data structures. Depending on which processor gets access to the memory first, it will assume the role of the "main" processor, while the other assumes the role of the "sub" processor. The main processor populates the MCAPI region with relevant data structures indicating its presence as well as determines the memory regions to be used for the inter-processor communication. Once the "sub" boots up, it discovers the MCAPI region has already been populated, parses the data structures and sets a flag in the region to indicate that it is ready to establish communication.
We now assume that the role of the main processor was assumed by the arm chip. Now, when the arm processor needs to communicate to the SHARC core, it will have to go through the following steps and establish a channel:
- Establish an RPMsg endpoint
- Announce its presence to the remote processor (sends the source address and endpoint name)
- Remote processor then establishes its own endpoint and a channel using the metadata obtained from the announcement
It should be noted that a channel is a software concept and hence, a single device can have multiple channels. Each channel is tied to endpoints, which are then tied to particular function callbacks.
When the ARM processor wants to communicate via the abovementioned channel, it populates the shared memory region associated with the channel and triggers and interrupt on the SHARC core via the TRU, notifying it about the change made. The SHARC+ core can now update its values and process the new changes accordingly. The inverse is also true when a SHARC core wants to communicate to the arm processor.
Before discussing the RPMsg interaction between a SHARC core running a baremetal application and an arm core running linux, it is essential that the boot process of the machine in focus is discussed since it also provides a couple of options to the user, allowing changing the "main" and "sub" processors.
When a board is powered up, the following takes place:
- The processors reads the bootrom and runs some pre-requisite tasks before each boot
- The processor then loads the first stage bootloader(SPL) from a fixed memory address
- SPL then passes the control to the second stage bootloader, or the full version of uboot in this case
After this point, there are two paths that the user can take:
- Let linux boot as normal
During initialization of the remoteproc driver, MCAPI region is populated, and the SHARC processor is initialized, forcing it to take the role of "sub"
- Load the firmware for the SHARC core and run it
This will enable the MCAPI region to be populated by the SHARC core and will allow it to assume the role of "main"
NOTE: In both cases, the SHARC core is initialized by sending a binary to it and running it.
Once the MCAPI interface has been populated and metadata exchanged, the RPMsg driver initializes similarly to the previous section. For security purposes, Linux creates RPMsg devices for each RPMsg channel. These are dynamically created.
Testing can be conducted via following methods:
- Include relevant packages in your Yocto distribution
- Build from scratch using the RPMsg guides listed in the rpmsg-utils and rpmsg-examples repositories.
Since this is an introductory guide, it will only document the former approach. For a more in-depth usage, it is encouraged for the user to refer to the abovementioned repositories.
- In your local.conf file, append the following:
INSTALL_IMAGE:append = " rpmsg-utils"
NOTE: Make sure
DISTRO_FEATURES:append = " linux_only_audio "
is present in your local.conf