RSS Daily tech news
  • A tiny light trap could unlock million qubit quantum computers
    A new light-based breakthrough could help quantum computers finally scale up. Stanford researchers created miniature optical cavities that efficiently collect light from individual atoms, allowing many qubits to be read at once. The team has already demonstrated working arrays with dozens and even hundreds of cavities. The approach could eventually support massive quantum networks with […]
  • A strange in-between state of matter is finally observed
    When materials become just one atom thick, melting no longer follows the familiar rules. Instead of jumping straight from solid to liquid, an unusual in-between state emerges, where atomic positions loosen like a liquid but still keep some solid-like order. Scientists at the University of Vienna have now captured this elusive “hexatic” phase in real […]
  • New catalyst makes plastic upcycling 10x more efficient than platinum
    Scientists are finding new ways to replace expensive, scarce platinum catalysts with something far more abundant: tungsten carbide. By carefully controlling how tungsten carbide’s atoms are arranged at extremely high temperatures, researchers discovered a specific form that can rival platinum in key chemical reactions, including turning carbon dioxide into useful fuels and chemicals. Even more […]
  • Engineers just created a “phonon laser” that could shrink your next smartphone
    Engineers have created a device that generates incredibly tiny, earthquake-like vibrations on a microchip—and it could transform future electronics. Using a new kind of “phonon laser,” the team can produce ultra-fast surface waves that already play a hidden role in smartphones, GPS systems, and wireless tech. Unlike today’s bulky setups, this single-chip device could deliver […]
  • An old jeweler’s trick could change nuclear timekeeping
    A team of physicists has discovered a surprisingly simple way to build nuclear clocks using tiny amounts of rare thorium. By electroplating thorium onto steel, they achieved the same results as years of work with delicate crystals — but far more efficiently. These clocks could be vastly more precise than current atomic clocks and work […]
  • Critical minerals are hiding in plain sight in U.S. Mines
    Researchers found that U.S. metal mines already contain large amounts of critical minerals that are mostly going unused. Recovering even a small fraction of these byproducts could sharply reduce dependence on imports for materials essential to clean energy and advanced technology. In many cases, the value of these recovered minerals could exceed the value of […]

Port-based configuration and signal manipulation in PIC16F877A

by Florius
Educational image illustrating the pin diagram of the PIC16F877A microcontroller, with focus on configuring I/O ports. The diagram highlights essential pins including VDD, VSS, MCLR, and PORTB pins (RB6, RB7) used for programming/debugging. Accompanied by a visual of the microcontroller and MPLAB X IDE logo.

1. Configuring the GPIO with TRISx

We need to set certain pins to work as General Purpose Input and Output. To do this, we need to look at the 8-bit TRIS register in the PIC16F877A datasheet. An example of port A and B can be seen in the figure below. This data has been taken out directly from the datasheet and it shows the 2 important registers, namely the PORTx and TRISx registers, where x stands for port A, B, C, D or E.

Diagram showing the relationship between TRIS and PORT registers in the PIC16F877A microcontroller. The TRIS register sets the direction of each pin (input or output), while the PORT register reads or writes data to the pin. Useful for understanding digital I/O configuration in embedded programming.

To set a specific pin, or the whole port, as an IO, we need to look at the TRISx register. From the pin diagram, explained in the previous tutorial, we know that port A only has 6 pins and port B has all 8.

  • Setting a TRISx bit (=1) will make the corresponding PORTx pin an input.
  • Clearing a TRISx bit (=0) will make the responding PORTx pin an output.

For the forgetful: 1 is similar to I of Input, while 0 is similar to O of Output.

1.1 Setting the IO

For setting the pin 0 on port A as output or input, we write the following code:

				
					TRISA0 = 0; //sets pin RA0 as output
TRISA0 = 1; // sets pin RA0 as input
				
			

Similar to setting a single pin, we can set a whole port as input/output by directly writing an 8-bit binary value:

				
					TRISB = 0b00000000; // sets all pins on port B as output
TRISB = 0b00001111; // Sets pins RA7 to RA4 as output and pins RA3 to RA0 as inputs
				
			

