Skip to content

Latest commit

 

History

History
444 lines (367 loc) · 20.3 KB

README.md

File metadata and controls

444 lines (367 loc) · 20.3 KB

Learn embedded Linux

I have started to learn embedded Linux and driver development with the BeagleBone Black board. There I will write some usefull notes or commands about work with the BBB or about driver development.

Table of contents

Study plan

Preparation

  • Build set of tools for Cortex-A8 ARM
  • Build and flash U-Boot in eMMC memory on board
  • Build the Linux Kernel
  • Build root file system

Learning

  • Learn how to boot the Linux kernel
  • Learn what is device tree table
  • Learn how to write kernel modules

Bootlin's laboratory works about Linux kernel course

  • Make "Hello world" kernel module
  • Make I2C-driver for external I2C-device
  • Make driver avaliable in user-space through /dev/devname
  • Realize fops struct for i2c-device
  • Realize interrupt handler from the device

Write drivers for homemade soldered board

  • SPI OLED display based on ssd1306 controller
  • Rotary
  • Temperature and humidity sensor DHT11
  • RTC I2C based on DS3231 controller
  • Buttons/Leds
  • Ambient light I2C-sensor based on BH1750 controller
  • Barometric pressure I2C-sensor based on BMP180 controller
  • Ultrasonic ranging module HC-SR04

Few words about homemade development board

Front of development board
My Frankenstein
  1. Servomotor, that allows for precise control of angular position
  2. Mechanical rotary encoder that converts the angular position to digital output signals
  3. Ultrasonic ranging module
  4. Power LED
  5. SPI OLED display
  6. User GPIO LED
  7. Temperature and humidity sensor
  8. Seven-segment display
  9. User's button
  10. Interface to BBB
  11. CR2032 batary for RTC
  12. RTC
  13. Ambient light sensor
  14. Barometric pressure sensor
The BBB and UART-USB module
The BBB and UART-USB module

Few words about schematic

I don't know why I have decided to make this ugly board, but I think it doesn't matter for driver development and learning the Linux kernel :)

---in progress---

Toolchain

Toolchain is a collection of different tools set up to tightly work together. First of all, in embedded systems case, toolchain is needed to build a bootloader, a kernel and a root file system. After this actions we need to use the toolchain to build kernel modules, system drivers and user-space applications.

Every toolchain contains of:

  • Compiler. It is need to translate source code to assemler mnemonics.
  • Binutils. It is need to use as and ld. Assembler translates the mnemonics to binary codes. Linker makes one executable file.
  • C library and inludes of the Linux kernel(for development).

We need a cross-toolchain, because we need to compile code for machine different from the machine it runs on. Build machine in my case is x86-64 platform. Target machine (where code will be executed) in my case is ARM Cortex-A8 32 bit platform.

Toolchain choice depends on target platform:

  • Processor architecture (ARM / MIPS / x86_64 and so on)
  • Little endian or big endian
  • Is there hard-float arithmetic?
  • Should we use ABI with or without hf?

If a processor has hardware hard-float block, we need to pick toolchain which will generate code for hf-block. It will be more faster, than chip without hf, because without it compiler will use soft-float for double and float operations. The Beagle Bone Black has ARMv7 arch and hardware float block.

There are some ways to get needed toolchain:

  • Download already pre-built toolchain, for example, from Linaro.
  • To make the toolchain itself.

There are some methods for the last way. We can build all set of tools by hand with help this guide. It's very long and hard way. Or we can use some frameworks for creating Linux distributions for embedded devices. It is can be Buildroot, Yocto Project and other Linux build systems. There is comparing embedded Linux build systems document, which can help to pick needed system.

I have decided to download alredy pre-built toolchain:

wget -c https://releases.linaro.org/components/toolchain/binaries/7.3-2018.05/arm-linux-gnueabihf/gcc-linaro-7.3.1-2018.05-x86_64_arm-linux-gnueabihf.tar.xz
tar xf gcc-linaro-7.3.1-2018.05-x86_64_arm-linux-gnueabihf.tar.xz
*use path to this toolchain during building modules, drivers and user-space*

Bootloader

The main role of a bootloader is base initialisation of processor peripherals and loading a kernel of operating system to RAM memory.

What are there stages of loading OS?

OS is loaded with help several stages

Power-reset ---> ROM-code ---> Preloader or Secondary-Program-Loader ---> u-boot/barebox or Third-Program-Loader ---> kernel-start

ROM code

