Build an embedded application for a microcontroller
The source code for this guide can be found on GitHub
In this tutorial we’ll be targeting a Raspberry Pi Pico as the embedded device that our Swift application will run on. If you don’t physically have one, don’t worry! You can still run the application in an online emulator.
Installing Swift
If you don’t have Swift installed, install it first. Because Embedded Swift is experimental and only available in preview toolchains, make sure to install the “Development Snapshot” toolchain (main) instead of a release toolchain (6.0). If you’re using a macOS machine, you will need to make sure the installed toolchain is selected as active e.g. by exporting the TOOLCHAINS
environment variable:
$ export TOOLCHAINS=org.swift.59202405011a
To test that you have Swift installed, run swift --version
from your shell or terminal app. It should say “6.0-dev”, meaning you have a “Development Snapshot” toolchain.
Installing dependencies for embedded development
Install the Raspberry Pi Pico SDK, and the Arm Embedded Toolchain by following the Getting Started With Pico guide. Export three environment variables to match your setup and hardware:
$ export PICO_BOARD=pico
$ export PICO_SDK_PATH=... # location to your Pico SDK
$ export PICO_TOOLCHAIN_PATH=... # location to the Arm Embedded Toolchain
If you have the Wi-Fi enabled Pico W board instead of the regular Pico, note that you will need a slightly different setup described in the Pico W example project, and just specifying PICO_BOARD=pico_w
is not going to work.
Install CMake 3.29 or newer.
To test that you have all the neccessary parts installed, you can run the following commands in a terminal:
$ swift --version
Apple Swift version 6.0-dev (LLVM b66077aefd3be08, Swift 84d36181a762913)
$ cmake --version
cmake version 3.29.2
$ echo $PICO_BOARD
pico
$ ls $PICO_SDK_PATH
CMakeLists.txt README.md external/ pico_sdk_version.cmake tools/
CONTRIBUTING.md cmake/ lib/ src/
LICENSE.TXT docs/ pico_sdk_init.cmake test/
$ ls $PICO_TOOLCHAIN_PATH
13.2.Rel1-darwin-arm64-arm-none-eabi-manifest.txt include/ share/
arm-none-eabi/ lib/
bin/ libexec/
Building a “blinky” embedded app
The standard “Hello, World” in embedded development is a program that repeatedly blinks an LED. Let’s build one. The following setup can be also found in swift-embedded-examples, but we’re going to show below that all you need is just three files. Let’s create a new empty directory and prepare a simple structure for a CMake-based project that can be used on top the Pico SDK:
embedded-swift-tutorial
├── BridgingHeader.h
├── CMakeLists.txt
└── Main.swift
The Main.swift and BridgingHeader.h files can initially have the following basic content:
// Main.swift
let led = UInt32(PICO_DEFAULT_LED_PIN)
gpio_init(led)
gpio_set_dir(led, /*out*/true)
while true {
gpio_put(led, true)
sleep_ms(250)
gpio_put(led, false)
sleep_ms(250)
}
// BridgingHeader.h
#include "pico/stdlib.h"
To build on top of the Pico SDK’s CMake support, we need a bit more CMake logic in the CMakeLists.txt file:
# CMakeLists.txt
cmake_minimum_required(VERSION 3.29)
include($ENV{PICO_SDK_PATH}/external/pico_sdk_import.cmake)
set(CMAKE_Swift_COMPILATION_MODE wholemodule)
set(CMAKE_Swift_COMPILER_WORKS YES)
project(blinky)
pico_sdk_init()
enable_language(Swift)
add_executable(blinky Main.swift)
set_target_properties(blinky PROPERTIES LINKER_LANGUAGE CXX)
# Clear the default COMPILE_OPTIONS which include C specific compiler flags that the Swift compiler will not accept
# Instead, set those options to only apply when compiling C code.
set_target_properties(pico_standard_link PROPERTIES INTERFACE_COMPILE_OPTIONS "")
target_compile_options(pico_standard_link INTERFACE "$<$<COMPILE_LANGUAGE:C>:SHELL: -ffunction-sections -fdata-sections>")
set(SWIFT_INCLUDES)
foreach(dir ${CMAKE_C_IMPLICIT_INCLUDE_DIRECTORIES})
string(CONCAT SWIFT_INCLUDES ${SWIFT_INCLUDES} "-Xcc ")
string(CONCAT SWIFT_INCLUDES ${SWIFT_INCLUDES} "-I${dir} ")
endforeach()
target_compile_options(blinky PUBLIC "$<$<COMPILE_LANGUAGE:Swift>:SHELL:
-enable-experimental-feature Embedded
-target armv6m-none-none-eabi -Xcc -mfloat-abi=soft -Xcc -fshort-enums -Xfrontend -function-sections
-import-bridging-header ${CMAKE_CURRENT_LIST_DIR}/BridgingHeader.h
${SWIFT_INCLUDES}
>")
target_link_libraries(blinky pico_stdlib hardware_uart hardware_gpio)
pico_add_extra_outputs(blinky)
Now we’re ready to configure and build this firmware for the Pico. Run the following commands:
$ cmake -B build -G Ninja . # configure step
$ cmake --build build # build step
The build should succeed and produce the firmware in several formats (ELF, HEX, UF2), including some information dump files (DIS, ELF.MAP):
$ ls -al build/blinky*
-rwxr-xr-x 1 kuba staff 8.0K Jan 1 12:00 build/blinky.bin*
-rw-r--r-- 1 kuba staff 145K Jan 1 12:00 build/blinky.dis
-rwxr-xr-x 1 kuba staff 30K Jan 1 12:00 build/blinky.elf*
-rw-r--r-- 1 kuba staff 222K Jan 1 12:00 build/blinky.elf.map
-rw-r--r-- 1 kuba staff 23K Jan 1 12:00 build/blinky.hex
-rw-r--r-- 1 kuba staff 16K Jan 1 12:00 build/blinky.uf2
Running the firmware on a device
If you have a Raspberry Pi Pico, we’re now going to upload the built firmware and run it. If you don’t have one, skip to the next section and run the exact same firmware file in an emulator instead.
Connect the Raspberry Pi Pico board via a USB cable to your Mac, and make sure it’s in the USB Mass Storage firmware upload mode. This is normally the case if you have never uploaded any firmware yet – the Pico boots to the firmware upload mode if its memory doesn’t contain any valid firmware. Once a valid firmware is uploaded, the device will run that firmware after plugging in. To go back to the firmware upload mode, hold the BOOTSEL button while plugging in the board.
The Pico should then show up as a mounted volume in /Volumes (as RPI-RP2 in this case):
$ ls -al /Volumes
lrwxr-xr-x 1 root wheel 1B Jan 1 12:00 Macintosh HD@ -> /
drwx------ 1 kuba staff 16K Dec 31 1969 RPI-RP2/
Copy the UF2 file to this volume:
$ cp build/blinky.uf2 /Volumes/RPI-RP2
This will make the Pico automatically install the firmware, reboot itself, and run the firmware.
The green LED should now be blinking repeatedly. Hooray! Our first Embedded Swift program is running on an embedded device!
Running the firmware in an emulator
If you don’t have a physical Pico, or if you want to iterate quickly, Wokwi is free online emulator of various embedded microcontrollers, including a Raspberry Pi Pico. It executes the same firmware binary that you would normally upload to a physical device, and emulates one instruction at a time.
Open a new Pico project in Wokwi. Instead of using the code editor to write C code, press F1 and choose “Upload Firmware and Start Simulation”. Then select the UF2 file that our build process produced.
Once you upload the UF2 file to Wokwi, the simulation will start, and the LED should begin blinking repeatedly. Hooray! Our first Embedded Swift program is running in an emulator!
Bonus: Building a simple program for your host OS with Embedded Swift
While desktop operating systems like macOS and Linux are not the typical targets for Embedded Swift, you absolutely can build code for them using the Embedded Swift mode. This is useful for experimentation, trying out Embedded Swift, or to be able to iterate fast on an idea for some code that doesn’t really need the physical device to work.
The simplest program in Embedded Swift can just be a regular “Hello, World”:
// HelloEmbedded.swift
print("Hello, Embedded Swift 😊")
And building it into an executable can be done by calling the swiftc
compiler directly, but we’ll want to add flags to enable Embedded Swift, and also Whole Module Optimization.
$ swiftc HelloEmbedded.swift -o HelloEmbedded -enable-experimental-feature Embedded -wmo
This will produce a regular executable binary, but notice that it’s very small in size and that it also does not actually depend on the Swift runtime in the OS (all Embedded Swift binaries carry their runtime+stdlib dependencies within):
$ ls -al
-rwxr-xr-x 1 kuba staff 18K May 16 17:19 HelloEmbedded*
-rw-r--r-- 1 kuba staff 59B May 16 17:16 HelloEmbedded.swift
$ otool -L HelloEmbedded
HelloEmbedded:
/usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 1000.0.0)
Let’s run it:
$ ./HelloEmbedded
Hello, Embedded Swift 😊
Hooray! Our first host-side Embedded Swift program is working!
Where to go next
- The Embedded Swift Vision Document will give you an overview of the approaches and goals of Embedded Swift, and also what exactly is in the Embedded Swift language subset.
- The collection of Embedded Swift example projects on GitHub shows on which embedded devices can Swift work today, and the examples can also be used as templates for your own projects.
- The Embedded Swift User Manual describes how to use the Embedded Swift compilation mode and how to interact with the compiler.
- You might want to set up Swift editor integration via SourceKit-LSP to get features like indexing, autocomplete, jump-to-definition, show declarations/documentations, etc. to work in your favorite editor or IDE.
- The Swift forums are the best place to ask questions, give feedback or share your cool projects.
The source code for this guide can be found on GitHub