haiyang201 发表于 2024-7-20 17:27:09

_奶咖君_ 发表于 2024-7-20 14:40
跳数啥意思,,走着走着 7 下一个变成9然后 然后把8跳了?

每次到17再加就变10

_奶咖君_ 发表于 2024-7-20 17:41:49

haiyang201 发表于 2024-7-20 17:27
每次到17再加就变10

啊?那应该是你的程序逻辑问题啊。。虽然不知道你是怎么配置的,,我这里是如果进入配置模式的话 秒针就不动了,直到退出配置模式。

haiyang201 发表于 2024-7-21 09:50:27

_奶咖君_ 发表于 2024-7-20 17:41
啊?那应该是你的程序逻辑问题啊。。虽然不知道你是怎么配置的,,我这里是如果进入配置模式的话 秒针就 ...

走时是正常的 就是调整时有点问题,继续找问题

陈家乐 发表于 2024-7-21 12:12:03

_奶咖君_ 发表于 2024-7-20 12:40
跳数啥意思,,走着走着 7 下一个变成9然后 然后把8跳了?

这么说的话,难不成是调整参数的按键处理的不太妥当?

陈家乐 发表于 2024-7-21 12:12:48

haiyang201 发表于 2024-7-20 15:26
不是边界问题

你可以私下加我,我给你看看?还是你可以给我发个视频,咱俩瞅瞅啥问题?

angmall 发表于 2024-7-21 12:22:59

这个是用硬件I2C 给这个单片机STC8A8K64D4

/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCMCU.com --------------------------------------------*/
/* --- Web: www.STCMCUDATA.com---------------------------------------*/
/* --- QQ:800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序            */
/*---------------------------------------------------------------------*/


/*************本程序功能说明**************

读写RTC, IC为PCF8563.

用STC的MCU的IO方式驱动8位数码管。

使用Timer0的16位自动重装来产生1ms节拍,程序运行于这个节拍下, 用户修改MCU主时钟频率时,自动定时于1ms.

8位数码管显示时间(小时-分钟-秒).

行列扫描按键键码为17~32.

按键只支持单键按下, 不支持多键同时按下, 那样将会有不可预知的结果.

键按下超过1秒后,将以10键/秒的速度提供重键输出. 用户只需要检测KeyCode是否非0来判断键是否按下.

调整时间键:
键码17: 小时+.
键码18: 小时-.
键码19: 分钟+.
键码20: 分钟-.

下载时, 选择时钟 24MHZ (用户可自行修改频率).

******************************************/


#include    "reg51.h"       //包含此头文件后,里面声明的寄存器不需要再手动输入,避免重复定义
#include    "intrins.h"

#define   MAIN_Fosc       24000000L   //定义主时钟

typedef   unsigned char   u8;
typedef   unsigned int    u16;
typedef   unsigned long   u32;

//手动输入声明"reg51.h"头文件里面没有定义的寄存器
sfr TH2= 0xD6;
sfr TL2= 0xD7;
sfr IE2   = 0xAF;
sfr INT_CLKO = 0x8F;
sfr AUXR = 0x8E;
sfr AUXR1 = 0xA2;
sfr P_SW1 = 0xA2;
sfr P_SW2 = 0xBA;
sfr S2CON = 0x9A;
sfr S2BUF = 0x9B;

sfr P4   = 0xC0;
sfr P5   = 0xC8;
sfr P6   = 0xE8;
sfr P7   = 0xF8;
sfr P1M1 = 0x91;    //PxM1.n,PxM0.n   =00--->Standard,    01--->push-pull
sfr P1M0 = 0x92;    //                  =10--->pure input,11--->open drain
sfr P0M1 = 0x93;
sfr P0M0 = 0x94;
sfr P2M1 = 0x95;
sfr P2M0 = 0x96;
sfr P3M1 = 0xB1;
sfr P3M0 = 0xB2;
sfr P4M1 = 0xB3;
sfr P4M0 = 0xB4;
sfr P5M1 = 0xC9;
sfr P5M0 = 0xCA;
sfr P6M1 = 0xCB;
sfr P6M0 = 0xCC;
sfr P7M1 = 0xE1;
sfr P7M0 = 0xE2;

