Friday, 9 August 2013

Tutorial: Using the DSP on the BeagleBoard

Introduction


Hello! In this tutorial I'll be showing you how to utilise the BeagleBoard's TMS320C64x+ digital signal processor (DSP). I've recently started working on a project using the BeagleBoard and have been trying to use its DSP for a while now, but have found that most online resources are either hilariously outdated or contradict each other, making using the DSP rather arduous. Now that I've finally gotten it working, I've written this concise tutorial to help others that are struggling like I was.  

Requirements


+ An internet connection for downloading required files
+ A BeagleBoard; I used a BeagleBoard xM Rev. C
+ An x86/x86_64 host machine running GNU/Linux; I used Linux Mint 15 on my laptop
+ A micro-SD card reader compatible with your host machine
+ A micro-SD card, preferably with at least 4 GB capacity
~ (Optional) An ethernet connection to your BeagleBoard, allowing you to SSH into it over LAN

1 - Install Ubuntu 12.10 on the BeagleBoard


While you can probably get the DSP working with Angstrom, it's much much easier to use Ubuntu. Robert C Nelson maintains a number of Ubuntu images tailored to the BeagleBoard with a pre-configured DSP/Bridge-compatible kernel (this saves you building the kernel yourself and avoids a lot of the nitty-gritty). Download the latest version here, or to get the same version as me:

Host machine:
$ cd ~
$ mkdir dsp-stuff && cd dsp-stuff
$ wget http://rcn-ee.net/deb/rootfs/quantal/ubuntu-12.10-console-armhf-2013-07-22.tar.xz
$ tar xf ubuntu-12.10-console-armhf-2013-07-22.tar.xz
$ cd ubuntu-12.10-console-armhf-2013-07-22

Now you'll need to flash the Ubuntu image onto your micro-SD card. First, identify your card's device handle by inserting it into the card reader and running lsblk. USB card readers tend to show up as /dev/sdX whereas built-in card readers show up as /dev/mmcblkX. My card shows up as /dev/mmcblk0.

Once you've identified your card's device handle, run the setup_sdcard.sh script:

Host machine:
$ chmod +x ./setup_sdcard.sh
$ sudo ./setup_sdcard.sh --mmc /dev/mmcblk0 --uboot beagle_xm

There are two things to note here: first, you point the --mmc flag to the entire memory card, not just the first partition on the card (ie /dev/mmcblk0 as opposed to /dev/mmcblk0p1). Secondly, if you're using a normal BeagleBoard (not an xM) then change the --uboot flag to point to beagle instead of beagle_xm.

If everything went OK, the micro-SD card is now ready to be used in the BeagleBoard.

2 - Setup Ubuntu on the BeagleBoard


Login to the BeagleBoard (for example using serial communication) with username ubuntu and password temppwd.

First, update the system:

BeagleBoard:
$ sudo apt-get update
$ sudo apt-get upgrade

Change the root password:

BeagleBoard:
$ sudo -i
# passwd
# exit
# logout

Now log back into the BeagleBoard, directly into the root account using the password you just set. From here you can change the default username and password (replacing NEWNAME with your desired username):

BeagleBoard:
# passwd ubuntu
# usermod -l NEWNAME ubuntu

Now change the BeagleBoard's hostname (its name on your network). I chose to use 'bbxm':

BeagleBoard:
# echo bbxm > /etc/hostname

You'll also need to change your hosts file:

BeagleBoard:
# nano /etc/hosts

Change the 2nd line (127.0.1.1), replacing 'arm' with your chosen hostname. Since I chose to use 'bbxm', my /etc/hosts file looks like this:

127.0.0.1 localhost
127.0.1.1 bbxm

Now reboot the BeagleBoard:

BeagleBoard:
# reboot

While you can do everything over serial communication, or even plug a mouse/keyboard/monitor directly into the BeagleBoard, I find it's much cleaner and simpler to leave my board in a shoe box under the stairs with a wired ethernet connection to my router. This keeps my desk tidy and means I can SSH into it with my laptop from anywhere in the house.

