Skip to main content

UART

In this chapter, we will learn how to use serial ports and terminal devices.

Sample program : Code.zip

1. Terminal Devices

Terminal devices, abbreviated as TTY (short for Teletypewriter), originally originated from teletypewriters and were used for interactive communication with computers. TTY initially referred to physical or virtual terminals connected to Unix systems. Over time, the concept of TTY has also extended to serial devices such as ttyn, ttySACn, and so on. In the Linux system, terminal devices are very well-supported, and they are typically accessed and controlled through special device files, enabling serial communication. These device files are located in the /dev directory, and they can be used to read and write data as well as perform various terminal control operations.

2. Serial Port Testing (Shell)

2.1 Pinout

LuckFox Pico and LuckFox Pico Mini A/B has three serial ports: UART2, UART3, and UART4, with UART2 being the debugging serial port.

LuckFox Pico Plus has four serial ports: UART2, UART3, UART4, and UART5, with UART2 being the debugging serial port.

LuckFox Pico Pro/Max has five serial ports: UART0, UART1, UART2,UART3 and UART4, with UART2 being the debugging serial port.

The serial ports UART3, UART4, and the debugging serial port are enabled by default.

  1. LuckFox Pico Diagram:

  2. LuckFox Pico Mini A/B Diagram:

  3. LuckFox Pico Plus Diagram:

  4. LuckFox Pico Pro/Max Diagram:

  5. LuckFox Pico Ultra/Ultra W Diagram:
    2.2 Device Directories

In the /dev directory, each UART device has its own directory. The directory names are ttyS followed by the serial port number, for example, /dev/ttyS3 represents UART3. You can check them using the following command:

# ls /dev/ttyS*
/dev/ttyS3 /dev/ttyS4

2.3 Configuring the Serial Port

  1. Use the stty tool to query its communication parameters:

    # stty -F /dev/ttyS3
    speed 9600 baud; line = 0;
    -brkint -imaxbel
  2. Modify the baud rate, where ispeed is the input speed, and ospeed is the output speed:

    # stty -F /dev/ttyS3 ispeed 115200 ospeed 115200
  3. Disable echo:

    stty -F /dev/ttyS3 -echo

2.4 Communicating with a Windows Host

  1. Connect one end of the serial module to your computer and the other end to pins 18 (GND), 19 (UART7_TX), and 20 (UART7_RX) on LuckFox Pico.

  2. Download and open MobaXterm, select the serial port, and set the baud rate (default is 9600; please adjust it according to your actual configuration).

  3. Execute the following commands on the terminal on your development board to write the strings "Hello" and "world!" to the terminal device file using the echo command:

    echo Hello > /dev/ttyS3
    echo "world !" > /dev/ttyS3
  4. The serial debugging assistant on Windows will receive the content:

3. Serial Communication (Python Program)

In the previous section, we demonstrated how to configure the serial port using commands and achieve serial communication by writing strings to the terminal device file. Next, we will implement serial communication using a Python program.

