Guide to GIGA R1 Dual Cores
Learn how to access and control the M4 and M7 cores on the GIGA R1, and how to communicate between them using RPC.
The GIGA R1's STM32H747XI has two cores, the M4 and the M7. Each core can be programmed individually, with M7 acting as the main processor, and the M4 as a co-processor.
The M7 is referred to as the main processor due its superior hardware features, as well as it is required to run to boot the M4 core (you boot the M4 from within the M7).
These two cores can run applications in parallel, for example, running a servo motor on one core, and a display on another, without blocking each other. In a single core, such operations would slow down the program, resulting in lesser performance.
The M4 and M7 cores are programmed with separate sketches, using the same serial port. In the Arduino IDE, you can select the core you want to program, and then upload the sketch you want to run on that specific core.
Goals
In this guide you will discover:
- How to configure and program the M4/M7 cores and conventional approaches to do so.
- How to boot the M4 core.
- How to communicate between the cores via Remote Procedure Call (RPC).
- Using the RPC Library with MicroPython.
- Useful examples based on the dual core & RPC features.
- The
library API.RPC
Hardware & Software Needed
- GIGA R1 WiFi
- Arduino IDE
- Arduino GIGA Board Package installed.*
*For instructions on how to install the GIGA Board Package, follow the Getting Started with GIGA R1 guide.
M4 Support
The M4 processor can access most of the peripherals that the M7 can access, with some exceptions.
The M4 supports:
- I2C
- SPI
- UART
- CAN
- DAC
- ADC
- Bluetooth® Low Energy (via ArduinoBLE Library)
The M4 does not support:
- Wi-Fi®
- Serial communication*
- Arduino Cloud sketches.
*Serial Communication from the M4 can be enabled by setting up an RPC that allows the M4 & M7 cores to communicate. Using
(M4) and RPC.print()
(M7) helps achieve this. See RPC Serial Example.RPC.read()
Boot / Disable M4
The M4 core is by manufacturing default, disabled when booting the board. The M4 core can however be booted by using the
RPC.begin()
command, which includes the necessary functions to boot the M4 core. See the RPC.cpp source file for more details.Boot / Disable M7
The M7 is booted by default and there is currently no option to disable this core.
Peripheral Interference
When booting the M4, the M4 will execute the sketch that has been uploaded to its flash memory. It is a good idea to track what type of code you are running on the M4, as you may create interference between different peripherals. If you run simply a blank sketch on the M4, it should not create any interference.
An example of this is if you use the
CAN
library. If you are running a CAN application on the M7, you will disrupt it if you enable it on the M4. The dual core feature is not intended for using the same peripheral, bus etc.Tip: name your sketches with a
and _M4
suffix/prefix, and create an initialization sequence. E.g. blink the blue LED three times whenever the M4 boots up._M7
Pin Priority
As the M7 and M4 core share their pins, which one gets priority to the pin? It can be assumed that as the M7 is the more powerful core, it gets first access.
This is however not true as pin priority is random. If both cores tries to access the same pin (e.g.
D27
), it is simply random who gets access.When developing dual core applications, it is a good idea avoiding using the same pins & peripheral for many reasons.
Programming M4/M7
When programming the GIGA R1 WiFi's M7 and M4, we create a sketch for each core, and program them like we would program two individual Arduino boards. As only a single serial port is available, we need to specify which core we want to target.
Some essential things to consider when programming the cores are:
- You need to partition the memory, allocating flash memory to the M4 core.
- You need to select the target core, which is either Main Core or M4 Co-processor.
- The M4 has no serial communication enabled, here we need to use RPC (see RPC Serial example).
When writing multiple sketches, there are some things to consider to make your development experience easier:
- Name your sketches with either
or_M4
suffix or prefix. This will make it easier if the code is intended to be shared with others._M7
- Consider having a starting sequence (e.g. the blue LED blinking 3 times), whenever a core is initialized.
- Always include
on your M7 core sketch.RPC.begin()
Partitioning The Flash Memory
To allocate the flash memory for the M4, the flash memory can be partitioned. This is done by navigating to Tools > Flash Split in the IDE.
Note that the flash memory is the space where the application code (your sketch) is stored. It is not the RAM memory (which is significantly lower).
- 2MB M7 + M4 in SDRAM (default) - this option is the default configuration, which is for programming the M7 only. This allocates no flash memory to the M4.
- 1.5MB M7 + 0.5MB M4 - useful when larger amount of flash memory is required on the M7.
- 1MB M7 + 1MB M4 - useful when you need to balance the flash memory equally between the M4 and M7 cores.
It is required to use option 2 or 3 if you intend to program the M4 via the IDE, as the default option provides no memory allocation for the M4.
Target Core
To select the core you want to program, navigate to Tools > Target Core in the IDE.
Here you can choose between:
- Main Core - this is the M7 core, the main processor on the board.
- M4 Co-processor - this is the M4 core, the co-processor on the board.
Uploading
As both cores share the same serial port, choosing the Flash Split + Target Core is required so that the program is uploaded to the correct core.
Uploading is no different than to any other Arduino board: simply click the upload button and wait for it to finish.
Booting M4 Core
The M4 core does not boot by itself as it requires interaction from the M7 core. This boot function is built into the
RPC
library, and needs to be included in the sketch uploaded to the M7:1#include <RPC.h>2
3void setup() {4 RPC.begin(); //boots M45}6void loop(){7}
Once the M4 is booted from the M7, both cores will run in parallel, much like two Arduinos sharing the same board.
Writing Over Existing Sketch
Uploading new sketches works the same as a typical upload procedure. The new sketch will overwrite the current sketch running on the core you upload to.
Identifying the Current CPU Core
The
RPC.cpu_id()
method can be used to identify current CPU core running the sketch. This can be used to run different code paths based on the CPU core ID. For example, the following sketch blinks the blue LED if the current core is the Cortex-M7, and the Green LED if the current core is the Cortex-M4:1/*2GIGA R1 WiFi - Core identify sketch.3
4This simple sketch blinks an LED on boot.5You will need to upload it to both the M7 and M4 core.6
7It checks whether current CPU is M7 or M4, and blinks either 8the blue LED or the green LED, 10 times. 9
10As the M4 is booted when invoking RPC.begin() on the M7,11the M4 sketch will run as soon as the blink() function12finishes on the M7. 13*/14
15#include <RPC.h>16
17void setup() {18 pinMode(LEDB, OUTPUT);19 pinMode(LEDG, OUTPUT);20
21 if (RPC.cpu_id() == CM7_CPUID) {22 blink(LEDB, 100); //blink blue LED (M7 core)23 } else {24 blink(LEDG, 100); //blink green LED (M4 core)25 }26}27
28void loop() {29}30
31void blink(int led, int delaySeconds) {32 for (int i; i < 10; i++) {33 digitalWrite(led, LOW);34 delay(delaySeconds);35 digitalWrite(led, HIGH);36 delay(delaySeconds);37 }38 RPC.begin();39}
Remote Procedure Call (RPC)
RPC is a method that allows programs to make requests to programs located elsewhere. It is based on the client-server model (also referred to as caller/callee), where the client makes a request to the server.
An RPC is a synchronous operation, and while a request is being made from the caller to another system, the operation is suspended. On return of the results, the operation is resumed.
The server side then performs the subroutine on request, and suspends any other operation as well. After it sends the result to the client, it resumes its operation, while waiting for another request.
RPCs in the Arduino Environment
At the moment, only a limited amount of boards supports RPC, as in this context, it is designed to be a communication line between two cores. The GIGA R1 is one of them.
What makes this implementation possible is the
RPC
library (see API section), which utilises the rpclib C++ library as well as functions from the Stream class.The library makes it possible to set up either of the M4/M7 cores as a server/client, where remote calls can be made between them. This is done by "binding" a function to a name on the server side, and calling that function from the client side.
On the server side, it could look like this:
1//server side, for example M72int addFunction(int a, int b){ 3 return a + b;4}5
6RPC.bind("addFunction", addFunction);
On the client side, it could look like this:
1int x = 10;2int y = 10;3
4RPC.call("addFunction", x, y);
When
call()
is used, a request is sent, it is processed on the server side, and returned. The x
and y
variables are used as arguments, and the result returned should be 20 (10+10).Using the RPC Library with MicroPython
The
msgpackrpc
library provides the same functionality as the Arduino RPC library for MicroPython, i.e., it allows the binding of local functions or objects, starting the M4 core, and invoking remote calls from Python scripts. This library and its supporting features are enabled by default on all compatible Arduino boards, starting with MicroPython release v1.23, and require no external dependencies to use. However, there are a few restrictions to using the RPC library with MicroPython:- Arduino sketches can only run on the M4 core.
- SDRAM-based firmware is not supported[1].
- Flash-based firmware must use a 1.5MB M7 + 0.5MB M4 flash partitioning scheme.
The following sections introduce the
msgpackrpc
library API and some use cases in detail.1. Although the
library does support loading firmware images to SDRAM, the firmware currently generated for the M4 core with an SDRAM target does not work. This issue may be fixed in future releases.msgpackrpc
The msgpackrpc
Library
msgpackrpc
The
msgpackrpc
library is the RPC library's counterpart for MicroPython, and it provides the same functionality as the Arduino RPC library. The first steps to using the msgpackrpc
library, are importing the module and creating a MsgPackRPC
object:1import msgpackrpc2
3# Create a MsgPackRPC object.4rpc = msgpackrpc.MsgPackRPC()
The RPC object created above can then be used to bind Python callables, start the M4 core and invoke remote calls from MicroPython scripts.
Binding Python Functions, Callables and Objects
Any Python callable object (such as functions, bound methods, callable instances, etc.) can be invoked by the remote core. To allow the remote core to invoke a callable, it must be first bound to a name. The following example binds a function to the name
sub
:1def sub(a, b):2 return a - b3
4# Register a function to be called by the remote processor.5rpc.bind("sub", sub)
Note that the name a callable is bound to does not have to match its name.
Similarly, an object's bound method can also be bound to a name. For example:
1foo = Foo()2rpc.bind("sub", foo.add)
Both of those functions can be called in the same way from the Arduino sketch:
1int res = RPC.call("sub", 2, 1).as<int>();
Objects can also be bound to allow their methods to be called by the remote core. When an object is passed to
bind()
, all of its public methods (the ones that don't start with an _
) are bound to their respective qualified names. For example, the following code binds the methods of an object of class Foo
:1class Foo:2 def __init__(self):3 pass4
5 def add(self, a, b):6 return a + b7
8 def sub(self, a, b):9 return a - b10
11# Register an object of Foo12rpc.bind("foo1", Foo())
Now the object's methods can be invoked by the Arduino sketch using their fully qualified name, for example:
1int res1 = RPC.call("foo1.add", 1, 2).as<int>();2int res2 = RPC.call("foo1.sub", 2, 1).as<int>();
Starting the M4 Core From Micropython
The next step is starting the M4 core by calling
MsgPackRPC.start()
with the firmware entry point as an argument:1# Start the remote processor and wait for it to be ready to communicate.2rpc.start(firmware=0x08180000)
This function will start the remote core (the M4), boot it from the specified firmware address, and wait for the core to be ready to communicate before it returns. The default arguments passed to this function are compatible with the Arduino RPC library and do not need to be changed for the purposes of this tutorial. Note that the firmware address used is a flash address, where the M4 firmware starts, and it's the same one used for the flash split of
1.5MB M7 + 0.5MB M4
.Calling Remote Functions From Micropython
Once the M4 core is started, the
MsgPackRPC
object can be used to invoke its remote functions. Remote calls can be invoked synchronously, i.e., the call does not return until a response is received from the other side, or asynchronously. In this case, the call returns a Future
object that must be joined at some point to read back the call's return value.1# Perform a synchronous call, which blocks until it returns.2res = rpc.call("add", 1, 2)3
4# Perform an asynchronous call, which returns immediately with a Future object.5f1 = rpc.call_async("add", 1, 2)6
7# The Future object returned above, must be joined at some point to get the results.8print(f1.join())
That covered most of the
msgpackrpc
library's API and use cases. For a complete example, see the MicroPython RPC LED Example. For more examples and applications, see the msgpackrpc
repository.RPC Examples
In this section, you will find a series of examples that is based on the
RPC
library. RPC Serial
The
Serial.print()
command only works on the M7 core. In order to print values on the M4, we need to:- Use
on the M4. This will print the values to the RPC1 stream.RPC.println()
- Use
andRPC.available()
.RPC.read()
M4 Sketch:
1#include <RPC.h>2
3void setup() {4 RPC.begin();5}6
7void loop() {8 RPC.println("Printed from M4 core");9 delay(1000);10}
M7 Sketch:
1#include <RPC.h>2
3void setup() {4 Serial.begin(9600);5 RPC.begin();6}7
8void loop() {9 String buffer = "";10 while (RPC.available()) {11 buffer += (char)RPC.read(); // Fill the buffer with characters12 }13 if (buffer.length() > 0) {14 Serial.print(buffer);15 }16}
RPC Sensor
This example demonstrates how to request a sensor reading from one core to the other, using:
- M4 as a client.
- M7 as a server.
M4 Sketch:
1#include "Arduino.h"2#include "RPC.h"3
4using namespace rtos;5
6Thread sensorThread;7
8void setup() {9 RPC.begin();10 Serial.begin(115200);11
12 /*13 Starts a new thread that loops the requestReading() function14 */15 sensorThread.start(requestReading);16}17
18void loop() {19
20}21
22/*23This thread calls the sensorThread() function remotely24every second. Result is printed to the RPC1 stream.25*/26void requestReading() {27 while (true) {28 delay(1000);29 auto result = RPC.call("sensorRead").as<int>();30 RPC.println("Result is " + String(result));31 }32}
M7 Sketch:
1#include "Arduino.h"2#include "RPC.h"3
4void setup() {5 RPC.begin();6 Serial.begin(115200);7
8 //Bind the sensorRead() function on the M79 RPC.bind("sensorRead", sensorRead);10}11
12void loop() {13 // On M7, let's print everything that is received over the RPC1 stream interface14 // Buffer it, otherwise all characters will be interleaved by other prints15 String buffer = "";16 while (RPC.available()) {17 buffer += (char)RPC.read(); // Fill the buffer with characters18 }19 if (buffer.length() > 0) {20 Serial.print(buffer);21 }22}23
24/*25Function on the M7 that returns an analog reading (A0)26*/27int sensorRead() {28 int result = analogRead(A0);29 return result;30}
RPC Servo Motor
This example demonstrates how to request a servo motor on another core to move to a specific angle, using:
- M4 as a client.
- M7 as a server.
Each example is written as a single sketch intended to be uploaded to both cores.
M4 sketch:
1#include "Arduino.h"2#include "RPC.h"3
4using namespace rtos;5
6Thread servoThread;7
8void setup() {9 RPC.begin();10 Serial.begin(115200);11
12 /*13 Starts a new thread that loops the requestServoMove() function14 */15 servoThread.start(requestServoMove);16}17
18void loop() {19}20
21/*22This thread calls the servoMove() function remotely23every second, passing the angle variable (0-180).24*/25void requestServoMove() {26 while (true) {27 //Read a pot meter28 int rawAnalog = analogRead(A0);29
30 //Map value to 18031 int angle = map(rawAnalog, 0, 1023, 0, 180);32
33 delay(1000);34 auto result = RPC.call("servoMove", angle).as<int>();35 RPC.println("Servo angle is: " + String(result));36 }37}
M7 sketch:
1#include "Arduino.h"2#include "RPC.h"3#include <Servo.h>4
5Servo myservo;6
7void setup() {8 RPC.begin();9 myservo.attach(5); //attach servo to pin 510
11 Serial.begin(115200);12
13 //Bind the servoMove() function on the M714 RPC.bind("servoMove", servoMove);15}16
17void loop() {18 // On M7, let's print everything that is received over the RPC1 stream interface19 // Buffer it, otherwise all characters will be interleaved by other prints20 String buffer = "";21 while (RPC.available()) {22 buffer += (char)RPC.read(); // Fill the buffer with characters23 }24 if (buffer.length() > 0) {25 Serial.print(buffer);26 }27}28
29/*30Function on the M7 that returns an analog reading (A0)31*/32int servoMove(int angle) {33 myservo.write(angle);34 delay(10);35 return angle;36 /*37 After the operation is done, return angle to the client.38 The value passed to this function does not change, but this39 verifies it has been passed correctly.40 */41}
MicroPython RPC LED
This example demonstrates how to use MicroPython (running on the M7 core) to remotely control an LED from the M4 core.
M4 sketch:
1#include "RPC.h"2
3void led(bool on) {4 if (on) {5 digitalWrite(LEDG, LOW);6 } else {7 digitalWrite(LEDG, HIGH);8 }9}10
11void setup() {12 pinMode(LEDG, OUTPUT);13 RPC.bind("led", led);14 RPC.begin();15}16
17void loop() {18
19}
M7 Python script:
1import time2import msgpackrpc3
4# Create an RPC object5rpc = msgpackrpc.MsgPackRPC()6
7# Start the remote processor.8rpc.start(firmware=0x08180000)9
10# Toggle the LED every 500ms11while True:12 rpc.call("led", True)13 time.sleep_ms(500)14 rpc.call("led", False)15 time.sleep_ms(500)
RPC Library API
The
RPC
library is based on the rpclib C++ library which provides a client and server implementation. In addition, it provides a method for communication between the M4 and M7 cores. This library is included in the GIGA Board Package, so it is automatically installed with the Board Package. To use this library, you need to include
RPC.h
:1#include <RPC.h>
RPC.begin()
Initializes the library. This function also boots the M4 core.
Syntax
1RPC.begin()
Returns
on success.1
on failure.0
RPC.bind()
Used on the server side to bind a name to a function, and makes it possible for remotely calling it from another system.
Syntax
1RPC.bind("this_function", thisfunction)
Parameters
- name given for the function to be called from the client side."name_of_func"
- name of the function on the server side.name_of_func
Returns
- None.
RPC.call()
Used on the client side to call a function with optional parameters.
1RPC.call("this_function", int args)
Parameters
- the name of the function declared on the server side."name_of_func"
- arguments to be passed to the function.args
Returns
- Result of the function if arguments are passed.
RPC Serial API
The RPC Serial methods are also included in the
RPC
library, and uses methods from the Stream base class, and is similar to the Serial class.As the
Serial
class is only available on the M7 core, the M4 core uses RPC
library to print data, where the M7 can read the data and print it to a computer.RPC.println()
Prints data to a serial port. This is used on the M4 core to send data to the M7.
Syntax
1RPC.println(val);
Parameters
- The value to print. Can be any data type, but not multiple (e.g. string + integer in the same call).
Returns
- Number of bytes used. E.g. printing ("hello") returns 7. As hello (5) + new line (2) = 7.
RPC.available()
Get the number of available bytes to read from the M4.
Syntax
1RPC.available();
Parameters
- None.
Returns
- The number of bytes available to read.
if there is none.-1
RPC.read()
Reads the first available byte from the M4.
Syntax
1RPC.read();
Parameters
- None.
Returns
- The first available byte from the M4.
if there is none.-1
RPC.cpu_id()
Returns the current CPU ID, which can be one of two values:
CM7_CPUID
for the Cortex-M7 core, or CM4_CPUID
for the Cortex-M4 core.Syntax
1RPC.cpu_id();
Parameters
- None.
Returns
for the Cortex-M7 core.CM7_CPUID
for the Cortex-M4 core.CM4_CPUID
Suggest changes
The content on docs.arduino.cc is facilitated through a public GitHub repository. If you see anything wrong, you can edit this page here.
License
The Arduino documentation is licensed under the Creative Commons Attribution-Share Alike 4.0 license.