# **GPIO**

Learn how to manipulate and read the voltage levels of the pins on the SJ Board.

- Bit Manipulation
- LPC40xx Memory Map
- General Purpose Input Output
- GPIO Lab Assignment

# Bit Manipulation

Bit-masking is a technique to selectively modify individual bits without affecting other bits.

#### Bit SET

To set a bit, we need to use the OR operator. This is just like an OR logical gate you should've learned in your Digital Design course.

```
// We want to set Bit #7 of a variable called: REG
REG = REG | 0x80;
// Let's set bit #31:
REG = REG | 0x80000000;
// Here is an easier way to write these:
// (1 << 31) means 1 gets shifted left 31 times to produce 0x80000000
REG = REG | (1 << 31);
// Simplify further:
REG |= (1 << 31);
// Set Bit #21 and Bit #23 at the same timeREG |= (1 << 21) | (1 << 23);</pre>
```

### Bit CLEAR

To set a bit to 0, in other words reset or clear a bit, the logic is similar, but instead of **ORing** a bit, we will an **AND** function to clear. **Note: that ANDing something with 0 clears it and ANDing something with a 1 does not change it. The tilde (~) operator can help us invert the bits of a value in the following examples:** 

```
// Assume we want to reset Bit#7 of a register called: REG REG = REG & 0x7F; REG = REG & \sim(0x80); // Same thing as above, but using \sim is easier // Let's reset bit#31: REG = REG & \sim(0x80000000); // Let's show you the easier way:
```

```
REG = REG & \sim(1 << 31);

// Simplify further:

REG &= \sim(1 << 31);

// Reset Bit#21 and Bit# 23:REG &= \sim( (1 << 21) | (1 << 23) );
```

### Bit TOGGLE

```
// Using XOR operator to toggle 5th bitREG ^= (1 << 5);</pre>
```

### Bit CHECK

Suppose you want to check bit 7 of a register is set:

Now let's work through another example in which we want to wait until bit#9 is 0:

```
// One way:
while(REG & (1 << 9) != 0)
{
   continue;
}
// Another way:
while(REG & (1 << 9))
{
   continue;}</pre>
```

### Multi-Bit Insertion

```
// Insert a set of continguous bits into a target value.
// Value within target is unknown. This is shown using X's
//
```

### Multi-Bit Extraction

```
/// Extract a set of contiguous bits from a target value.
///
/// \text{ target} = 0x00FE'DCBA
///
///
///
/// value = 4 -----+
/// width = 8
///
/// return = 0xCB
// Shift target to the left by 4 to make the 0th bit the start of the bits you want to extract.
// Store the result in to a local variable
uint32 t result = target >> 4;
// Since we only want 8 bits from the result, we need to clear away the rest of the bits from
// the original target.
// AND the result with 0xFF, to clear everything except for the first 8 bits.
result = result & 0xFF;
```

# LPC40xx Memory Map

## What is a Memory Map

A **memory map** is a layout of how the memory maps to some set of information. With respect to embedded systems, the memory map we are concerned about maps out where the Flash (ROM), peripherals, interrupt vector table, SRAM, etc are located in address space.

## Memory mapped IO

Memory mapped IO is a a means of mapping memory address space to devices external (**IO**) to the CPU, that is not memory.

#### For example (assuming a 32-bit system)

- Flash could be mapped to addresses **0x00000000** to **0x00100000** (1 Mbyte range)
- GPIO port could be located at address **0x1000000** (1 byte)
- Interrupt vector table could start from **0xFFFFFFF** and run backwards through the memory space
- SRAM gets the rest of the usable space (provided you have enough SRAM to fill that area)

It all depends on the CPU and the system designed around it.

### Port Mapped IO

Port mapped IO uses additional signals from the CPU to qualify which signals are for memory and which are for IO. On Intel products, there is a (~M/IO) pin that is **LOW** when selecting **MEMORY** and **HIGH** when it is selecting **IO**.

The neat thing about using port mapped IO, is that you don't need to sacrifice memory space for IO, nor do you need to decode all 32-address lines. You can limit yourself to just using 8-bits of address space, which limits you to 256 device addresses, but that may be more than enough for your purposes.



Figure 2. Address Decoding with port map

