Versions Compared

Key

  • This line was added.
  • This line was removed.
  • Formatting was changed.

At a Glance

Table of Contents
excludeAt a Glance

...

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

...

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

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 and pylint: 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:

Image Added

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.