MPLAB XC8编译器PIC18F452多路复用七段显示代码正常工作

时间:2018-03-30 00:04:09

标签: c microcontroller pic mplab xc8

我目前正在研究涉及MPLAB XC8编译器的代码,PIC18F452具有多路复用七段显示器。我想使用连接到PIC18F452的PORTB的RB2和RB3引脚的两个按钮来递增和递减变量“count”,并在此显示屏上显示从0到99的数字。原理图和代码如下所示。

这段代码相对来说功能相同,我不相信原理图是我所看到的问题的原因,也不是字节数组不正确,因为我在使用数组1时能够看到每个数字分段显示。

当尝试使用下图所示的多路复用方案时,问题就出现了。我可以在七段显示器上成功显示两个数字,但执行此代码时会出现奇怪的异常。例如,我似乎无法在任一显示器上显示数字1,4和偶尔显示7,但是当该数字未显示时显示为空白,并且当再次按下该按钮时,下一个数字将按预期显示。

例如:

显示屏显示数字序列的数字如下:     9 ... 1 0 ... 11 ... 1 2 1 3 ... 14 ...等......

3 4 .... 35 ... 36 ... 3 7 ....

不确定问题出在哪里,调试进展不顺利......任何帮助都会受到赞赏。

Schematic for Multiplexed 7 Segment Display

#define _XTAL_FREQ 10000000
#include <xc.h>
#include <stdlib.h>
#define Digit1 PORTBbits.RB1 //variable to sink current to PNP base
#define Digit2 PORTBbits.RB2 //variable to sink current to PNP base
#define Switch1 PORTBbits.RB4 //switch decrement variable
#define Switch2 PORTBbits.RB3 //switch increment variable
#define Pressed1 1 //pressed is high
#define Pressed2 1 //pressed is high
void initialize();
void segment1 (void);
void segment2 (void);
void buttonPress(void);
void delay_ms(unsigned int);
void sendPattern1(unsigned char pattern);
void sendPattern2(unsigned char pattern3);
unsigned char rotateLeft(unsigned char pattern, int no);
unsigned char MSD, LSD, count=0;

主要代码

void main(void){
  initialize();
  Digit1 = 0;
  Digit2 = 0;
  while(1){
  buttonPress();
  MSD = count/10 ;  
  segment1();
  Digit2 = 1;
  delay_ms(10); // Delay for 10 ms
  Digit2 = 0;
  LSD = count%10; 
  segment2();
  Digit1 = 1;
  delay_ms(10); // Delay for 10 ms
  Digit1 = 0;
   }
    return;
}

从阵列中索引最高有效数字和最低有效数字的函数,将其发送到端口以吸收当前低电平信号显示的低电流。

void segment1(void){
unsigned char segArrayC[]={0b11000000,0b11111001,0b00100100, 
0b00110000,0b00011001,0b00010010, 
0b00000010,0b11111000,0b00000000,0b00011000};
 unsigned char pattern;
    pattern = segArrayC[MSD];
    sendPattern1(pattern);
    return;
}
void segment2(void){
unsigned char segArrayD[]= {0b11000000,0b11111001,0b00100100,
0b00110000,0b00011001,0b00010010,0b00000010,
0b11111000,0b00000000,0b00011000};
unsigned char pattern3;
     pattern3 = segArrayD[LSD];
     sendPattern2(pattern3);
     return;
}

按钮按代码

void buttonPress(void){
  if (Switch1 == Pressed1) {
        ++count;
        delay_ms(100);
    }
 if (Switch2 == Pressed2) {
        --count;
        delay_ms(100);
    }

 if(count>=99||count<0)
 {
     count=0; 
     delay_ms(100);
 }
  return;
}

将数组中的字节旋转两个位置左侧以显示在PORT上的功能

/** Rotate pattern to the left 'no' number of bits
 */
unsigned char rotateLeft(unsigned char pattern, int no) {
    return (((pattern << no) & 0xFF) | (pattern >> (8-no)));
}

将索引数组char输出到PORTC和PORTB引脚的函数

void sendPattern1(unsigned char pattern) {
    // Send pattern to appropriate port pins
    unsigned char pattern2;
    PORTC = pattern;
    pattern2=rotateLeft(pattern, 2);
    PORTB = pattern2;
    return;
}
void sendPattern2(unsigned char pattern3) {
    unsigned char pattern4;
    PORTC = pattern3;
    pattern4=rotateLeft(pattern3, 2);
    PORTB = pattern4;
    return;
}

延迟功能

    void delay_ms(unsigned int n){
    while (--n) _delay(2500);
}

初始化要使用的引脚(0输出,1输入)

void initialize() {
     TRISC = 0;
     TRISBbits.TRISB0 = 0;
     TRISBbits.TRISB1 = 0;
     TRISBbits.TRISB2 = 0;
     TRISBbits.TRISB4 = 1;
     TRISBbits.TRISB3 = 1;
     PORTC = 0x00;
     PORTB = 0x00;
}

1 个答案:

答案 0 :(得分:0)

在sendPattern()中,您将一个旋转的位模式写入PORTB。 这会干扰共阳极控制的设置。因此,如果两个右手段都打开,您只能看到两个数字。根据您的原理图,您应该写一个0来打开共阳极。试试这个:

void main()
{
     static const unsigned char segArray[]= 
     { 0b11000000, 0b11111001, 0b00100100, 0b00110000, 0b00011001,
       0b00010010, 0b00000010, 0b11111000, 0b00000000, 0b00011000
     };
     TRISC = 0; //PortC all OUTPUT
     PORTC = 0xFF; //PortC all HIGH = IDLE = LED_OFF

     TRISBbits.TRISB0 = 0; //Output unused
     TRISBbits.TRISB1 = 0; //Output Digit1
     TRISBbits.TRISB2 = 0; //Output Digit2
     TRISBbits.TRISB4 = 1; //Input: Switch PLUS
     TRISBbits.TRISB3 = 1; //Input: Switch MINUS 

     PORTB = 0x00;
     unsigned char count=0;
     for(;;)
     {
        //Handle buttons
        if (Switch1 && count<99) 
        {
           ++count;
           delay_ms(100);
        }
        if (Switch2 && count > 0) 
        {
           --count;
           delay_ms(100);
        }

        //Write high digit
        PORTC = segArray[count/10];  
        Digit2 = 0;
        delay_ms(10); // Delay for 10 ms
        Digit2 = 1;

        //Write low digit
        PORTC = segArray[count%10];  
        Digit1 = 0;
        delay_ms(10); // Delay for 10 ms
        Digit1 = 1;
   }
}