(http://www.dgtal-sysworld.co.in/2012/04/memory-intercaing-to-8085.html)

## LPC40xx memory map

Chapter 2: LPC17xx Memory map

Rev. 2 — 19 August 2010

User manual

#### 2.1 Memory map and peripheral addressing

The ARM Cortex-M3 processor has a single 4 GB address space. The following table shows how this space is used on the LPC17xx.

Table 3. LPC17xx memory usage and details

| Address range                 | General Use                                             | Address range details and des | scription                                                                        |
|-------------------------------|---------------------------------------------------------|-------------------------------|----------------------------------------------------------------------------------|
| 0x0000 0000 to<br>0x1FFF FFFF | On-chip non-volatile memory                             | 0x0000 0000 - 0x0007 FFFF     | For devices with 512 kB of flash memory.                                         |
|                               |                                                         | 0x0000 0000 - 0x0003 FFFF     | For devices with 256 kB of flash memory.                                         |
|                               |                                                         | 0x0000 0000 - 0x0001 FFFF     | For devices with 128 kB of flash memory.                                         |
|                               |                                                         | 0x0000 0000 - 0x0000 FFFF     | For devices with 64 kB of flash memory.                                          |
|                               |                                                         | 0x0000 0000 - 0x0000 7FFF     | For devices with 32 kB of flash memory.                                          |
|                               | On-chip SRAM                                            | 0x1000 0000 - 0x1000 7FFF     | For devices with 32 kB of local SRAM.                                            |
|                               |                                                         | 0x1000 0000 - 0x1000 3FFF     | For devices with 16 kB of local SRAM.                                            |
|                               |                                                         | 0x1000 0000 - 0x1000 1FFF     | For devices with 8 kB of local SRAM.                                             |
|                               | Boot ROM                                                | 0x1FFF 0000 - 0x1FFF 1FFF     | 8 kB Boot ROM with flash services.                                               |
| 0x2000 0000 to<br>0x3FFF FFFF | On-chip SRAM<br>(typically used for<br>peripheral data) | 0x2007 C000 - 0x2007 FFFF     | AHB SRAM - bank 0 (16 kB), present on devices with 32 kB or 64 kB of total SRAM. |
|                               |                                                         | 0x2008 0000 - 0x2008 3FFF     | AHB SRAM - bank 1 (16 kB), present on devices with 64 kB of total SRAM.          |
|                               | GPIO                                                    | 0x2009 C000 - 0x2009 FFFF     | GPIO.                                                                            |
| 0x4000 0000 to<br>0x5FFF FFFF | APB Peripherals                                         | 0x4000 0000 - 0x4007 FFFF     | APB0 Peripherals, up to 32 peripheral blocks 16 kB each.                         |
|                               |                                                         | 0x4008 0000 - 0x400F FFFF     | APB1 Peripherals, up to 32 peripheral blocks 16 kB each.                         |
|                               | AHB peripherals                                         | 0x5000 0000 - 0x501F FFFF     | DMA Controller, Ethernet interface, and USE interface.                           |
| 0xE000 0000 to<br>0xE00F FFFF | Cortex-M3 Private<br>Peripheral Bus                     | 0xE000 0000 - 0xE00F FFFF     | Cortex-M3 related functions, includes the NVIC and System Tick Timer.            |
|                               |                                                         |                               |                                                                                  |

Figure 3. LPC17xx Memory Map, which is nearly the same as the LPC40xx memory map

From this you can get an idea of which section of memory space is used for what. This can be found in the UM10562 LPC40xx user manual. If you take a closer look you will see that very little of the address space is actually taken up. With up to 4 billion+ address spaces (because 2^32 is a big number) to use you have a lot of free space to spread out your IO and peripherals.

### Reducing the number of lines needed to decode IO

The LPC40xx chips, reduce bus line count, make all of the peripherals 32-bit aligned. Which means you must grab 4-bytes at a time. You cannot grab a single byte (8-bits) or a half-byte (16-bits) from memory. This eliminates the 2 least significant bits of address space.

## Accessing IO using Memory Map in C

Please read the following code snippet. This is runnable on your system now. Just copy and paste it into your **main.cpp** file.

```
//The goal of this software is to set the GPIO pin P1.0 to
// low then high after some time. Pin P1.0 is connected to an LED.
// The address to set the direction for GPIOs in port 1 is below:
//
       FI01DIR = 0 \times 2009C020
//
//
// The address to set the output value of a pin in port 1 is below:
//
//
       FI01PIN = 0 \times 2009C034
#include <cstdint>
volatile uint32_t * const FI01DIR = (uint32_t *)(0x2009C020);
volatile uint32_t * const FI01PIN = (uint32_t *)(0x2009C034);
int main(void)
    // Set 0th bit, setting Pin 0.0 to an output pin
    *FI01DIR |= (1 << 0);
    // Set 0th bit, setting Pin 0.0 to high
    *FI01PIN &= ~(1 << 0);
    // Loop for a while (volatile is needed, otherwise this will not loop for very long!)
    for(volatile uint32_t i = 0; i < 1000000; i++);</pre>
    // Clear Oth bit, setting Pin 0.0 to low
    *FI01PIN |= (1 << 0);
    return 0;}
```

volatile keyword tells the compiler not to optimize this variable out, even if it seems useless

Notice "const" placement and how it is placed after the uint32\_t \*. This is because we want to make sure the pointer address never changes and remains constant, but the value that it references should be modifiable.

## Using the LPC40xx.h

The above is nice and it works, but its a lot of work. You have to go back to the user manual to see which addresses are for what register. There must be some better way!!

Take a look at the LPC40xx.h file, which It is located in the

SJSU-Dev/firmware/library/L0\_LowLevel/LPC40xx.h . Here you will find definitions for each peripheral memory address in the system.

Lets say you wanted to port the above code to something a bit more structured:

- Open up "LPC40xx.h"
- Search for "GPIO"
- ? o You will find a struct with the name LPC\_GPIO\_TypeDef.
- Now search for "LPC\_GPIO\_TypeDef" with a #define in the same line.
- You will see that LPC\_GPIO\_TypeDef is a pointer of these structs

```
#define LPC_GPI00 ((LPC_GPI0_TypeDef *) LPC_GPI00_BASE )

#define LPC_GPI01 ((LPC_GPI0_TypeDef *) LPC_GPI01_BASE )

#define LPC_GPI02 ((LPC_GPI0_TypeDef *) LPC_GPI02_BASE )

#define LPC_GPI03 ((LPC_GPI0_TypeDef *) LPC_GPI03_BASE )

#define LPC_GPI04 ((LPC_GPI0_TypeDef *) LPC_GPI04_BASE )
```

- We want to use **LPC\_GPIO1** since that corrisponds to GPIO port 1.
- If you inspect LPC\_GPIO\_TypeDef, you can see the members that represent register FIODIR and FIOPIN
- You can now access FIODIR and FIOPIN in the following way:

```
#include "LPC40xx.h"
int main(void)
{
    // Set direction of P0.0 to 1, which means OUTPUT
```

```
LPC_GPI01->FIODIR |= (1 << 0);

// Set 0th bit, setting Pin 0.0 to high

LPC_GPI01->FIOPIN &= ~(1 << 0);

for(volatile uint32_t i = 0; i < 10000000; i++);

// Clear 0th bit, setting Pin 0.0 to low

LPC_GPI01->FIOPIN |= (1 << 0);

return 0;}</pre>
```

At first this may get tedious, but once you get more experience, you won't open the **LPC40xx.h** file very often. This is the preferred way to access registers in this course.

On occasions, the names of registers in the user manual are not exactly the same in this file.

# General Purpose Input Output

## Objective

To be able to General Purpose Input Output (GPIO), to generate digital output signals and to read input signals. Digital outputs can be used as control signals to other hardware, to transmit information, to signal another computer/controller, to activate a switch or, with sufficient current, to turn on or off LEDs or to make a buzzer sound.

Below will be a discussion on using GPIO to drive an LED.

Although the interface may seem simple, you do need to consider hardware design and know some of the fundamental of electricity. There are a couple of goals for us:

- No hardware damage if faulty firmware is written.
- Circuit should prevent excess amount of current to avoid processor damage.

## Required Background

### You should know the following:

- bit-masking in C
- wire-wrapping or use of a breadboard
- Fundamentals of electricity such as Ohm's law (V = IR) and how diodes work.

## **GPIO**



Figure 1. Internal Design of a GPIO

GPIO stands for "General Purpose Input Output". Each pin can at least be used as an output or input. In an output configuration, the pin voltage is either 0v or 3.3v. In input mode, we can read whether the voltage is 0v or 3.3v.

You can locate a GPIO that you wish to use for a switch or an LED by first starting with the schematic of the board. The schematic will show which pins are "available" because some of the microcontroller pins may be used internally by your development board. After you locate a free pin, such as P2.0, then you can look-up the microcontroller user manual to locate the memory that you can manipulate.

## Hardware Registers Coding

The hardware registers map to physical pins. If we want to attach our switch and the LED to our

microcontroller's PORT0, then here are the relevant registers and their functionality:





## Switch

We will interface our switch to PORT0.2, or port zero's 3rd pin (counting from 0).

Note that the "inline" resistor is used such that if your GPIO is mis-configured as an OUTPUT pin, hardware damage will not occur from badly written software.



Figure 2. Button Switch Circuit Schematic

```
// Set the direction of P0.2 to input
LPC_GPI00->FIODIR &= ~(1 << 2);
// Now, simply read the 32-bit FIOPIN registers, which corresponds to
// 32 physical pins of PORT 0.
// Use AND logic to test if JUST the pin number 2 of port zero is set.
if (LPC_GPI00->FIOPIN & (1 << 2))
{
    // Switch is logical HIGH
}
else
{
    // Switch is logical LOW}</pre>
```

### **LED**

We will interface our LED to PORT0.3, or port zero's 4th pin (counting from 0).

Given below are two configuration is

GPIO

Illy, the "sink" current is higher than "source", hence

```
// Make direction of PORTO.3 as OUTPUT
LPC_GPIOO->FIODIR |= (1 << 3);
// Setting bit 3 to 1 of IOPIN will turn ON LED
// and resetting to 0 will turn OFF LED.
LPC_GPIOO->FIOPIN |= (1 << 3);
// An alternative way, is to use the FIOSET and FIOCLR registers (no OR logic needed)
LPC_GPIOO->FIOSET = (1 << 3);
// Likewise, reset to OLPC_GPIOO->FIOCLR = (1 << 3);</pre>
```

Figure 4. Act

Figure 3. Active High LED circuit schematic

# GPIO Lab Assignment

## Objective

Gain experience doing the following:

- 1. Manipulating a registers in order to access and control physical pins
- 2. Use implemented driver to sense input signals and control LEDs.

## Assignment

Test your knowledge by doing the following:

#### Part 0. Basic GPIO Driver to blink an onboard LED

```
int main()
{
    // 1) Find and choose an onboard LED to manipluate.
    // 2) Use the schematic to figure out which pin it is connected to
    // 3) Use FIODIR to set that pin as an output
    while (true)
{
        // 4) use FIOCLR to set the pin LOW, turning ON the LED
        LOG_INFO("Turning LED ON!");
        Delay(500);    // Delay in milliseconds
        // 5) use FIOSET to set the pin HIGH, turning OFF the LED
        LOG_INFO("Turning LED OFF!");
        Delay(500);
}
return 0;}
```

### Part 1. Implement the LabGPIO Driver

Using the following class template

- 1. Implement ALL class methods.
- 2. All methods must function work as expected of their method name.
- 3. Must be able to handle pins in port 0, 1, and 2.

```
#pragma once
#include <cstdint>
class LabGPIO
public:
 enum class Direction : uint8 t
   kInput = 0,
   kOutput = 1
 };
 enum class State : uint8_t
   kLow = 0,
  kHigh = 1
 };
 /// You should not modify any hardware registers at this point
 /// You should store the port and pin using the constructor.
 ///
 /// @param port - port number between 0 and 5
 /// @param pin - pin number between 0 and 32
  constexpr LabGPIO(uint8_t port, uint8_t pin);
 /// Sets this GPIO as an input
 void SetAsInput();
 /// Sets this GPIO as an output
 void SetAsOutput();
 /// Sets this GPIO as an input
 /// @param output - true => output, false => set pin to input
```

```
void SetDirection(Direction direction);
 /// Set voltage of pin to HIGH
 void SetHigh();
 /// Set voltage of pin to LOW
 void SetLow();
 /// Set pin state to high or low depending on the input state parameter.
 /// Has no effect if the pin is set as "input".
  ///
 /// @param state - State::kHigh => set pin high, State::kLow => set pin low
 void set(State state);
 /// Should return the state of the pin (input or output, doesn't matter)
 /// @return level of pin high => true, low => false
 State Read();
 /// Should return the state of the pin (input or output, doesn't matter)
 ///
 /// @return level of pin high => true, low => false
 bool ReadBool();
private:
 /// port, pin and any other variables should be placed here.
 /// NOTE: Pin state should NEVER be cached! Always check the hardware
           registers for the actual value of the pin.
};
```

### Part 2. Use Driver for an application

The application is to use all 4 internal buttons to control the on board LEDs above them.

```
int main(void)
{
   LabGpio button0(?, ?);
   LabGpio led0(?, ?);

// Initialize button and led here
   while(true)
{
      // Logic to read if button has been RELEASED and if so, TOGGLE LED state;
```

```
}
return 0;}
```

### Requirements:

You MUST NOT use any pre-existing library such as a GPIO class for this assignment.

You **MAY USE** LPC40xx.h as it is not a library but a list of registers mapped to the appropriate locations.

The code must read from the internal button. If a button is **RELEASED**, toggle the state of the LED.

Upload only relevant source files into canvas. A good example is: *main.cpp*, *LabGPIO.hpp*, *LabGPIO.cpp*. See Canvas for rubric and grade breakdown.

#### **Extra Credit**

Add a flashy easter egg feature to your assignment, with your new found LED and switch powers! The extra credit is subject to the instructor's, ISA's and TA's discretion about what is worth the extra credit.

Consider using additional switches and/or LEDs.