sbit P00 = P0^0;
sbit P01 = P0^1;
sbit P02 = P0^2;
sbit P03 = P0^3;
sbit P04 = P0^4;
sbit P05 = P0^5;
sbit P06 = P0^6;
sbit P07 = P0^7;
sbit P10 = P1^0;
sbit P11 = P1^1;
sbit P12 = P1^2;
sbit P13 = P1^3;
sbit P14 = P1^4;
sbit P15 = P1^5;
sbit P16 = P1^6;
sbit P17 = P1^7;
sbit P20 = P2^0;
sbit P21 = P2^1;
sbit P22 = P2^2;
sbit P23 = P2^3;
sbit P24 = P2^4;
sbit P25 = P2^5;
sbit P26 = P2^6;
sbit P27 = P2^7;
sbit P30 = P3^0;
sbit P31 = P3^1;
sbit P32 = P3^2;
sbit P33 = P3^3;
sbit P34 = P3^4;
sbit P35 = P3^5;
sbit P36 = P3^6;
sbit P37 = P3^7;
sbit P40 = P4^0;
sbit P41 = P4^1;
sbit P42 = P4^2;
sbit P43 = P4^3;
sbit P44 = P4^4;
sbit P45 = P4^5;
sbit P46 = P4^6;
sbit P47 = P4^7;
sbit P50 = P5^0;
sbit P51 = P5^1;
sbit P52 = P5^2;
sbit P53 = P5^3;
sbit P54 = P5^4;
sbit P55 = P5^5;
sbit P56 = P5^6;
sbit P57 = P5^7;

#define I2CCFG          (*(unsigned char volatile xdata *)0xfe80)
#define I2CMSCR         (*(unsigned char volatile xdata *)0xfe81)
#define I2CMSST         (*(unsigned char volatile xdata *)0xfe82)
#define I2CSLCR         (*(unsigned char volatile xdata *)0xfe83)
#define I2CSLST         (*(unsigned char volatile xdata *)0xfe84)
#define I2CSLADR      (*(unsigned char volatile xdata *)0xfe85)
#define I2CTXD          (*(unsigned char volatile xdata *)0xfe86)
#define I2CRXD          (*(unsigned char volatile xdata *)0xfe87)
#define I2CMSAUX      (*(unsigned char volatile xdata *)0xfe88)

sbit SDA = P1^4;
sbit SCL = P1^5;

/***********************************************************/

#define DIS_DOT   0x20
#define DIS_BLACK   0x10
#define DIS_      0x11

#define SLAW    0xA2
#define SLAR    0xA3


/****************************** 用户定义宏 ***********************************/

#define Timer0_Reload   (65536UL -(MAIN_Fosc / 1000))       //Timer 0 中断频率, 1000次/秒

/*****************************************************************************/


/*************本地常量声明    **************/
u8 code t_display[]={                     //标准字库
//   0    1    2    3    4    5    6    7    8    9    A    B    C    D    E    F
    0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71,
//black-   H    J    K    L    N    o   P    U   t    G    Q    r   M    y
    0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e,
    0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46};    //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1

u8 code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};      //位码


/*************本地变量声明    **************/

u8LED8;      //显示缓冲
u8display_index;//显示位索引
bit B_1ms;          //1ms标志

u8 IO_KeyState, IO_KeyState1, IO_KeyHoldCnt;    //行列键盘变量
u8KeyHoldCnt; //键按下计时
u8KeyCode;    //给用户使用的键码
u8cnt50ms;

u8hour,minute,second; //RTC变量
u16 msecond;


/*************本地函数声明    **************/
void    IO_KeyScan(void);   //50ms call
void    WriteNbyte(u8 addr, u8 *p, u8 number);
void    ReadNbyte( u8 addr, u8 *p, u8 number);
void    DisplayRTC(void);
void    ReadRTC(void);
void    WriteRTC(void);