3.1 Using pyserial

  1. Complete Code

    This code uses the Python serial library to implement serial communication on serial port 3.

    import serial
    import time

    with serial.Serial(
    "/dev/ttyS3",
    baudrate=115200,
    bytesize=serial.EIGHTBITS,
    stopbits=serial.STOPBITS_ONE,
    parity=serial.PARITY_NONE,
    timeout=1,
    ) as uart3:
    uart3.write(b"Hello World!\n")
    buf = uart3.read(128)
    print("Raw data:\n", buf)
    data_strings = buf.decode("utf-8")
    print("Read {:d} bytes, printed as string:\n {:s}".format(len(buf), data_strings))
  2. Open Serial Port

    In this code, the serial.Serial object opens a serial connection on the device file /dev/ttyS3. Different parameters, such as baud rate (baudrate), data bits (bytesize), stop bits (stopbits), parity (parity), and timeout (timeout), can be configured to set the communication properties of the serial port. This code uses the with statement to ensure that the connection is closed correctly after use.

    with serial.Serial(
    "/dev/ttyS3",
    baudrate=115200,
    bytesize=serial.EIGHTBITS,
    stopbits=serial.STOPBITS_ONE,
    parity=serial.PARITY_NONE,
    timeout=1,
    ) as uart3:
  3. Send Data

    By calling the write method of the serial port object, this code sends a byte string (b"Hello World!\n") to the opened serial port.

    uart3.write(b"Hello World!\n")
  4. Receive Data

    In this code, the read method is used to read up to 128 bytes of data from the serial port, with a timeout of 1 second. The raw data read is printed in byte form, and then the decode("utf-8") method is used to decode the byte data into a UTF-8 encoded string. Finally, the number of bytes read and the decoded string are printed.

    buf = uart3.read(128)
    print("Raw data:\n", buf)
    data_strings = buf.decode("utf-8")
    print("Read {:d} bytes, printed as string:\n {:s}".format(len(buf), data_strings))

3.2 Using python-periphery

  1. Complete Code

    This code uses the Serial class from the periphery library to implement serial communication on serial port 3.

    from periphery import Serial

    try:
    serial = Serial(
    "/dev/ttyS3",
    baudrate=115200,
    databits=8,
    parity="none",
    stopbits=1,
    xonxoff=False,
    rtscts=False,
    )

    serial.write(b"Hello World!\n")
    buf = serial.read(128, 1)
    print("Raw data:\n", buf)
    data_strings = buf.decode("utf-8")
    print("Read {:d} bytes, printed as string:\n {:s}".format(len(buf), data_strings))

    finally:
    serial.close()
  2. Open Serial Port

    This code uses the Serial class from the periphery library to open a serial connection. The configuration of the serial port includes parameters such as the device file path (/dev/ttyS3), baud rate, data bits, parity bits, stop bits, etc.

    serial = Serial(
    "/dev/ttyS3",
    baudrate=115200,
    databits=8,
    parity="none",
    stopbits=1,
    xonxoff=False,
    rtscts=False,
    )
  3. Send Data

    This code, through the opened serial connection serial, sends a data packet containing "Hello World!\n" bytes to the serial port.

    serial.write(b"Hello World!\n")
  4. Receive Data

    In this code, the serial.read(128, 1) method is called to read up to 128 bytes of data from the serial port, with a timeout of 1 second. The raw data read is printed in byte form, and then the decode("utf-8") method is used to decode the byte data into a UTF-8 encoded string. Finally, the number of bytes read and the decoded string are printed.

    buf = serial.read(128, 1)
    print("Raw data:\n", buf)
    data_strings = buf.decode("utf-8")
    print("Read {:d} bytes, printed as string:\n {:s}".format(len(buf), data_strings))

3.3 Running the Program

  1. Use the vi tool in the terminal to create a Python file, paste the Python program, and save it.

    # vi uart.py
  2. Run the program.

    # python3 uart.py
  3. Experimental Observations.

    Connect the TX and RX of UART3, run the program:

4. Serial Communication (C Program)

In the previous section, we demonstrated how to achieve serial communication using Shell commands and Python programs. Additionally, we can use C library functions or system calls to read and write device files for the purpose of serial communication. Please note that to run programs on specific embedded systems, it is typically necessary to use cross-compilation tools to compile the code and generate executable files that can run on the target development board. Now, let's explore the specific implementation steps.

4.1 Complete Code

Serial communication can be achieved through the following program.

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <termios.h>
#include <unistd.h>

