Serial Communication via Software

Projects

Continuing from the last project (7-segment display device driver), I wanted to expand the project to use hardware interrupts. So, this project was aimed towards implementing a software approach to serial communication that is able to handle both transmission and receival (TX/RX).

There’s a lot of overlap in the code for the transmission part, but in summary, instead of having a workqueue in the middle, the hrtimer directly takes out the bytes from the buffer and bitbangs it to the other end. The same goes for the receiving side as the hrtimer that handles reading in the bytes just puts them directly into the buffer. The main reason for the simplication of the code was probably due to the assumption that there will only be ONE instance of the hardware. Therefore, a lot of race condition handling was removed.

The software serial device driver is largely divided into two parts: TX (transmission) and RX (receival). Each part is also divided into two parts: upper half and lower half. The upper half is what the user interacts with. It handles file operations like open, read, write, and close. There’s still some race condition handling as multiple users may write to the same device at the same time. The lower half handles the interaction with the hardware. It does bitbanging and it also handles reading in the bytes using hardware interrupts. The fifos (linux/kfifo.h) provided by the kernal is used to not only store bytes from the user, but also connect the upper half and the lower half.

As the main focus of this project was on hardware interrupts, especially hardware interrupts on the Raspberry Pi, Device Tree was again needed to let the operating system know which pins to use and what kind of interrupts to expect. Following are some useful resources to get started on how to use interrupts with Device Tree.

For my use case, the device tree overlay has specified that an interrupt would listen on “falling-edges” on the specified GPIO pin (pin 17 by default). As UART starts with a start bit which goes from a logic high (1) to a low (0), this interrupt would be triggered on the start bits. However, there may also be falling edges in the data bits as well. Therefore, instead of just starting the hrtimer to handle the reading of the following data bits, it always checks if the hrtimer is active (i.e. waiting or executing hrtimer callback function). If the hrtimer is active, it means that it is already handling the data bits and it is safe to ignore this particular falling edge interrupt. I think it’s quite an elegant method of handling incoming bits, but the results were not great.

After implementing the device driver, I tested with multiple baudrates to see how reliable it would be. As someone who is not so familiar with hardware in general, based on what I’ve found online, bit error rates were the goto for testing serial communcation. So, although this is not exactly comparing bit-by-bit, I went with byte-by-byte comparison. The error rates are probably higher than the actual, but my main purpose was to compare between different baudrates.

Baudrate (bps) TX Bytes TX Error Bytes TX Error Rate RX Bytes RX Error Bytes RX Error Rate
9600 23855 0 0 % 27216 0 0 %
19200 19465 1 0.005137 % 27180 9 0.033113 %
38400 35037 171 0.488055 % 27918 539 1.930654 %
57600 29416 76 0.258363 % 12642 703 5.560829 %
115200 29484 3858 13.085063 % 3041 1367 44.952318 %

graph

The results were pretty disappointing and I think it’s due to my poor implementation. The absence of oversampling (as well as parity bits) may explain how RX is more unstable at higher baudrates than TX. In addition, the bytes used in the tests were randomly generated which may be quite different from the usual use case. On the positive side, at lower baudrates, it was quite stable and error-free. Below is just a demo video just to show a functioning example.

Overall, this was a great project to get a taste of interrupt handling, and seeing the bytes actually arrive and pop up on the terminal was very satisfying. I think software approach may not be great for all use cases, but I liked how flexible it can be. My code was probably was the issue, but it was a good experience nonetheless.

Go Back