In the case of TRISA, bits 7 and 6 are unimplemented and these cells are not used. Thus, in the case of TRISA you could write the following code and it has the same outcome (take caution with TRISE, I recommend to check the datasheet for this).

				
					TRISA = 0b11000000; //Sets RA5 to RA0 as output
TRISA = 0b00000000; //Sets RA5 to RA0 as output
				
			

Binary can also be written in hexadecimal numbers 0 – F. There is no difference in MPLAB, and it depends on your own preference what you would like.

				
					TRISB = 0b01111001; // In binary
TRISB = 0x79; // In hexadecidemal gives the same result
				
			

2. Configuring the GPIO with PORTx

Similar to the TRISx registers, each port has its own PORTx register, as shown in the example figure above, which can be used to read the register or write a high / low to the pin.

2.1 Writing a High or Low on the output

When the  pin is set as output with the TRISx register, we can use the PORTx register to set the pin either as low (0V) or high (5V). This can be done by the following:

				
					RB1 = 0; // Set the output of the pin to low (0V)
RB1 = 1; // Set the output of the pin to high (5V)

PORTB = 0b00000001; // Set all pins of port B to low, besides RB0 which is high 
PORTB  = 0xFF; // Set all pins of port B as high
				
			

2.2 Reading the input

When the pin is set as a digital input, the value of 0 or 1 is stored in the PORTx register, and we can obtain it by accessing either a single pin (e.g. RB0) or the entire port (e.g. PORTC). In the examples below, I write an example code of how it could look like.

Reading a single pin
				
					void main() {

    TRISB0 = 1;  // Set RB0 as input

    unsigned char X;

    X = RB0;  // Read RB0 and assign it to X

    // Now X contains the value of RB0 (0 or 1)
    // ... Rest of your code ...
}
				
			
Reading the whole port
				
					void main() {

    TRISC = 0xFF;  // Set all bits of TRISC to 1 to configure PORTC as inputs

    unsigned char inputValues;

    inputValues = PORTC;

    // Now the variable inputValues holds the states of all input pins of PORTC
    // ... Rest of your code ...
}
				
			

3. Example with switch & LED

In this demonstration, we will integrate the PIC16F877A microcontroller with a switch, which triggers the activation of an LED upon connection. The switch is connected on one end to the 5V power supply, while on the other side it connects to pin RB5 of the microcontroller. An additional resistor of 10K Ohm is placed that is connected to the ground. The purpose of this pull-down resistor is to ensure that when the button is not pressed, the RB5 pin is pulled to a known logic level (GND). This prevents the input from floating and causing erratic behavior.

On pin RD5 we connect the LED, that is programmed to turn on when the switch is ON, as shown in the Figure below. Why we use a resistor of 330 Ohm in series with the LED is explained in my following article on connecting an LED

Circuit diagram of PIC16F877A interfaced with a switch as input and LED as output.

The program for the setup above will be the following:

				
					#include <xc.h>

// Configuration bits
#pragma config FOSC = HS   // External high-speed crystal oscillator
#pragma config WDTE = OFF  // Watchdog timer disabled
#pragma config PWRTE = OFF // Power-up timer disabled
#pragma config BOREN = OFF // Brown-out reset disabled
#pragma config LVP = OFF   // Low voltage programming mode disabled

#define _XTAL_FREQ 20000000 // 20MHz external crystal frequency

void main(void) {
    //Set the pins as GPIO
    TRISB5 = 1; // Set RB5 as input
    TRISD5 = 0; // Set RD5 as output

    RD5 = 0; // Turn OFF the LED initially

    while (1) {
        if (RB5 == 1) {
            RD5 = 1; // Turn ON the LED when switch is ON (RB5 ==1)
        } else {
            RD5 = 0; // Turn OFF the LED when switch is OFF (RB5 == 0)
        }
    }
}
				
			

4. Summary

In summary, we learned how to set the general purpose input output using the TRISx register. Furthermore, for outputs we can use the PORTx register to set the output to either Low (0V) or High (5V). For the inputs, we can directly read them and put them in a variable. Finally, we showed an example of a circuit and program for the PIC16F877A microcontroller interfaced with a switch that controls the LED.

Leave a comment

This site uses Akismet to reduce spam. Learn how your comment data is processed.