/**********************************************/
void main(void)
{
    u8i;

    P0M1 = 0x00;   P0M0 = 0x00;   //设置为准双向口
    P1M1 = 0x00;   P1M0 = 0x00;   //设置为准双向口
    P2M1 = 0x00;   P2M0 = 0x00;   //设置为准双向口
    P3M1 = 0x00;   P3M0 = 0x00;   //设置为准双向口
    P4M1 = 0x00;   P4M0 = 0x00;   //设置为准双向口
    P5M1 = 0x20;   P5M0 = 0x00;   //设置为准双向口, 设置P5.5(蜂鸣器)高阻输入
    P6M1 = 0x00;   P6M0 = 0x00;   //设置为准双向口
    P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口
   
    display_index = 0;
   
    for(i=0; i<8; i++)LED8 = 0x10; //上电消隐
   
    AUXR = 0x80;    //Timer0 set as 1T, 16 bits timer auto-reload,
    TH0 = (u8)(Timer0_Reload / 256);
    TL0 = (u8)(Timer0_Reload % 256);
    ET0 = 1;    //Timer0 interrupt enable
    TR0 = 1;    //Tiner0 run

    P_SW2 |= 0x80;
    I2CCFG = 0xe0;                  //使能I2C主机模式
    I2CMSST = 0x00;

    EA = 1;   //打开总中断
   
    ReadRTC();
    F0 = 0;
    if(second >= 60)    F0 = 1; //错误
    if(minute >= 60)    F0 = 1; //错误
    if(hour   >= 24)    F0 = 1; //错误
    if(F0)//有错误, 默认12:00:00
    {
      second = 0;
      minute = 0;
      hour= 12;
      WriteRTC();
    }

    DisplayRTC();
    LED8 = DIS_;
    LED8 = DIS_;

    KeyHoldCnt = 0; //键按下计时
    KeyCode = 0;    //给用户使用的键码

    IO_KeyState = 0;
    IO_KeyState1 = 0;
    IO_KeyHoldCnt = 0;
    cnt50ms = 0;

    while(1)
    {
      if(B_1ms)   //1ms到
      {
            B_1ms = 0;
            if(++msecond >= 1000)   //1秒到
            {
                msecond = 0;
                ReadRTC();
                DisplayRTC();
            }

            if(++cnt50ms >= 50)   //50ms扫描一次行列键盘
            {
                cnt50ms = 0;
                IO_KeyScan();
            }
            
            if(KeyCode != 0)      //有键按下
            {
                if(KeyCode == 17)   //hour +1
                {
                  if(++hour >= 24)    hour = 0;
                  WriteRTC();
                  DisplayRTC();
                }
                if(KeyCode == 18)   //hour -1
                {
                  if(--hour >= 24)    hour = 23;
                  WriteRTC();
                  DisplayRTC();
                }
                if(KeyCode == 19)   //minute +1
                {
                  second = 0;
                  if(++minute >= 60)minute = 0;
                  WriteRTC();
                  DisplayRTC();
                }
                if(KeyCode == 20)   //minute -1
                {
                  second = 0;
                  if(--minute >= 60)minute = 59;
                  WriteRTC();
                  DisplayRTC();
                }

                KeyCode = 0;
            }

      }
    }
}

/********************** 显示时钟函数 ************************/
void    DisplayRTC(void)
{
    if(hour >= 10)LED8 = hour / 10;
    else            LED8 = DIS_BLACK;
    LED8 = hour % 10;
    LED8 = DIS_;
    LED8 = minute / 10;
    LED8 = minute % 10;
    LED8 = second / 10;
    LED8 = second % 10;
}

/********************** 读RTC函数 ************************/
void    ReadRTC(void)
{
    u8tmp;

    ReadNbyte(2, tmp, 3);
    second = ((tmp >> 4) & 0x07) * 10 + (tmp & 0x0f);
    minute = ((tmp >> 4) & 0x07) * 10 + (tmp & 0x0f);
    hour   = ((tmp >> 4) & 0x03) * 10 + (tmp & 0x0f);
}

/********************** 写RTC函数 ************************/
void    WriteRTC(void)
{
    u8tmp;

    tmp = ((second / 10) << 4) + (second % 10);
    tmp = ((minute / 10) << 4) + (minute % 10);
    tmp = ((hour / 10) << 4) + (hour % 10);
    WriteNbyte(2, tmp, 3);
}


/*****************************************************
    行列键扫描程序
    使用XY查找4x4键的方法,只能单键,速度快

   Y   P04      P05      P06      P07
          |      |      |      |
X         |      |      |      |
P00 ---- K00 ---- K01 ---- K02 ---- K03 ----
          |      |      |      |
P01 ---- K04 ---- K05 ---- K06 ---- K07 ----
          |      |      |      |
P02 ---- K08 ---- K09 ---- K10 ---- K11 ----
          |      |      |      |
P03 ---- K12 ---- K13 ---- K14 ---- K15 ----
          |      |      |      |
******************************************************/


u8 code T_KeyTable = {0,1,2,0,3,0,0,0,4,0,0,0,0,0,0,0};

void IO_KeyDelay(void)
{
    u8 i;
    i = 60;
    while(--i);
}

