At a Glance
Table of Contents | ||
---|---|---|
|
...
In this module you'll be making a simple hello world application. You can either run this application on your virtual machine, or on a micro-controller. Running in a micro0controller requires hardware.
Prerequisites
- Our development environment
- some basic knowledge of git
...
Code Block |
---|
cd ~/shared/firmware_xiv
# Create a new branch off of master
git checkout master
git checkout -b wip_getting_started |
...
By convention, we use lower snake case for most of our naming schemes. For example, these documents were created under elec_281_add_basic_docs
. This lets us keep our work organized. See our Git Workflow for more information.
Hardware Requirements
- STM32F072B Discovery board + mini-USB cable
- UART to USB adapter
Info |
---|
If you aren't sure where to get a Discovery board or UART adapter, ask a lead! We have a few in the bay. Otherwise, feel free to just skip the parts that involve UART or STM32. |
Hardware (Optional)
For now, we'll be using STM32 Discovery boards. If you've heard of an Arduino, the Discovery boards are very similar in intention. They're just plug-and-play, no setup required.
To use a Discovery board, just connect the mini-USB cable to the header marked "USB ST-LINK" on the Discovery board and plug the other end into your computer.
For the UART to USB adapter, plug each wire of the adapter into its corresponding pin on the Discovery board.
...
New Project - Hello World
Creating a New Project
While being at the firmware
directory, run make new the following command:
Code Block |
---|
# Initialize the directory structure for a new project named "hello_world" # See projects/README.md for more information on how our projects work make new PROJECT=hello_world |
...
You'll notice that make new
also created a rules.mk
. This file is what identifies the folder hello_world
as a project to our build system. Please refer to the projects readme for more information.
...
Now, it's time to build and run the project! First, let's try it on x86.
Code Block |
---|
# Build and run the project on x86 make run PROJECT=hello_world PLATFORM=x86 |
Neat! Hopefully, you saw "Hello World" appear in your terminal. Now, if you have the hardware set up, let's try it on STM32!
Running on STM32
...
Warning |
---|
This will only work if you have the hardware set up properly and you're in the vagrant box! If it isn't working, be sure to ask a lead for help! |
In a new terminal (or tmux pane), open a serial terminal:
Code Block |
---|
# If no device is found, try /dev/tty[tab] or /dev/serial/[tab] where [tab] represents pressing the tab key
minicom -D /dev/seria |
(Note that the minicom baud rate should be set to 115200)
Back to the old terminal:
Code Block |
---|
# Build and run the project on STM32F0xx (discovery board)
# PLATFORM=stm32f0xx is implied
# PROBE=cmsis is normally implied, override for Discovery boards
make program PROJECT=hello_world PROBE=stlink-v2 |
If you look at minicom, you should see "Hello World" appear. Try changing the text!
Building for STM32 - Hello World
If you don't have access to hardware, no problem! You can still build the project for STM32. It should compile successfully.
Code Block |
---|
# Just build the project for STM32
make build PROJECT=hello_world |
As you can see, our build system and HAL (hardware abstraction layer) allow us to build applications that can run natively on mutliple platforms. In this case, LOG_DEBUG
is targeted to stdout
on x86 (standard output) and UART1
on STM32. This makes testing a lot easier, and allows us to develop application code even without access to the hardware.
Clean Code - Hello World
With your code workring, we want to make sure your code matches our Coding Standards. We use two tools for that:
clang-format
: Reformats code according to our specified style.cpplint
andpylint
: Parses code to find discrepancies between it and our style guide. We actually use our own fork.
Note that these are both just tools, and should not be relied on to be perfect. We have a few rules that would be very difficult to parse, so please try your best to follow our style guide.
Code Block |
---|
# Runs clang-format
make format
# Runs lint.py (cpplint) and pylint
make lint |
Committing Changes - Git
Now that your code (hopefully) meets our standards, let's save your work. You should commit your changes relatively often. The goal is to have each commit represent a cohesive chunk of work. Basically, if you hit a point where you think you might want to come back to your work later, you should probably commit.
Code Block |
---|
# Stage all changes
git add .
# Commit staged changes to the local repo
git commit -m "WIP: Adds working hello world project" |
This time, we've used the -m
argument to add a message directly on the command line. Alternatively, you can use git commit
(no -m
). This should result in a temporary COMMIT_EDITMSG
being opened in your editor of choice. This method is recommended for multiline commit messages.
Notice how the commit message followed the format "WIP: ...
". Similarly to branch naming, we follow a convention of "ELEC-[ticket number]: ...
" for our commit messages. For example, "ELEC-281: Added part 1 of getting started guides
".
Using a version control system (VCS) such as git allows you to keep track of changes over time and revert back if necessary. Feature branches allow you to organize your work and keep different projects separate from each other.
Since git is a distributed VCS, you have a local copy of the repository that needs to be synced with the remote server (GitHub). This means you can commit and make branches without internet access. Normally, we recommend you push your changes to GitHub as often as possible (git push
), but since this is a tutorial, we won't bother with that.
Troubleshooting/FAQ
Q: Why can't I see output from the STM32 Discovery board?
A: Make sure you have the FTDI cable connected properly! By default, printf does not do anything on the STM32 (After all, where would it go?). We've decided to redirect it to UART, so we need the FTDI cable to convert it to something your computer can understand. Note that both the Discovery board and FTDI cable need to be connected to your computer. PB6 should be connected to the yellow wire, and PB7 should be connected to the orange wire.
Summary
We covered:
...
To run on STM32, you need the following pieces of hardware:
Connect the Controller Board to you computer according to the picture above. Be sure to short the power lines on the SWD adapter, that ensures that your controller board is getting power from your computer.
Depending on your operating system, you'll need to do different things:
- Mac OS: Before Catalina
- Windows & Mac OS: Catalina or newer
MacOS (Before Catalina)
Note: If your Mac's Operating System version is older than Catalina, you'll need to do this part. Otherwise, skip to the other section! If you don't know how to check your OS's version, google it! (come on really?)
On older versions of MacOS for some reason our programmer doesn't get passed into the virtualbox
image. To solve that problem, we wrote a script that installs the toolchain on your Mac and SSH's into your Mac every time you want to program a Micro-Controller.
- Run the
macos.sh
script on your Mac.
Code Block |
---|
cd ~/box # If box is not in your home directory, delete it and run through module 1 again.
./macos.sh # This installs a whole bunch of stuff |
Now let's test to see if you can program the controller board with your laptop. Go on vagrant and flash the example
project:
Code Block |
---|
# ssh into your virtual machine
cd ~/box
vagrant ssh
# go to the firmware repo
cd shared/firmware_xiv/
# program the controller board with the "example" project
make program PROJECT=example |
Now the LEDs on the controller board must blink, cute right? If they don't or if you have problems, ask other firmware people or Arshan Khanifar.
Now let's run the hello world
program that you just wrote:
Code Block |
---|
make program PROJECT=hello_world |
Nothing happens, right? You'll notice that even the LED's aren't blinking anymore. This is because LOG_DEBUG
by default prints to the screen for Linux, but for STM32 we have to tell it where to go, since STM32's don't really have a display on them. We've designed our system so that the output of printf
is redirected to the UART
interface on our boards. To be able to read the printf
message, open another terminal window on your Mac terminal, and list out the devices, and filter them by usb
.
Code Block |
---|
~ ls /dev | grep usbcu.usbmodem143102
tty.usbmodem143102 |
You can see two devices cu.usbmodem143102
and tty.usbmodem143102
. That's because in Unix operating systems serial devices appear on two different names: cu
and tty
. The reason is beyond this tutorial's scope but you can read here for more info. The exact numbers also may be different on your computer. To listen on the serial port, use minicom
:
Code Block |
---|
minicom -D /dev/tty.usbmodem143102
# trick: type the following, hit tab, then enter
minicom /dev/serial/by-id/ |
You should see something like this:
Code Block |
---|
Welcome to minicom 2.7.1
OPTIONS:
Compiled on Sep 18 2017, 15:01:35.
Port /dev/tty.usbmodem143102, 19:59:33
Press Meta-Z for help on special keys |
You're now listening on that port! You don't see "Hello World!" just yet because when you programmed your hello_world
program, it simply printed "Hello World!" and exited. So you need to run it again to see it on this window. Now go back to your first terminal window that was in vagrant
and run make program
again:
Code Block |
---|
make program PROJECT=hello_world |
Now you should see "Hello World!" appear on your minicom
window! Yay!
Windows or MacOS (Catalina or newer)
Get in vagrant:
Code Block |
---|
vagrant ssh |
Now let's test to see if you can program the controller board with your laptop. Go ahead and flash the example
project:
Code Block |
---|
cd shared/firmware_xiv/
# program the controller board with the "example" project
make program PROJECT=example |
Now the LEDs on the controller board must blink, cute right? If they don't or if you have problems, ask other firmware people or Arshan Khanifar.
Now open another Git bash Window and ssh into the box, then list out the devices to see what the device name for the programmer is:
Code Block |
---|
~ ls /dev | grep CMSIS
tty.CMSIS-DAP |
It should return tty.CMSIS-DAP
or something like that. Now run minicom
with that device path to listen on the UART
interface:
Code Block |
---|
minicom -D /dev/tty.CMSIS-DAP # --> this should be what the last command showed you!!!! |
So now, you should see something like so:
Code Block |
---|
Welcome to minicom 2.7.1
OPTIONS:
Compiled on Sep 18 2017, 15:01:35.
Port /dev/tty.usbmodem143102, 19:59:33
Press Meta-Z for help on special keys |
Now your computer is listening on the UART port for all of the messages that get out of your Microcontroller Board! YAY! Now open another Git bash window, ssh into vagrant
, cd into the firmware
directory, and then run the application you just created: hello_world
Code Block |
---|
make program PROJECT=hello_world |
YEET! now you should see "Hello World!" on the first terminal window. Talk to other firmware people or Arshan Khanifar if this doesn't work.