A lot of modern SoCs have on-chip ROM memory. Code into the ROM is flashed on factory and it will executed by the CPU from power-up. Immediately аfter power-up CPU has no access to the peripheral devices, such as RAM, storage and so on, because the CPU doesn't know something about its board. Only the next stages can know about board's chipset, because we itself have configured, built and flashed right code(uboot/device-tree) to flash/eMMC card. That's why the one can use only on-chip RAM memory. It is a SRAM usually.

This bootloader initializes a minimal amount of CPU and board hardware, then accesses the first partition of the SD or MMC card, which must be in FAT format, and loads a file called "MLO", and executes it. "MLO" is the second-stage bootloader.

If thare aren't MMC, ROM-code tries to load the preloader into SRAM with help this interfaces:

  • SPI flash card
  • Ethernet/USB/UART

In the end of this stage SPL-code is loaded into SRAM and starts to execute. If SoC-system has enough RAM memory, ROM code can load and TPL(uboot.bin) too.

SPL

SPL-code must configure DRAM-chip, using SoC DRAM controller, to load the TPL to it. This bootloader apparently also just reads the first partition of the SD card, and loads a file called "u-boot.bin", and executes it. "u-boot.bin" is the third-stage bootloader.

TPL: uboot or barebox

u-boot code loads OS kernel from MMC/SD to DRAM, flattened device tree and ramdisk.

Building uboot

git clone https://github.com/u-boot/u-boot.git uboot
cd uboot
git checkout v2019.10

We need to build the SPL and TPL. Other words we need to get two binary files:

  • MLO as the SPL
  • u-boot.bin as the TPL

To compile this code, we should configure the uboot sources with help make <config-file-for-your-board>. There are a lot of ready config files in the ./uboot/configs/. We can take ready file for a board or write own file. Config file must specify a device tree file for a board and define some constants. See Makefile which builds uboot for the BeagleBoneBlack board or type make uboot in the root of the repository.

After building you can find this files in the./out/uboot/ directory:

  • MLO is secodary program loader
  • u-boot is itself bootloader in ELF format for debugging
  • u-boot.map is table of symbols
  • u-boot.bin is itself bootloader with dtb ready for executing in a target

The Linux kernel

First of all we need to configure the kernel for our board. There are a lot of ready config files. We can find its into arch/$ARCH/configs/ directory.

Configuration and buildings

The kernel uses Kconfig system to configure whole system and then Kbuild system to build the one. You can learn the Kconfig system in the $(KERNEL_SRC_DIR)/Documentation/Kbuild. Also in the Linux source tree there is utilities which can read Kconfig files and show it in graphics menu. For example:

make ARCH=arm menuconfig

The utilities generate .config file in root of Linux sources. Then we can build kernel for target. Build system will use the .config file to build the kernel. There are a lot of configuration variables in Kconfig. If we don't want to make own config file from scratch, we can use ready default config. For example for armv7:

make ARCH=arm multi_v7_defconfig
make ARCH=arm CROSS_COMPILE=... all

After building into ./out/kernel/ directory this files will appeared:

  • Image is image of the kernel
  • zImage is image of the kernel in compressed state
  • am335x-boneblack.dtb - is Device Tree Blob for the BBB board
  • vmlinux is ELF file for debugger

Installing all images to the BBB

The BBB has a JTAG-connector on the board, that's why we can flash the uboot to RAM, run it and, with help uboot command line, copy its code to flash card. But I have no JTAG for armv7. I will use a SD-card and my work station to prepare the uboot for the BBB.

The AM335x chip has internal ROM memory, in which ROM-code was saved. This code can access to SD card during boot process. We just need to make the first part of a SD card as booting part and format it to FAT. Use this or this guides to format SD-card

Flash uboot to SD

We can use the fdisk console utility for format SD card:

1. Insert SD card to PC and check with help `dmesg` name of the device. For example /dev/sdb
2. sudo umount /dev/sdb
3. sudo fdisk /dev/sdb
4. p - print the partition table
5. d - delete a partition
6. o - create a new empty DOS partition table
7. Create first boot partition:
Command (m for help): n
Partition type
   p   primary (0 primary, 0 extended, 4 free)
   e   extended (container for logical partitions)
Select (default p): p
Partition number (1-4, default 1): 1
First sector (2048-30535679, default 2048): 
Last sector, +sectors or +size{K,M,G,T,P} (2048-30535679, default 30535679): 204800

Created a new partition 1 of type 'Linux' and of size 99 MiB.

