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功能还未使用过,简单了解下,以后用到了再仔细学学
可以的没问题,用过之后再进行对比