void IO_KeyScan(void)    //50ms call
{
    u8j;

    j = IO_KeyState1;   //保存上一次状态

    P0 = 0xf0;//X低,读Y
    IO_KeyDelay();
    IO_KeyState1 = P0 & 0xf0;

    P0 = 0x0f;//Y低,读X
    IO_KeyDelay();
    IO_KeyState1 |= (P0 & 0x0f);
    IO_KeyState1 ^= 0xff;   //取反
   
    if(j == IO_KeyState1)   //连续两次读相等
    {
      j = IO_KeyState;
      IO_KeyState = IO_KeyState1;
      if(IO_KeyState != 0)    //有键按下
      {
            F0 = 0;
            if(j == 0)F0 = 1; //第一次按下
            else if(j == IO_KeyState)
            {
                if(++IO_KeyHoldCnt >= 20)   //1秒后重键
                {
                  IO_KeyHoldCnt = 18;
                  F0 = 1;
                }
            }
            if(F0)
            {
                j = T_KeyTable;
                if((j != 0) && (T_KeyTable != 0))
                  KeyCode = (j - 1) * 4 + T_KeyTable + 16;    //计算键码,17~32
            }
      }
      else    IO_KeyHoldCnt = 0;
    }
    P0 = 0xff;
}


/********************** 显示扫描函数 ************************/
void DisplayScan(void)
{   
    P7 = ~T_COM;
    P6 = ~t_display];
    if(++display_index >= 8)    display_index = 0;//8位结束回0
}


/********************** Timer0 1ms中断函数 ************************/
void timer0 (void) interrupt 1
{
    DisplayScan();//1ms扫描显示一位
    B_1ms = 1;      //1ms标志
}


void Wait()
{
    while (!(I2CMSST & 0x40));
    I2CMSST &= ~0x40;
}

void Start()
{
    I2CMSCR = 0x01;                         //发送START命令
    Wait();
}

void SendData(char dat)
{
    I2CTXD = dat;                           //写数据到数据缓冲区
    I2CMSCR = 0x02;                         //发送SEND命令
    Wait();
}

void RecvACK()
{
    I2CMSCR = 0x03;                         //发送读ACK命令
    Wait();
}

char RecvData()
{
    I2CMSCR = 0x04;                         //发送RECV命令
    Wait();
    return I2CRXD;
}

void SendACK()
{
    I2CMSST = 0x00;                         //设置ACK信号
    I2CMSCR = 0x05;                         //发送ACK命令
    Wait();
}

void SendNAK()
{
    I2CMSST = 0x01;                         //设置NAK信号
    I2CMSCR = 0x05;                         //发送ACK命令
    Wait();
}

void Stop()
{
    I2CMSCR = 0x06;                         //发送STOP命令
    Wait();
}

void WriteNbyte(u8 addr, u8 *p, u8 number)/*WordAddress,First Data Address,Byte lenth   */
{
    Start();                              //发送起始命令
    SendData(SLAW);                         //发送设备地址+写命令
    RecvACK();
    SendData(addr);                         //发送存储地址
    RecvACK();
    do
    {
      SendData(*p++);
      RecvACK();
    }
    while(--number);
    Stop();                                 //发送停止命令
}

void ReadNbyte(u8 addr, u8 *p, u8 number)   /*WordAddress,First Data Address,Byte lenth   */
{
    Start();                              //发送起始命令
    SendData(SLAW);                         //发送设备地址+写命令
    RecvACK();
    SendData(addr);                         //发送存储地址
    RecvACK();
    Start();                              //发送起始命令
    SendData(SLAR);                         //发送设备地址+读命令
    RecvACK();
    do
    {
      *p = RecvData();
      p++;
      if(number != 1) SendACK();          //send ACK
    }
    while(--number);
    SendNAK();                              //send no ACK       
    Stop();                                 //发送停止命令
}
/****************************/

haiyang201 发表于 2024-7-22 07:55:36

陈家乐 发表于 2024-7-21 12:12
你可以私下加我,我给你看看?还是你可以给我发个视频,咱俩瞅瞅啥问题? ...

找到问题了 ,就是调整时间时 ,读取时间函数中断有影响,禁掉就好了

陈家乐 发表于 2024-7-22 09:38:05

haiyang201 发表于 2024-7-22 05:55
找到问题了 ,就是调整时间时 ,读取时间函数中断有影响,禁掉就好了

好好好,这么整哈哈

llyymm 发表于 2024-7-22 09:52:35

STC自带的RTC功能还未使用过,简单了解下,以后用到了再仔细学学

陈家乐 发表于 2024-7-23 10:14:26

llyymm 发表于 2024-7-22 07:52
STC自带的RTC功能还未使用过,简单了解下,以后用到了再仔细学学

可以的没问题,用过之后再进行对比
页: 1 2 3 4 [5] 6
查看完整版本: 用软件IIC驱动的PCF8563实时时钟芯片