If you want to be able to SSH into the BeagleBoard, log in (as your username or root, it doesn't matter) and find the board's local IP address:

BeagleBoard:
# ip addr

You're looking for the inet address of eth0. My BeagleBoard's local IP address is 192.168.0.18, as shown below:


To SSH into the BeagleBoard:

Host machine:
$ ssh 192.168.0.18

You may receive a warning about the server's key/certificate, just click Yes.

3 - Install DSP/Bridge drivers on the BeagleBoard


If your host Linux machine is 64-bit, you'll need to ensure you have a multiarch system. To check whether your machine is 32-bit or 64-bit, do the following:

Host machine:
$ dpkg --print-architecture

If the output is amd64 then you'll need the 32-bit libraries. You can check if you already have them by entering:

Host machine:
$ dpkg --print-foreign-architectures

If you see an i386 then you're already multiarch. If not, install the 32-bit libraries:

Host machine:
$ sudo apt-get install ia32-libs

You can now download and run the create_dsp_package.sh script from Robert C Nelson's github repository:

Host machine:
$ cd ~/dsp-stuff
$ wget https://raw.github.com/RobertCNelson/tools/master/x86/ti_omap/create_dsp_package.sh
$ chmod +x ./create_dsp_package.sh
$ sudo ./create_dsp_package.sh

If the script encounters a connection timeout when attempting to download a file from gforge.ti.com, you can download the required file from here instead. Place the file in your ~/dsp-stuff/dl folder and run the script again; it should work now.

The script will have created a DSP_Install_libs.tar.gz archive, which you should move over to the BeagleBoard:

Host machine:
$ scp ./DSP_Install_libs.tar.gz ash@192.168.0.18:~/DSP_Install_libs.tar.gz

Now SSH into the BeagleBoard and install DSP/Bridge:

BeagleBoard:
$ tar xf DSP_Install_libs.tar.gz
$ chmod +x ./install-DSP.sh
$ sudo ./install-DSP.sh

Next build the DSP-side drivers:

BeagleBoard:
$ sudo /boot/uboot/tools/update.sh
$ sudo /boot/uboot/tools/pkgs/ti-tidspbridge.sh

Start DSP/Bridge:

BeagleBoard:
$ sudo /etc/init.d/dsp_init start

And finally, check that DSP/Bridge is working:

BeagleBoard:
$ sudo dsp-test

If everything went OK, you should see the following:



4 - Install DSP for Dummies


DSP for Dummies (DFD) is a "framework designed to allow programmers to easily interface their programs to the DSP". It really does make using the DSP rather simple!

Before you can install DFD, you'll need some pre-requisite tools. First, install the ARM Linux cross compiler on your host machine:

Host machine:
$ sudo apt-get install gcc-arm-linux-gnueabi

Next, download the Texas Instruments C6000 Code Generation tools. You'll need to register for a free Texas Instruments account, and if you're outside of the US you'll need to fill out a short export form too. You can get the latest Linux version here.

I ran into issues with the installer crashing Cinnamon over and over again (I use Linux Mint 15). I found that by dropping into a TTY (Ctrl-Alt-F3 for example) and running the .bin directly from there instead worked just fine.

When asked for an installation directory, I chose /opt/dsp-tools, but you can install it wherever you want.

Next, you'll need the DLLcreate tool (copy the whole wget command in one go, it will work!):

Host machine:
$ sudo -i
# cd /opt
# wget -q "http://dev.omapzoom.org/?p=tidspbridge/userspace-dspbridge.git;a=blob_plain;f=source/dsp/bdsptools/bridge-tools.tar;hb=HEAD" -O - \
    | tar xf - --strip-components 3 --wildcards ti/dspbridge/dsp/doffbuild/bin/*
# exit

Now you're ready to install DFD. First, grab the source code:

Host machine:
$ cd ~/dsp-stuff
$ wget http://processors.wiki.ti.com/images/c/c3/Dsp-for-dummies.tar.gz
$ tar xf Dsp-for-dummies.tar.gz
$ cd dsp-for-dummies

Before you can compile the matrix multiplication example, you'll need to make a slight alteration to the Makefile:

Host machine:
$ nano Makefile

Ensure that Line 2 reads:

CROSS_COMPILE ?= arm-linux-gnueabi-

Also, if you installed the Texas Instruments C6000 Code Generation tools to a directory other than /opt/dsp-tools, you'll need to change Line 21 accordingly. For example, if you installed the tools to /home/ash/c6000, Line 21 should read:

DSP_TOOLS := /home/ash/c6000

You're now ready to compile!

5 - Test DFD


Compile the matrix multiplication example:

Host machine:
$ make clean
$ make

This will generate two files in the dsp-for-dummies directory, dummy and dummy.dll64P. The first of these is the GPP-side binary executable and can be placed anywhere on the BeagleBoard. The second file is the DSP-side binary which needs to be placed in the BeagleBoard's /lib/dsp directory.

SSH into the BeagleBoard and navigate to where you copied the dummy file:

BeagleBoard:
$ ./dummy 128 0 3



You should see lots of numbers be printed to the screen. If so, everything is working! :)
If not, see below.

6 - Troubleshooting


When you try to run the dummy example, you may see something like this:

-bash: ./dummy: No such file or directory

This is most likely due to a hardware/software-float ABI mismatch with the particular ARM Linux cross compiler you're using on your host machine. There are two solutions to this. The first is a bit hackish, and is to simply install the ARMEL libraries on the BeagleBoard:

BeagleBoard:
$ sudo apt-get install libc6-armel

The "better" solution is to explicitly use a hardware-float ARM Linux cross compiler on your host machine:

Host machine:
$ sudo apt-get gcc-arm-linux-gnueabihf

Note the hf at the end of gnueabi. You'll also have to modify ~/dsp-stuff/dsp-for-dummies/Makefile again so that Line 2 now reads:

CROSS_COMPILE ?= arm-linux-gnueabihf-

Again noting the hf at the end of gnueabi. Remake the example and copy it over to the BeagleBoard again; it should work now.

Closing Words


Hopefully after following this tutorial you have a working DSP/Bridge and DSP for Dummies installation on your BeagleBoard and can now successfully communicate between the GPP and DSP!

Feel free to leave comments, especially if you spot any errors :)