int main() {
int serial_port_num;
char serial_port[15];

printf("Select a serial port (3/4): ");
scanf("%d", &serial_port_num);

sprintf(serial_port,"/dev/ttyS%d",serial_port_num);
int serial_fd;

serial_fd = open(serial_port, O_RDWR | O_NOCTTY);
if (serial_fd == -1) {
perror("Failed to open serial port");
return 1;
}

struct termios tty;
memset(&tty, 0, sizeof(tty));

if (tcgetattr(serial_fd, &tty) != 0) {
perror("Error from tcgetattr");
return 1;
}

cfsetospeed(&tty, B9600);
cfsetispeed(&tty, B9600);

tty.c_cflag &= ~PARENB;
tty.c_cflag &= ~CSTOPB;
tty.c_cflag &= ~CSIZE;
tty.c_cflag |= CS8;

if (tcsetattr(serial_fd, TCSANOW, &tty) != 0) {
perror("Error from tcsetattr");
return 1;
}

char tx_buffer[] = "hello world!\n";
ssize_t bytes_written = write(serial_fd, tx_buffer, sizeof(tx_buffer));
if (bytes_written < 0) {
perror("Error writing to serial port");
close(serial_fd);
return 1;
}
printf("\rtx_buffer: \n %s ", tx_buffer);

char rx_buffer[256];
int bytes_read = read(serial_fd, rx_buffer, sizeof(rx_buffer));
if (bytes_read > 0) {
rx_buffer[bytes_read] = '\0';
printf("\rrx_buffer: \n %s ", rx_buffer);
} else {
printf("No data received.\n");
}

close(serial_fd);

return 0;
}

4.2 Opening the Serial Port

This code first allows the user to choose between communicating via UART3 or UART4, and then opens the corresponding serial port device file, saving its file descriptor in the serial_fd variable.

printf("Select a serial port (3/4): ");
scanf("%d", &serial_port_num);

sprintf(serial_port,"/dev/ttyS%d",serial_port_num);
int serial_fd;

serial_fd = open(serial_port, O_RDWR | O_NOCTTY);
if (serial_fd == -1) {
perror("Failed to open serial port");
return 1;
}

4.3 Configuring the Serial Port

In this part of the code, we define a termios structure named tty for configuring the parameters of serial communication. First, we initialize it to zero using memset. Then, we use the tcgetattr function to get the current attributes of the serial port and store them in the tty structure.

struct termios tty;
memset(&tty, 0, sizeof(tty));

if (tcgetattr(serial_fd, &tty) != 0) {
perror("Error from tcgetattr");
return 1;
}

In this part of the code, we set some parameters for serial communication. We use the cfsetospeed and cfsetispeed functions to set the baud rate to 9600, for output and input, respectively. We clear the PARENB flag to disable parity; clear the CSTOPB flag using the c_cflag attribute to use one stop bit; clear the data bits by clearing the CSIZE flag and set the data bits to 8 bits using the CS8 flag. Finally, we use the tcsetattr function to set the modified attributes as the current attributes of the serial port, using the TCSANOW flag to apply these settings immediately.

cfsetospeed(&tty, B9600);
cfsetispeed(&tty, B9600);

tty.c_cflag &= ~PARENB;
tty.c_cflag &= ~CSTOPB;
tty.c_cflag &= ~CSIZE;
tty.c_cflag |= CS8;

if (tcsetattr(serial_fd, TCSANOW, &tty) != 0) {
perror("Error from tcsetattr");
return 1;
}

4.4 Sending Data

This code achieves serial data transmission by writing the string "hello world!\n" to serial_fd. Successful transmission will print the data on the terminal.

char tx_buffer[] = "hello world!\n";
ssize_t bytes_written = write(serial_fd, tx_buffer, sizeof(tx_buffer));
if (bytes_written < 0) {
perror("Error writing to serial port");
close(serial_fd);
return 1;
}
printf("\rtx_buffer: \n %s ", tx_buffer);

4.5 Receiving Data

This code achieves serial data reception by reading from serial_fd. Successful reception will print the data on the terminal.

char rx_buffer[256];
int bytes_read = read(serial_fd, rx_buffer, sizeof(rx_buffer));
if (bytes_read > 0) {
rx_buffer[bytes_read] = '\0';
printf("\rrx_buffer: \n %s ", rx_buffer);
} else {
printf("No data received.\n");
}

