-2

在此处输入图像描述 问题 1- 在如下所示的系统中,用户需要输入 4 位密码。程序第一次运行时,它应该说“输入密码”。当用户输入4位密码后按Square键时,如果密码为1234,则b0连接的LED会闪烁,液晶屏应显示您知道密码。如果他输入错误的密码3次,屏幕上应该写“BLOKE”。然后他应该输入4位数的“4321”阻止密码。如果他正确输入了阻止密码,连接到 b0 的 LED 应该会闪烁,并且 LCD 应该会显示您知道密码。

4

1 回答 1

0

我希望版主会在几个小时或几天内删除这个帖子。

在那之前,我将获得一些乐趣并发布一个应用程序,该应用程序实现了“应该”让 PIC16F877A、LCD 模块和 3x4 键盘工作的 Proteus 仿真模型的基础设施。

实现密码验证的代码留给原始发帖人来实现。

请注意,我尚未使用 Proteus 模拟工具测试此代码。它似乎在 MPLABX 模拟器中工作,但这并没有说太多。

/*
 * File:    main.c
 * Author:  dan1138
 * Target:  PIC16F877A
 * Compiler: XC8 v2.32
 * IDE: MPLABX v5.50
 * 
 * Created on January 27, 2022, 10:29 PM
 * 
 *                          PIC16F877A
 *                  +----------:_:----------+
 *        VPP ->  1 : MCLR/VPP      PGD/RB7 : 40 <> LCD_D7/PGD
 *            <>  2 : RA0/AN0       PGC/RB6 : 39 <> LCD_D6/PGC
 *            <>  3 : RA1               RB5 : 38 <> LCD_D5
 *            <>  4 : RA2               RB4 : 37 <> LCD_D4
 *            <>  5 : RA3               RB3 : 36 <> LCD_EN
 *            <>  6 : RA4               RB2 : 35 <> LCD_RW
 *            <>  7 : RA5               RB1 : 34 <> LCD_RS
 *            <>  8 : RE0          INT0/RB0 : 33 <> LED
 *            <>  9 : RE1               VDD : 32 <- PWR
 *            <> 10 : RE2               VSS : 31 <- GND
 *        PWR -> 11 : VDD               RD7 : 30 <> 
 *        GND -> 12 : VSS               RD6 : 29 <> 
 * 20MHz XTAL <> 13 : OSC1              RD5 : 28 <> 
 * 20MHz XTAL <> 14 : OSC2              RD4 : 27 <> 
 *      KP_C3 <> 15 : RC0/SOSCO   RX/DT/RC7 : 26 <> KP_RD
 *      KP_C2 <> 16 : RC1/SOSCI   TX/CK/RC6 : 25 <> KP_RC
 *      KP_C1 <> 17 : RC2/CCP1          RC5 : 24 <> KP_RB
 *            <> 18 : RC3               RC4 : 23 <> KP_RA
 *            <> 19 : RD0               RD3 : 22 <> 
 *            <> 20 : RD1               RD2 : 21 <> 
 *                  +-----------------------:
 *                           DIP-40
 * Description:
 * 
 */
// CONFIG
#pragma config FOSC = HS        // Oscillator Selection bits (HS oscillator)
#pragma config WDTE = OFF       // Watchdog Timer Enable bit (WDT disabled)
#pragma config PWRTE = OFF      // Power-up Timer Enable bit (PWRT disabled)
#pragma config BOREN = OFF      // Brown-out Reset Enable bit (BOR disabled)
#pragma config LVP = OFF        // Low-Voltage (Single-Supply) In-Circuit Serial Programming Enable bit (RB3 is digital I/O, HV on MCLR must be used for programming)
#pragma config CPD = OFF        // Data EEPROM Memory Code Protection bit (Data EEPROM code protection off)
#pragma config WRT = OFF        // Flash Program Memory Write Enable bits (Write protection off; all program memory may be written to by EECON control)
#pragma config CP = OFF         // Flash Program Memory Code Protection bit (Code protection off)

/*
 * Tell compiler what we intend to set the system oscillator frequency as.
 */
#define _XTAL_FREQ (20000000UL)