8. a - toggle a bootable flag
9. t - change a partition type:
Command (m for help): t
Selected partition 1
Hex code (type L to list all codes): L

 0  Empty           24  NEC DOS         81  Minix / old Lin bf  Solaris        
 1  FAT12           27  Hidden NTFS Win 82  Linux swap / So c1  DRDOS/sec (FAT-
 2  XENIX root      39  Plan 9          83  Linux           c4  DRDOS/sec (FAT-
 3  XENIX usr       3c  PartitionMagic  84  OS/2 hidden or  c6  DRDOS/sec (FAT-
 4  FAT16 <32M      40  Venix 80286     85  Linux extended  c7  Syrinx         
 5  Extended        41  PPC PReP Boot   86  NTFS volume set da  Non-FS data    
 6  FAT16           42  SFS             87  NTFS volume set db  CP/M / CTOS / .
 7  HPFS/NTFS/exFAT 4d  QNX4.x          88  Linux plaintext de  Dell Utility   
 8  AIX             4e  QNX4.x 2nd part 8e  Linux LVM       df  BootIt         
 9  AIX bootable    4f  QNX4.x 3rd part 93  Amoeba          e1  DOS access     
 a  OS/2 Boot Manag 50  OnTrack DM      94  Amoeba BBT      e3  DOS R/O        
 b  W95 FAT32       51  OnTrack DM6 Aux 9f  BSD/OS          e4  SpeedStor      
 c  W95 FAT32 (LBA) 52  CP/M            a0  IBM Thinkpad hi ea  Rufus alignment
 e  W95 FAT16 (LBA) 53  OnTrack DM6 Aux a5  FreeBSD         eb  BeOS fs        
 f  W95 Ext d (LBA) 54  OnTrackDM6      a6  OpenBSD         ee  GPT            
10  OPUS            55  EZ-Drive        a7  NeXTSTEP        ef  EFI (FAT-12/16/
11  Hidden FAT12    56  Golden Bow      a8  Darwin UFS      f0  Linux/PA-RISC b
12  Compaq diagnost 5c  Priam Edisk     a9  NetBSD          f1  SpeedStor      
14  Hidden FAT16 <3 61  SpeedStor       ab  Darwin boot     f4  SpeedStor      
16  Hidden FAT16    63  GNU HURD or Sys af  HFS / HFS+      f2  DOS secondary  
17  Hidden HPFS/NTF 64  Novell Netware  b7  BSDI fs         fb  VMware VMFS    
18  AST SmartSleep  65  Novell Netware  b8  BSDI swap       fc  VMware VMKCORE 
1b  Hidden W95 FAT3 70  DiskSecure Mult bb  Boot Wizard hid fd  Linux RAID auto
1c  Hidden W95 FAT3 75  PC/IX           bc  Acronis FAT32 L fe  LANstep        
1e  Hidden W95 FAT1 80  Old Minix       be  Solaris boot    ff  BBT            
Hex code (type L to list all codes): c (or e)
Changed type of partition 'Linux' to W95 FAT32 (LBA).

10. Create second partition for future user-space file system (partition type - Linux)
11. w - write table to disk and exit

Then we need to make file systems in partitions and copy MLO and uboot.bin to boot partitoin (/dev/sdb1 in my case):

# boot partition
sudo umount /dev/sdb1
sudo mkfs.vfat -F 32 /dev/sdb1
sudo mkdir /mnt/tmp
sudo mount /dev/sdb1 /mnt/tmp
sudo cp ./out/uboot/MLO ./out/uboot/u-boot.bin /mnt/tmp
sudo umount /dev/sdb1

# ext4 data partition
sudo umount /dev/sdb2
sudo mkfs.ext4 /dev/sdb2

After this manipulations we need to disable the board, insert SD card, press uSD button on the board (that will be booting from SD, not MMC) and turn on power. After this we should see uboot booting log in uart console.

Setup NFS

The next step is to configure u-boot and workstation to let the board download files, such as the kernel image and Device Tree Binary (DTB), using the TFTP protocol through a network connection. The BBB and u-boot(see compiling config) support Ethernet over USB.

Setup TFTP

  • For u-boot:
=> setenv ipaddr 192.168.0.100
=> setenv serverip 192.168.0.1
=> setenv ethact usb_ether
=> setenv usbnet_devaddr f8:dc:7a:00:00:02
=> setenv usbnet_hostaddr f8:dc:7a:00:00:01
=> saveenv

Then reset board(need that uboot read params again) and go to the u-boot again.

We won’t be able to see the network interface corresponding to the Ethernet over USB device connection yet, because it’s only active when the board turns it on, from u-boot or from Linux. When this happens, the network interface name will be enx. Given the value we gave to usbnet_hostaddr, it will therefore be enxf8dc7a000001.

  • For workstation:
sudo apt install tftpd-hpa
sudo apt install tftp
  • Set network interface for eth-ver-usb:
nmcli con add type ethernet ifname enxf8dc7a000001 ip4 192.168.0.1/24

After this we can try to ping 192.168.0.1 from u-boot. It can help to ensure that ethernet over usb works from uboot right and that the interface on workstation has been uped right.

After installing we can put any files in the /var/lib/tftpboot on workstation and download this file from u-boot with help:

tftp 0x81000000(any SDRAM address) <file-name-in-/var/lib/tftpboot-directory>
  • Add support ethernet over USB to LInux kernel config(needs for NSF over USB) and rebuild the kernel:
CONFIG_USB_GADGET=y
CONFIG_USB_MUSB_HDRC=y
CONFIG_USB_MUSB_GADGET=y
CONFIG_USB_MUSB_DSPS=y
CONFIG_AM335X_PHY_USB=y
CONFIG_USB_ETH=y
CONFIG_ROOT_NFS=y

I have used make menuconfig(it setups dependencies) and this to enable eth-over-usb. We need to use eth-over-usb during booting, that's why we must set this configs only as static [=y], not as modules [=m].

Setup NFS server:

1. sudo apt-get install nfs-kernel-server nfs-common portmap
2. Add
`<pwd-to-rootfs> 192.168.0.100(rw,no_root_squash,no_subtree_check)` line to /etc/exports file as root
3. sudo /usr/sbin/exportfs -arv
4. sudo /etc/init.d/nfs-kernel-server restart

Boot the system

Before booting the kernel, we need to tell it which console to use and that the root filesystem should be mounted over NFS, by setting some kernel parameters.

=> setenv bootargs root=/dev/nfs rw ip=192.168.0.100:::::usb0 console=ttyO0,115200n8 g_ether.dev_addr=f8:dc:7a:00:00:02 g_ether.host_addr=f8:dc:7a:00:00:01 nfsroot=192.168.0.1:<pwd-to-rootfs>,nfsvers=3 init=/linuxrc
=> saveenv

Automate the boot process:

=> setenv bootcmd 'tftp 0x81000000 zImage; tftp 0x82000000 am335x-boneblack.dtb; bootz 0x81000000 - 0x82000000'
=> saveenv

Device Tree

My board uses some external GPIO pins, i2c and SPI buses. See to the schematics above. We must tell to the Linux kernel what pins we will use. We can describe our hardware with help device tree.

Pin control settings

AM335x chip - is a SoC. There is a control module into the one. See the am3358_reference_manual.pdf p.1211. This module responds for device control and status and for I/O multiplexing. Each configurable pin has its own configuration register for pullup/down control and for it assignment. Control unit is started with 0x44e10000 address in memory map and after 0x800 offset there are all pin registers. It is named like conf_<module>_<pin>. This registers have only seven least significant bits (0-6). If we want set some assignment to some pin, we should write this 7 bits to corresponding register-pin address.

Bits in pin-register:

      0 - fast slew rate
     /
bit 6
     \
      1 - slow slew rate

      0 - disable receiver for input
     /
bit 5
     \
      1 - enable receiver for input

      0 - pulldown selected
     /
bit 4
     \
      1 - pullup selected

      0 - enabled pullup/down resistor
     /
bit 3
     \
      1 - disable pullup/down resistor

bits 2-0 - pin mode.

For example if I want to setup user GPIO led on my board like ouput without pullip/down resistor (because there is external pulldown resistor and mosfet for enabling the led) I should write to 0x44e10000 + 0x844 register value (uint32_t)(0 | 1 << 3 | mode) = output pullup/down disable.

What is happening here?

My gpio led connected to P9.23 pin to the beaglebone. It pin corresponds to GPIO1_17. This corresponds conf_gpmc_a1 regtister. It has 0x844 offset from the start of control module. It's all. See am3358_briefly_datasheet.pdf for pin mode details (Pin Attributes table).

Linux pinmux system

There is one-register-per-pin type of device tree. It is case of AM335x chip. It means that for any processor pin there is special register to setup pin functions. We can enable this in config file with help CONFIG_PINCTRL_SINGLE=y. After this we need to describe every pin, which we want to use in device tree as a child of control module. In case of GPIO pins it is named pinmux controller. It has this device tree bindings. And see in my custom-dts file. The pin configuration nodes for pinctrl-single are specified as pinctrl register offset and value pairs using pinctrl-single,pins. Only the bits specified in pinctrl-single,function-mask are updated. For example, setting a pin for a device could be done with:

	pinctrl-single,pins = <0xdc 0x118>;

Where 0xdc is the offset from the pinctrl register base address for the device pinctrl register, and 0x118 contains the desired value of the pinctrl register.

Linux device drivers

--in progress--