4.6 Cross-Compilation

  1. Specify the Cross-Compilation Tool

    First, you need to add the path to the cross-compilation tool to the system's PATH environment variable so that you can use the cross-compilation tool from anywhere. You can add the following line to your shell configuration file (usually ~/.bashrc or ~/.bash_profile or ~/.zshrc, depending on your shell). Note that the path after PATH= should point to the directory where the cross-compilation tool is located.

    • gcc path

      <SDK Directory>/tools/linux/toolchain/arm-rockchip830-linux-uclibcgnueabihf/bin/arm-rockchip830-linux-uclibcgnueabihf-gcc
    • Open the shell configuration file.

      nano ~/.bashrc 
    • Add the path of the cross-compilation tool to the system's PATH environment variable. Replace <SDK Directory> with your own SDK path, such as /home/luckfox/luckfox-pico/.

      export PATH=<SDK Directory>/tools/linux/toolchain/arm-rockchip830-linux-uclibcgnueabihf/bin:$PATH
    • Reload the shell configuration file to apply the changes:

      source ~/.bashrc  
  2. Compile the Program Using the Cross-Compilation Tool

    arm-rockchip830-linux-uclibcgnueabihf-gcc uart.c -o uart
  3. After successful cross-compilation, an executable file that can run on the development board will be generated in the current directory.

    # ls
    uart uart.c

4.7 Running the Program

  1. File Transfer

    First, transfer the uart program from the virtual machine to Windows, and then transfer it to the development board via TFTP or ADB. Here are the steps to transfer the file from Windows to the development board using ADB:

    adb push path_to_file destination_on_development_board

    eg: (Transferring the `uart` file from the current directory to the root directory of the development board)
    adb push uart /
  2. Running the Program

    Modify the permissions of the uart file and then run the program:

    # chmod 777 uart
    # ./uart
  3. Experimental Observations

    Connect the TX and RX of UART3, run the program, and select UART3 for communication:

5. Modifying Device Tree

In the SDK's device tree files, UART3 and UART4 interfaces are enabled by default and can be used directly. If you want to use the UART5 interface of the LuckFox Pico Plus, you can also modify the device tree as described below.

5.1 Modifying Device Tree Files

  1. Device Tree File Paths

    • Development Board Configuration File

      In the <SDK directory>/project/cfg/BoardConfig_IPC/ directory, there are configuration files for different models of development boards. These files primarily include configuration parameters for various Luckfox Pico board models, covering aspects such as target architecture, boot medium, Uboot and kernel configurations, and partition settings. The SDK directory structure is as follows:

      ├── build.sh -> project/build.sh ---- SDK compilation script
      ├── media --------------------------- Multimedia encoding, ISP, and related algorithms (can be compiled independently in the SDK)
      ├── sysdrv -------------------------- U-Boot, kernel, rootfs directory (can be compiled independently in the SDK)
      ├── project ------------------------- Reference applications, compilation configurations, and script directories
      │ ├── cfg
      │ ├── BoardConfig_IPC
      │ ├── BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico-IPC.mk
      │ ├── BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico_Mini_A-IPC.mk
      │ ├── BoardConfig-SPI_NAND-NONE-RV1103_Luckfox_Pico_Mini_B-IPC.mk
      │ ├── BoardConfig-SPI_NAND-NONE-RV1103_Luckfox_Pico_Plus-IPC.mk
      │ ├── BoardConfig-SPI_NAND-NONE-RV1106_Luckfox_Pico_Pro_Max-IPC.mk
      │ └── ...
      ├── output -------------------------- Directory for storing SDK compiled image files
      ├── docs ---------------------------- SDK documentation directory
      └── tools --------------------------- Image burning packaging tools and burning tools

      Among them, RK_KERNEL_DTS specifies the Device Tree Source (DTS) file for the kernel. Taking Luckfox Pico as an example, by opening the BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico-IPC.mk file, we can see that the file pointed to by RK_KERNEL_DTS is rv1103g-luckfox-pico.dts

    • Device Tree File Path

      Based on RK_KERNEL_DTS, the device tree file path for Luckfox Pico can be determined as follows:

      <SDK directory>/sysdrv/source/kernel/arch/arm/boot/dts/rv1103g-luckfox-pico.dts
  2. Enabling UART5

    The following is an example that demonstrates how to enable UART5 in the device tree.
    image