#include <xc.h>
#include <stdio.h>

/* Define the LCD port pins */
#define LCD_DATA_BITS_MASK  0x3C
#define LCD_PORT_OUT        PORTC
#define LCD_PORT_DIR        TRISC

#define LCD_RS_PIN          PORTBbits.RB1
#define LCD_RW_PIN          PORTBbits.RB2
#define LCD_EN_PIN          PORTBbits.RB3
#define LCD_D4_PIN          PORTBbits.RB4
#define LCD_D5_PIN          PORTBbits.RB5
#define LCD_D6_PIN          PORTBbits.RB6
#define LCD_D7_PIN          PORTBbits.RB7

#define LCD_RS_DIR          TRISBbits.TRISB1
#define LCD_RW_DIR          TRISBbits.TRISB2
#define LCD_EN_DIR          TRISBbits.TRISB3
#define LCD_D4_DIR          TRISBbits.TRISB4
#define LCD_D5_DIR          TRISBbits.TRISB5
#define LCD_D6_DIR          TRISBbits.TRISB6
#define LCD_D7_DIR          TRISBbits.TRISB7

/* Clear display command */
#define CLEAR_DISPLAY       0b00000001

/* Return home command */
#define RETURN_HOME         0b00000010

/* Display ON/OFF Control defines */
#define DON                 0b00001111   /* Display on      */
#define DOFF                0b00001011   /* Display off     */
#define CURSOR_ON           0b00001111   /* Cursor on       */
#define CURSOR_OFF          0b00001101   /* Cursor off      */
#define BLINK_ON            0b00001111   /* Cursor Blink    */
#define BLINK_OFF           0b00001110   /* Cursor No Blink */

/* Cursor or Display Shift defines */
#define SHIFT_CUR_LEFT      0b00010011   /* Cursor shifts to the left   */
#define SHIFT_CUR_RIGHT     0b00010111   /* Cursor shifts to the right  */
#define SHIFT_DISP_LEFT     0b00011011   /* Display shifts to the left  */
#define SHIFT_DISP_RIGHT    0b00011111   /* Display shifts to the right */

/* Function Set defines */
#define FOUR_BIT            0b00101111   /* 4-bit Interface               */
#define EIGHT_BIT           0b00111111   /* 8-bit Interface               */
#define LINE_5X7            0b00110011   /* 5x7 characters, single line   */
#define LINE_5X10           0b00110111   /* 5x10 characters               */
#define LINES_5X7           0b00111011   /* 5x7 characters, multiple line */

/* Start address of each line */
#define LINE_ONE            0x00
#define LINE_TWO            0x40

static void LCD_E_Pulse(void)
{
    LCD_EN_PIN = 1;
    __delay_us(4);
    LCD_EN_PIN = 0;
    __delay_us(4);
}

static void LCD_DelayPOR(void)
{
    __delay_ms(15);
}

static void LCD_Delay(void)
{
    __delay_ms(5);
}

static void LCD_PutByte(unsigned char LCD_Data)
{
    LCD_PORT_DIR &= ~LCD_DATA_BITS_MASK; /* make LCD data bits outputs */
    
    /* send first(high) nibble */
    LCD_PORT_OUT &= ~LCD_DATA_BITS_MASK;
    if(LCD_Data & 0x10) LCD_D4_PIN = 1;
    if(LCD_Data & 0x20) LCD_D5_PIN = 1;
    if(LCD_Data & 0x40) LCD_D6_PIN = 1;
    if(LCD_Data & 0x80) LCD_D7_PIN = 1;
    LCD_E_Pulse();
    
    /* send second(low) nibble */
    LCD_PORT_OUT &= ~LCD_DATA_BITS_MASK;
    if(LCD_Data & 0x01) LCD_D4_PIN = 1;
    if(LCD_Data & 0x02) LCD_D5_PIN = 1;
    if(LCD_Data & 0x04) LCD_D6_PIN = 1;
    if(LCD_Data & 0x08) LCD_D7_PIN = 1;
    LCD_E_Pulse();

    LCD_PORT_DIR |= LCD_DATA_BITS_MASK; /* make LCD data bits inputs */
}

