Serial Communications with the ATtiny84
In a previous post, I talked about serial communications with an ATmega168. But that chip has USART – hardware support for serial communications. But what about the tinyAVRs? As continuation of my last post on setting up the ATtiny84 for programming, this time, I will talk about sending data from an ATtiny84 to a computer using serial communications.
First, I recommend that you watch this fun and informative video on serial communications by Pete from Sparkfun, USA:
So, now that we have some idea about serial communications, let’s talk about implementing a transmit only (TX) for the ATtiny84. Here is our scheme:

The above image shows the transmission of a byte (with value 0x95). The value is sent one bit at a time, starting with the least significant bit (LSB), every 1/9600 seconds, thus giving us a baud rate of 9600. The data format for a packet for the 9600 baud 8-N-1 serial connection is as follows:
start(low)-[0]-[1]-[2]-[3]-[4]-[5]-[6]-[7]-stop(high)-idle(high)-idle(high)
(In the above, the 2 idle bits at the end are not strictly necessary.)
So how do we implement sending a bit every 1/9600 seconds? For this, we use the 16-bit timer (Timer1) of the Attiny84 in CTC mode. The value for the top of the counter is calculated as shown in the image above.
In the ISR for the timer, we keep track of the current data byte and the current bit being sent, and keep setting the output pin high/low as appropriate.
Here is the code that implements the above ideas.
The Makefile is the same as we used in the ATtiny84 introduction post:
https://gist.github.com/electronut/5689137
This is the schematic for wiring it up:
//
// Serials communications (TX only) with ATtiny84
//
// electronut.in
//
#include <avr/io.h>
#include <string.h>
#include <util/delay.h>
#include <avr/interrupt.h>
#define F_CPU 8000000
char pinTX = PB1;
volatile unsigned char currByte = 0;
volatile int sendingData = 0;
// initialize
void init_serial()
{
// set TX pin as output
DDRB = (1 << pinTX);
// set up timer1:
cli();
// 16 bit timer
// Divide by 1
TCCR1B |= (1<<CS10);
// Count cycles - (1/9600)*8000000
OCR1A = 833;
// Put Timer/Counter1 in CTC mode
TCCR1B |= 1<<WGM12;
// set interrupt flag
TIMSK1 |= 1<<OCIE1A;
sei();
}
// send a byte
void send_byte(char data)
{
// set flag
sendingData = 1;
// set current data byte
currByte = data;
// wait till done
while(sendingData);
}
// which bit was sent last? (0-10)
// -1 implies none sent
volatile char bitNum = -1;
// 16-bit timer CTC handler - Interrupt Service Routine
ISR(TIM1_COMPA_vect)
{
// serial data sent as:
//
// 9600 8 N 1
//
// start(low)-0-1-2-3-4-5-6-7-stop(high)-idle(high)-idle(high)
// 12 bits sent per packet
// idle => high
if(!sendingData) {
PORTB |= (1 << pinTX);
}
else {
if(bitNum < 0) {
// start bit - low
PORTB &= ~(1 << pinTX);
// set bit number
bitNum = 0;
}
else if(bitNum >= 8) {
// stop bit - high
PORTB |= (1 << pinTX);
// increment
bitNum++;
// send 2 idle - high - bits
// not necessary strictly speaking
if(bitNum == 10) {
// done
sendingData = 0;
// unset bit number
bitNum = -1;
}
}
else {
// data bits:
// bit num is in [0, 7]
// extract relevant bit from data
char dataBit = currByte & (1 << bitNum);
if(dataBit) {
PORTB |= (1 << pinTX);
}
else {
PORTB &= ~(1 << pinTX);
}
// update bit number
bitNum++;
}
}
}
// write null terminated string
void send_str(const char* str)
{
int len = strlen(str);
int i;
for (i = 0; i < len; i++) {
send_byte(str[i]);
}
}
int main (void)
{
init_serial();
int count = 10;
// loop
while (1) {
// print count
char str[16];
sprintf(str, "%d\n", count);
if(count) {
send_str(str);
count--;
}
else {
send_str("BOOM\n");
count = 10;
}
_delay_ms(200);
}
return 1;
}
The Makefile is the same as we used in the ATtiny84 introduction post:
https://gist.github.com/electronut/5689137
This is the schematic for wiring it up:

Here is what the breadboard setup looks like:

And here is the output from CoolTerm. It was surprisingly easy to get this working. 😉

Note:
On OS X, you can use the screen command to see the serial output. For example:
$ screen /dev/tty.usbserial-A7006Yqh 9600