5.2 Compiling the Kernel

  1. Compile by selecting branches for LuckFox Pico, LuckFox Pico Mini A, LuckFox Pico Mini B, LuckFox Pico Plus, and LuckFox Pico Pro/Max.

    luckfox@luckfox:~/luckfox-pico$ ./build.sh lunch
    ls: cannot access 'BoardConfig*.mk': No such file or directory

    You're building on Linux
    Lunch menu...pick a combo:

    BoardConfig-*.mk naming rules:
    BoardConfig-"启动介质"-"电源方案"-"硬件版本"-"应用场景".mk
    BoardConfig-"boot medium"-"power solution"-"hardware version"-"applicaton".mk

    ----------------------------------------------------------------
    0. BoardConfig_IPC/BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico-IPC.mk
    boot medium(启动介质): EMMC
    power solution(电源方案): NONE
    hardware version(硬件版本): RV1103_Luckfox_Pico
    applicaton(应用场景): IPC
    ----------------------------------------------------------------

    ----------------------------------------------------------------
    1. BoardConfig_IPC/BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico_Mini_A-IPC.mk
    boot medium(启动介质): EMMC
    power solution(电源方案): NONE
    hardware version(硬件版本): RV1103_Luckfox_Pico_Mini_A
    applicaton(应用场景): IPC
    ----------------------------------------------------------------

    ----------------------------------------------------------------
    2. BoardConfig_IPC/BoardConfig-SPI_NAND-NONE-RV1103_Luckfox_Pico_Mini_B-IPC.mk
    boot medium(启动介质): SPI_NAND
    power solution(电源方案): NONE
    hardware version(硬件版本): RV1103_Luckfox_Pico_Mini_B
    applicaton(应用场景): IPC
    ----------------------------------------------------------------

    ----------------------------------------------------------------
    3. BoardConfig_IPC/BoardConfig-SPI_NAND-NONE-RV1103_Luckfox_Pico_Plus-IPC.mk
    boot medium(启动介质): SPI_NAND
    power solution(电源方案): NONE
    hardware version(硬件版本): RV1103_Luckfox_Pico_Plus
    applicaton(应用场景): IPC
    ----------------------------------------------------------------

    ----------------------------------------------------------------
    4. BoardConfig_IPC/BoardConfig-SPI_NAND-NONE-RV1106_Luckfox_Pico_Pro_Max-IPC.mk
    boot medium(启动介质): SPI_NAND
    power solution(电源方案): NONE
    hardware version(硬件版本): RV1106_Luckfox_Pico_Pro_Max
    applicaton(应用场景): IPC
    ----------------------------------------------------------------

    Which would you like? [0]: 0
    [build.sh:info] switching to board: /home/luckfox/luckfox-pico/project/cfg/BoardConfig_IPC/BoardConfig-EMMC-NONE-RV1103_Luckfox_Pico-IPC.mk
    [build.sh:info] Running build_select_board succeeded.
  2. Recompile the kernel.

    luckfox@luckfox:~/Luckfox-Pico/luckfox-pico$ ./build.sh kernel

5.3 Flashing the Firmware Again

  1. After the kernel compilation is successful, the files can be found in the <SDK directory>/output/image directory.
    image
  2. Replace the boot.image and env.txt files in the original firmware.
    image
  3. Modify the corresponding partitions, select DownloadBin, env, and boot, then click on download.
    image