void LCD_SetPosition(unsigned char data)
{
    LCD_RS_PIN = 0;
    LCD_PutByte((unsigned char)(data | 0x80));
    __delay_us(40);
}

void LCD_WriteCmd(unsigned char data)
{
    LCD_RS_PIN = 0;
    LCD_PutByte(data);
    __delay_ms(4);
}

void LCD_WriteData(unsigned char data)
{
    LCD_RS_PIN = 1;
    LCD_PutByte(data);
    LCD_RS_PIN = 0;
    __delay_us(40);
}

void LCD_Init(void) 
{
    unsigned char LCD_Data;
    
    LCD_PORT_DIR &= ~LCD_DATA_BITS_MASK;    /* make LCD data bits outputs */
    LCD_EN_DIR = 0;                         /* make LCD Enable strobe an output */
    LCD_RW_DIR = 0;                         /* make LCD Read/Write an output */
    LCD_RS_DIR = 0;                         /* make LCD Register select an output */
    LCD_EN_PIN = 0;                         /* set LCD Enable strobe to not active */
    LCD_RW_PIN = 0;                         /* set LCD write mode  */
    LCD_RS_PIN = 0;                         /* set LCD Register select to command group */
    LCD_PORT_OUT &= ~LCD_DATA_BITS_MASK;    /* set LCD data bits to zero */
    LCD_DelayPOR();                         /* wait for LCD power on to complete */

    /* Force LCD to 8-bit mode */
    LCD_PORT_OUT &= ~LCD_DATA_BITS_MASK;    /* set LCD data bits to zero */
    LCD_D4_PIN = 1;
    LCD_D5_PIN = 1;
    LCD_E_Pulse();
    LCD_Delay();
    LCD_E_Pulse();
    LCD_Delay();
    LCD_E_Pulse();
    LCD_Delay();
    
    /* Set LCD to 4-bit mode */
    LCD_PORT_OUT &= ~LCD_DATA_BITS_MASK;    /* set LCD data bits to zero */
    LCD_D5_PIN = 1;
    LCD_E_Pulse();
    LCD_Delay();

    /* Initialize LCD mode */
    LCD_WriteCmd(FOUR_BIT & LINES_5X7);

    /* Turn on display, Setup cursor and blinking */
    LCD_WriteCmd(DOFF & CURSOR_OFF & BLINK_OFF);
    LCD_WriteCmd(DON & CURSOR_OFF & BLINK_OFF);
    LCD_WriteCmd(CLEAR_DISPLAY);
    LCD_WriteCmd(SHIFT_CUR_LEFT);

    /* Set first position on line one, left most character */
    LCD_SetPosition(LINE_ONE);
}
/*
 * Hook for printf
 */
void putch(char txData)
{
    LCD_WriteData(txData);
}
/*
 * Keypad 3x4
 */
#define KP_DEBOUNCE_COUNT (16)

#define KP_RA_IN      PORTCbits.RC4
#define KP_RB_IN      PORTCbits.RC5
#define KP_RC_IN      PORTCbits.RC6
#define KP_RD_IN      PORTCbits.RC7
#define KP_C1_OUT     PORTCbits.RC2
#define KP_C2_OUT     PORTCbits.RC1
#define KP_C3_OUT     PORTCbits.RC0

#define KP_RA_IN_DIR  TRISCbits.TRISC4
#define KP_RB_IN_DIR  TRISCbits.TRISC5
#define KP_RC_IN_DIR  TRISCbits.TRISC6
#define KP_RD_IN_DIR  TRISCbits.TRISC7
#define KP_C1_OUT_DIR TRISCbits.TRISC2
#define KP_C2_OUT_DIR TRISCbits.TRISC1
#define KP_C3_OUT_DIR TRISCbits.TRISC0

enum eKeyEvent
{
    eNoEvent = 0,
    eKeyChanged
};

typedef enum eKeyEvent eKeyEvent_t;

struct sKeypadEvent
{
    unsigned int ButtonMatrix;
    unsigned int ChangedMask;    
};

