Let us look at the basics of 'C' for programming AVR Micrcontrollers in this tutorial. Simple stuff like setting and clearing bits is important to any project you do.
It is often required to set, clear, toggle and check bit status of a Register without affecting any other bits. Also it is important to do in a way that is compiler independent and code can be ported without modifications.


Consider the Data Direction Register (DDRD) shown below. As you know we need to set or clear bits in order to make the corresponding pin output or input respectively.

7 6 5 4 3 2 1 0
DDD7 DDD6 DDD5 DDD4 DDD3 DDD2 DDD1 DDD0

Let us also say that there is a switch connected to PORTD2 and LED connected to PORTD4 as shown.

0 AVR C basics.png

Now what we need to accomplish is this:

7 6 5 4 3 2 1 0
DDD7 DDD6 DDD5 DDD4 DDD3 DDD2 DDD1 DDD0
0 0 0 1 0 0 0 0

Setting a bit

Let's set the bit 4 to make the port pin as output. There is no direct bit set or clear instruction in 'C' that we can call to do this. Hence lets start with the simple binary number: $$0b00000001$$ Now lets left shift it 4 times to get $$0b00010000$$ Now if we OR this result with DDRD register which has default value of 0b00000000 we get:

\begin{array}{r} &0b00000000\\ &0b00010000\\ \hline &0b00010000 \end{array}

It is important to note that even if the value of DDRD was different, only bit 4 would have been set with the operation above without affecting other bits. Now let us write that in the hex notation in a C Statement.

  1. DDRD = DDRD | (1 << 4);

we can also concatenate the OR operation and write as:

  1. DDRD | = (1 << 4);

Taking this a little further, you might want to define the number 4 as a constant, so that if LED is connected to some other pin, you may not want to change in all the places you've used it. Like so,

  1. #define LED 4
  2. DDRD | = (1 << LED);

Clearing a bit

Now let us go ahead and clear the bit-2 and also assume this time that the DDRD value is 0b01010101 we need to obtain 0b01010001

Again Lets start with $$0b00000001$$ Left shift 2 times to get $$0b00000100$$ Let's invert the result to get $$0b11111011$$

Now let's take this result and & it with the DDRD value \begin{array}{r} &0b01010101\\ &0b11111011\\ \hline &0b01010001 \end{array}

Let's write a this in 'C' statements:

  1. #define mySwitch 2
  2. DDRD &= ~(1<<mySwitch);

Checking a bit

Lots of times during programming, you will want check if a bit is set or clear. For example to check if the timer flag is set or let's say in our example if the switch is pressed or released. Often this is accomplished by checking status of particular bit in a given register.

Lets say we want to check the status of 4th bit of PIND register. Then the status can be obtained by

  1. status = (PIND & (1<<mySwitch) ! = 0);

Toggling a bit

If we need to toggle the LED connected to PORTD, bit-4 as with our example it can be accomplished by XOR operation as shown. Assuming the intial value of PORTD register is 0b00000000

As before let's start with $$0b00000001$$

Let's left shift by 4 times to get $$0b00010000$$

Now lets XOR the value obtained with the PORTD value. \begin{array}{r} &0b00000000\\ &0b00010000\\ \hline &0b00010000 \end{array}

It can be summed up as

  1. #define LED 4
  2. PORTD ^= (1 << LED);

So every time the statement is executed the LED changes it state.

Macros

We have defined all of these operations in stdutils.h file. Which you can include and use these and other similar functions.

I have included a snippet below, you may download the complete file.

  1. #define util_GetBitMask(bit) (1<<(bit))
  2. #define util_BitSet(x,bit) ((x) |= util_GetBitMask(bit))
  3. #define util_BitClear(x,bit) ((x) &= ~util_GetBitMask(bit))
  4. #define util_BitToggle(x,bit) ((x) ^= util_GetBitMask(bit))
  5. #define util_UpdateBit(x,bit,val) ((val)? util_BitSet(x,bit): util_BitClear(x,bit))
  6.  
  7.  
  8. #define util_GetBitStatus(x,bit) (((x)&(util_GetBitMask(bit)))!=0u)
  9. #define util_IsBitSet(x,bit) (((x)&(util_GetBitMask(bit)))!=0u)
  10. #define util_IsBitCleared(x,bit) (((x)&(util_GetBitMask(bit)))==0u)
  11.  
  12.  
  13. #define util_AreAllBitsSet(x,BitMask) (((x)&(BitMask))==BitMask)
  14. #define util_AreAnyBitsSet(x,BitMask) (((x)&(BitMask))!=0x00u)

AVR I/O Register Configuration

In this tutorial we are going to discuss the port configuration of AVR/Atmel controllers or in general Atmega family. In this tutorial we will be using Atmega32 as reference, same will be applicable...

Basics of AVR 'C'

Let us look at the basics of 'C' for programming AVR Micrcontrollers in this tutorial. Simple stuff like setting and clearing bits is important to any project you do. It is often required to set...

AVR Hardware and Software Setup

In this tutorial we will look at the basic setup required to get started with AVR series of microcontrollers. There are two aspects to it, the software and the hardware. Fortunately, we for AVR...

AVR Timer programming

Basics Timers come in handy when you want to set some time interval like your alarm. This can be very precise to a few microseconds. Timers/Counters are essential part of any modern...