PS2 Keyboard Interface with PIC16F877A

PS2KeyboardInterface

In this project, an interface between a PS2 keyboard and PIC16F877A is implemented. The main goal of this project is to use a computer keyboard with any instrument without needing a computer. keyboards are basically connected via either USB or PS2. In this project, PS2 Keyboard is interfaced with PIC16F877A. PS2 connector has 6-pin Mini-DIN connector used to connect keyboard to a PC compatible computer system. these 6-pins are:

  • Data

  • Clock

  • +Vcc

  • GND

  • The other two pins are not used.

The first 4 pins are the necessary pins needed to perform a valid communication between the keyboard and the MCU. +Vcc and GND must connected to a 7805 voltage regulator, and the Data and clock lines should be connected to the MCU.

osc1

The data and clock waveform are shown in Fig.(1). The first waveform represents the clock signal and the second one represents the data.  As can be noticed, there is a data bit at every falling edge. By using the capture function in PIC MC, we can get the ASCII code of each key and display it on the character LCD.

// Author: Abdullah M. Zyarah
// www.DigitalComputing.net

#include <16f877a.h>
#use     delay(clock=20MHZ)
#include <LCD420-d.c>
#fuses   hs,nowdt,nocpd,nolvp,noprotect

#byte intcon=0x0b                  // to turn off the interrupt flag
#byte portb=0x06
#byte portd=0x08                   // for data display
#bit  ex_int_f=intcon.1            // external interrupt flag bit
#bit  data_line=portb.1            // data line is RB1

short  data;                       // a variable to store the incoming data
char   j=0,x=1,m=0,letter=0;

/* The interrupt routine has the following:
1- a variable (j) to count the number of input bits (11 bits)
at the press, and 22 bits at the press+release, so the total will be 33bit.
2- a variable (letter) to hold the required data bit, "ASCII of letters".
3- a variable (m) to count the number of data bits in letter.

#int_ext                        
void clock_int()
{
    j++;
    if(j==33)                // number of the sent data bits (press11+release(11+11)
    j=0;                     // clear to receive a data from new button
    data=data_line;    
    if((j>=2)&&(j<=9))        // extract the data (start bit (1)+data bits(8)+party and stop(2))
    {
        if(x==data)            // making a mask, and save it in a letter
        {
            bit_set(letter,m);
        }
        m++;
        if(m==8)
        {
            m=0;
            switch(letter)
            {
                case 0x1C:
                lcd_putc("a");
                break;

                case 0x1B:
                lcd_putc("s");
                break;
        
                case 0x23:
                lcd_putc("d");
                break;

                case 0x2B:
                lcd_putc("f");
                break;

                case 0x34:
                lcd_putc("g");
                break;

                case 0x33:
                lcd_putc("h");
                break;

                case 0x3B:
                lcd_putc("j");
                break;

                case 0x42:
                lcd_putc("k");
                break;

                case 0x4B:
                lcd_putc("l");
                break;

                case 0x4C:
                lcd_putc(";");
                break;

                case 0x15:
                lcd_putc("q");
                break;

                case 0x1D:
                lcd_putc("w");
                break;

                case 0x24:
                lcd_putc("e");
                break;

                case 0x2D:
                lcd_putc("r");
                break;

                case 0x2C:
                lcd_putc("t");
                break;

                case 0x35:
                lcd_putc("y");
                break;

                case 0x3C:
                lcd_putc("u");
                break;

                case 0x43:
                lcd_putc("i");
                break;

                case 0x44:
                lcd_putc("o");
                break;

                case 0x4D:
                lcd_putc("p");
                break;

                case 0x54:
                lcd_putc("[");
                break;
    
                case 0x5B:
                lcd_putc("]");
                break;

                case 0x16:
                lcd_putc("1");
                break;

                case 0x1E:
                lcd_putc("2");
                break;

                case 0x26:
                lcd_putc("3");
                break;

                case 0x25:
                lcd_putc("4");
                break;

                case 0x2E:
                lcd_putc("5");
                break;

                case 0x36:
                lcd_putc("6");
                break;

                case 0x3D:
                lcd_putc("7");
                break;

                case 0x3E:
                lcd_putc("8");
                break;

                case 0x46:
                lcd_putc("9");
                break;

                case 0x45:
                lcd_putc("0");
                break;

                case 0x1A:
                lcd_putc("z");
                break;

                case 0x22:
                lcd_putc("x");
                break;

                case 0x21:
                lcd_putc("c");
                break;

                case 0x2A:
                lcd_putc("v");
                break;

                case 0x32:
                lcd_putc("b");
                break;

                case 0x31:
                lcd_putc("n");
                break;

                case 0x3A:
                lcd_putc("m");
                break;

                case 0x49:
                lcd_putc(".");
                break;

                case 0x4A:
                lcd_putc("?");
                break;

                case 0x66:
                lcd_putc('\b');
                break;

                case 0x5A:
                lcd_putc('\n');
                break;

                case 0x29:
                lcd_putc(" ");
                break;

            }
            letter=0;
        }    
    }       
    ex_int_f=0;
}

void main()
{
    set_tris_b(0x0F);
    set_tris_d(0x00);
    portd=0x00;            
    portb=0x00;
    delay_ms(100);
    lcd_init();
    ext_int_edge(H_To_L);            // interrupt on high to low event
    enable_interrupts(int_ext);        // enable the external interrupt
    enable_interrupts(global);

    while(1)
    {
                    
    }
}