typedef struct sKeypadEvent KeypadEvent_t;

static unsigned int  KP_Sample;
static unsigned int  KP_Last;
static unsigned int  KP_Changed;
static unsigned int  KP_Stable;
static unsigned char KP_DebounceCounter;
/*
 * Initialize the GPIO pins used for the 3x4 keypad
 */
void Keypad_Init(void)
{
    KP_RB_IN_DIR  = 1;
    KP_RB_IN_DIR  = 1;
    KP_RC_IN_DIR  = 1;
    KP_RD_IN_DIR  = 1;
    KP_C1_OUT_DIR = 0;
    KP_C2_OUT_DIR = 0;
    KP_C3_OUT_DIR = 0;
    KP_Last = 0;
    KP_DebounceCounter = 0;
}
/*
 * Called from ISR handler to sample all keys
 * in the keypad matrix, debounce and update the
 * stable state.
 */
void Keypad_Scan(void)
{
    KP_Sample = 0;
    KP_C1_OUT = 1;
    KP_C2_OUT = 1;
    KP_C3_OUT = 1;
    KP_C2_OUT_DIR = 1;
    KP_C3_OUT_DIR = 1;
    KP_C1_OUT_DIR = 0;
    KP_C1_OUT = 0;
    if (!KP_RA_IN) KP_Sample |= 0x0001;
    if (!KP_RB_IN) KP_Sample |= 0x0002;
    if (!KP_RC_IN) KP_Sample |= 0x0004;
    if (!KP_RD_IN) KP_Sample |= 0x0008;
    KP_C1_OUT = 1;
    KP_C1_OUT_DIR = 1;
    KP_C3_OUT_DIR = 1;
    KP_C2_OUT_DIR = 0;
    KP_C2_OUT = 0;
    if (!KP_RA_IN) KP_Sample |= 0x0010;
    if (!KP_RB_IN) KP_Sample |= 0x0020;
    if (!KP_RC_IN) KP_Sample |= 0x0040;
    if (!KP_RD_IN) KP_Sample |= 0x0080;
    KP_C2_OUT = 1;
    KP_C2_OUT_DIR = 1;
    KP_C1_OUT_DIR = 1;
    KP_C3_OUT_DIR = 0;
    KP_C3_OUT = 0;
    if (!KP_RA_IN) KP_Sample |= 0x0100;
    if (!KP_RB_IN) KP_Sample |= 0x0200;
    if (!KP_RC_IN) KP_Sample |= 0x0400;
    if (!KP_RD_IN) KP_Sample |= 0x0800;
    KP_C2_OUT = 0;
    KP_C1_OUT = 0;
    KP_C1_OUT_DIR = 0;
    KP_C2_OUT_DIR = 0;
    KP_C3_OUT_DIR = 0;
    
    /* check if matrix changed since last scan */
    if ((KP_Sample ^ KP_Last) != 0)
    {
        KP_Last = KP_Sample;
        KP_DebounceCounter = 0;
        return;
    }

    /* check if we have sampled inputs for long enough to debounce */
    if (KP_DebounceCounter < KP_DEBOUNCE_COUNT)
    {
        KP_DebounceCounter++;
        return;
    }

    /* Update the stable output only after pevious stable state has been read */
    if (KP_Changed == 0)
    {
        KP_Changed = KP_Sample ^ KP_Stable;
        KP_Stable = KP_Sample;
    }
}
/*
 * Returns non-zero when a key event occurs.
 * A key event is when one key is pressed or released.
 */
eKeyEvent_t Keypad_GetEvent(void)
{
    eKeyEvent_t Event;
    

    INTCONbits.TMR0IE = 0;  /* disable tick to read keypad sample memory */
    if (KP_Changed == 0)
    {
        Event = eNoEvent;
    }
    else
    {
        Event = eKeyChanged;
    }
    INTCONbits.TMR0IE = 1;  /* enable tick */
    
    return Event;
}
/*
 * Returns ASCII character of keypad event.
 * If more than one key is pressed returns ZERO.
 */
unsigned char Keypad_GetKey(KeypadEvent_t * KeypadEvent)
{
    unsigned char Key;
    unsigned int ButtonMatrix;
    unsigned int ChangedMask;    

    Key = 0;
    INTCONbits.TMR0IE = 0;  /* disable tick to read keypad sample memory */
    ButtonMatrix = KP_Stable;
    ChangedMask  = KP_Changed;
    /* Tell ISR we have read the current state */
    KP_Changed = 0;
    INTCONbits.TMR0IE = 1;  /* enable tick */

    /* return current state of the keypad matrix */
    if (KeypadEvent)
    {
        KeypadEvent->ButtonMatrix = ButtonMatrix;
        KeypadEvent->ChangedMask  = ChangedMask;
    }
    /* decode key in ASCII */
    if (ChangedMask)
    {
        switch (ButtonMatrix)
        {
        case 0x0001U:
            Key = '1';
            break;
        case 0x0002U:
            Key = '4';
            break;
        case 0x0004U:
            Key = '7';
            break;
        case 0x0008U:
            Key = '*';
            break;
        case 0x0010U:
            Key = '2';
            break;
        case 0x0020U:
            Key = '5';
            break;
        case 0x0040U:
            Key = '8';
            break;
        case 0x0080U:
            Key = '0';
            break;
        case 0x0100U:
            Key = '3';
            break;
        case 0x0200U:
            Key = '6';
            break;
        case 0x0400U:
            Key = '9';
            break;
        case 0x0800U:
            Key = '#';
            break;
        default:
            Key = 0;
            break;
        }
    }
    return Key;
}
/*
 * Setup TIMER0 to assert an interrupt every 16384 instruction cycles
 */
void Tick_Init(void)
{
    INTCONbits.TMR0IE = 0;
    OPTION_REG = 0xD3;      /* TMR0 clock FOSC/4Y, TMR0 prescale 1:16      */
    TMR0 = 0;               /* TIMER0 will assert the overflow flag every 256*16 (4096)               */
    INTCONbits.TMR0IF = 0;  /* instruction cycles, with a 5MHz oscillator this is 0.8192 milliseconds. */
    INTCONbits.TMR0IE = 1;
}
/*
 * Main application
 */

volatile unsigned char SysTick;

void main(void) 
{
    unsigned int  KP_sample;
    unsigned char Key;
    KeypadEvent_t Keypad_Event;

    /* Disable all interrupt sources */
    INTCON = 0;
    PIE1 = 0;
    PIE2 = 0;
    /* Make all GPIO pins digital */
    ADCON1 = 0x0F;
    CMCON = 0x07;
    
    LCD_Init();
    Keypad_Init();
    Tick_Init();
    
    /* Enable interrupt system */
    INTCONbits.PEIE = 1;
    INTCONbits.GIE  = 1;

    __delay_ms(100);

    LCD_SetPosition(LINE_ONE);
    printf("Enter Password");
    
    /*
     * Application loop
     */
    for(;;)
    {
        /* check for and process key presses */
        if (Keypad_GetEvent() == eKeyChanged)
        {
            LCD_SetPosition(LINE_TWO);
            printf("Key Pressed:    ");
            Key = Keypad_GetKey(&Keypad_Event);
            if (Key != 0)
            {
                LCD_SetPosition(LINE_TWO+13);
                LCD_WriteData(Key);
                switch (Key)
                {
                    case '0':
                        break;
                    case '1':
                        break;
                    case '2':
                        break;
                    case '3':
                        break;
                    case '4':
                        break;
                    case '5':
                        break;
                    case '6':
                        break;
                    case '7':
                        break;
                    case '8':
                        break;
                    case '9':
                        break;
                    case '*':
                        break;
                    case '#':
                        break;
                    default:
                        break;
                }
            }
        }
    }
}
/*
 * Interrupt handlers
 */
void __interrupt() ISR_Handler(void)
{
    /* Handle system tick */
    if (INTCONbits.TMR0IE)
    {
        if(INTCONbits.TMR0IF)
        {
            INTCONbits.TMR0IF = 0;
            SysTick++;
            Keypad_Scan();
        }
    }
}
于 2022-01-28T08:05:30.157 回答