echo. 发表于 2024-5-29 18:53:36

求求了,有善良可爱的网友可以帮忙看看为啥P1.1不能置为低电平吗


STC8H8K64U芯片,代码希望实现串口收到信号P1.1至于高电平,单片机掉电休眠;再次收到信号单片机被唤醒开始工作,P1.1至于低电平。我的程序现在掉电休眠和唤醒后P1.1都是高电平,求求大佬帮忙看看为啥


#include "STC8h.h"   //包含此头文件后,不需要再包含"reg51.h"头文件
#include "usb.h"   //USB调试及复位所需头文件
#include "intrins.h"
#define MAIN_Fosc       22118400L   //定义主时钟(精确计算115200波特率)
#define Timer0_Reload   (65536UL -(MAIN_Fosc / 1000))       //Timer 0 中断频率, 1000次/秒

#define Baudrate1         115200L
#define UART1_BUF_LENGTH    64

#define INTCLKO_ADDR 0x8FH// 寄存器地址   

// 定义EX4的位掩码(B6,即第7位)
#define EX4_BIT (1 << 6)// 1左移6位,得到0x40

volatile bit wake_flag = 0;

bit B_1ms;      //1ms标志
bit B_TX1_Busy; //发送忙标志
u8TX1_Cnt;    //发送计数
u8RX1_Cnt;    //接收计数
u8RX1_TimeOut;

u8xdata RX1_Buffer; //接收缓冲

//USB调试及复位所需定义
char *USER_DEVICEDESC = NULL;
char *USER_PRODUCTDESC = NULL;
char *USER_STCISPCMD = "@STCISP#";                      //设置自动复位到ISP区的用户接口命令

//P3.2口按键复位所需变量
bit Key_Flag;
u16 Key_cnt;

void Timer0_config(void);
void UART1_config(u8 brt);   // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
void UART1_TxByte(u8 dat);
void UART1_Send(u8 addr,u8 *dat,u8 len);
void KeyResetScan(void);

//========================================================================
// 函数: void main(void)
// 描述: 主函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
u16 ccnt=0;
void main(void)
{
    P_SW2 |= 0x80;//扩展寄存器(XFR)访问使能

    RSTFLAG |= 0x04;   //设置硬件复位后需要检测P3.2的状态选择运行区域,否则硬件复位后进入USB下载模式

    P0M1 = 0x00;   P0M0 = 0x00;   //设置为准双向口
    P1M1 = 0x00;   P1M0 = 0x00;   //设置为准双向口
    P2M1 = 0x00;   P2M0 = 0x00;   //设置为准双向口
    P3M1 = 0x00;   P3M0 = 0x00;   //设置为准双向口
    P4M1 = 0x00;   P4M0 = 0x00;   //设置为准双向口
    P5M1 = 0x00;   P5M0 = 0x00;   //设置为准双向口
    P6M1 = 0x00;   P6M0 = 0x00;   //设置为准双向口
    P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口

   
         INTCLKO |=EX4_BIT;          //P3.0使能INT4下降沿中断
         
         ES = 1;
   // usb_init();
   // Timer0_config();
    UART1_config(1);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
   // IE2 |= 0x80;   //IE2相关的中断位操作使能后,需要重新设置EUSB
    EA = 1; //允许总中断
               
    while (1)
    {
                     
                        if (wake_flag==1) // 检查唤醒标志位
      {
            wake_flag = 0; // 清除唤醒标志位,防止重复处理
            P11 = 0;       // 设置P1.1为低电平
            P32 = 0;       // 假设P3.2也需要设置为低电平
      }
      
      if((ccnt%2)==1)
                              { P11=1;
                                        P32=1;
                                        PCON = 0x02; //进入休眠   
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
                                        _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
          _nop_();
                              }else
                              {
                              }
    }
}

void timer0(void) interrupt 1
{
      B_1ms = 1;
}

//========================================================================
// 函数: void Timer0_config(void)
// 描述: Timer0初始化函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2023-9-21
// 备注:
//========================================================================
void Timer0_config(void)
{
    //Timer0初始化
    TMOD &= 0xf0;//16 bits timer auto-reload
    AUXR |= 0x80;//Timer0 set as 1T
    TH0 = (u8)(Timer0_Reload / 256);
    TL0 = (u8)(Timer0_Reload % 256);
    ET0 = 1;    //Timer0 interrupt enable
    TR0 = 1;    //Tiner0 run
}

//========================================================================
// 函数: void UART1_TxByte(u8 dat)
// 描述: 发送一个字节.
// 参数: 无.
// 返回: 无.
// 版本: V1.0, 2014-6-30
//========================================================================
void UART1_TxByte(u8 dat)
{
    B_TX1_Busy = 1;
    SBUF = dat;
    while(B_TX1_Busy);
}

//========================================================================
// 函数: void UART1_Send(u8 addr,u8 *dat,u8 len)
// 描述: 发送一个字节.
// 参数: 无.
// 返回: 无.
// 版本: V1.0, 2014-6-30
//========================================================================
void UART1_Send(u8 addr,u8 *dat,u8 len)
{
    u8 i;

    TB8 = 1;    //地址帧
    UART1_TxByte(addr);

    TB8 = 0;    //数据帧
    for (i=0;i<len;i++)
    {
      UART1_TxByte(dat);
    }
}

//========================================================================
// 函数: SetTimer2Baudraye(u16 dat)
// 描述: 设置Timer2做波特率发生器。
// 参数: dat: Timer2的重装值.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void SetTimer2Baudraye(u16 dat)// 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
    AUXR &= ~(1<<4);    //Timer stop
    AUXR &= ~(1<<3);    //Timer2 set As Timer
    AUXR |=(1<<2);    //Timer2 set as 1T mode
    T2H = dat / 256;
    T2L = dat % 256;
    IE2&= ~(1<<2);    //禁止中断
    AUXR |=(1<<4);    //Timer run enable
}

//========================================================================
// 函数: void UART1_config(u8 brt)
// 描述: UART1初始化函数。
// 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_config(u8 brt)    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
    /*********** 波特率使用定时器2 *****************/
    if(brt == 2)
    {
      AUXR |= 0x01;       //S1 BRT Use Timer2;
      SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate1);
    }

    /*********** 波特率使用定时器1 *****************/
    else
    {
      TR1 = 0;
      AUXR &= ~0x01;      //S1 BRT Use Timer1;
      AUXR |=(1<<6);    //Timer1 set as 1T mode
      TMOD &= ~(1<<6);    //Timer1 set As Timer
      TMOD &= ~0x30;      //Timer1_16bitAutoReload;
      TH1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) / 256);
      TL1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) % 256);
      ET1 = 0;    //禁止中断
      INTCLKO &= ~0x02;   //不输出时钟
      TR1= 1;
    }
    /*************************************************/

    SCON = (SCON & 0x3f) | 0xc0;    //UART1模式, 0x00: 同步移位输出, 0x40: 8位数据,可变波特率, 0x80: 9位数据,固定波特率, 0xc0: 9位数据,可变波特率
//   SM2 = 1;    //允许多机通信
//PS= 1;    //高优先级中断
    ES= 1;    //允许中断
    REN = 1;    //允许接收
//   SADDR = 0x53;   //从机地址
   // SADEN = 0xf0;   //高4位匹配
    P_SW1 &= 0x3f;
    P_SW1 |= 0x00;      //UART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7, 0xC0: P4.3 P4.4

    RX1_TimeOut = 0;
    B_TX1_Busy = 0;
    TX1_Cnt = 0;
    RX1_Cnt = 0;
}


//========================================================================
// 函数: void UART1_int (void) interrupt UART1_VECTOR
// 描述: UART1中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_int (void) interrupt 4
{
    if(RI)
    {
                        ccnt++;
          RI = 0;
      RX1_Buffer = SBUF;
      
    }

    if(TI)
    {
      TI = 0;
      B_TX1_Busy = 0;
    }
}

//========================================================================
// 函数: void delay_ms(u8 ms)
// 描述: 延时函数。
// 参数: ms,要延时的ms数, 这里只支持1~255ms. 自动适应主时钟.
// 返回: none.
// 版本: VER1.0
// 日期: 2022-6-3
// 备注:
//========================================================================
void delay_ms(u8 ms)
{
    u16 i;
    do{
      i = MAIN_Fosc / 10000;
      while(--i);   //10T per loop
    }while(--ms);
}

//========================================================================
// 函数: void KeyResetScan(void)
// 描述: P3.2口按键长按1秒触发软件复位,进入USB下载模式。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2022-6-11
// 备注:
//========================================================================
void KeyResetScan(void)
{
    if(!P32)
    {
      if(!Key_Flag)
      {
            Key_cnt++;
            if(Key_cnt >= 1000)                //连续1000ms有效按键检测
            {
                Key_Flag = 1;                //设置按键状态,防止重复触发

                USBCON = 0x00;      //清除USB设置
                USBCLK = 0x00;
                IRC48MCR = 0x00;
               
                delay_ms(10);
                IAP_CONTR = 0x60;   //触发软件复位,从ISP开始执行
                while (1);
            }
      }
    }
    else
    {
      Key_cnt = 0;
      Key_Flag = 0;
    }
}
void UART2_int (void) interrupt 17
{
P10 = !P10;
}

void UART3_int (void) interrupt 18
{

    // 设置EX4位为1(下降沿触发)   
    INTCLKO |= EX4_BIT;   // 使用位或操作来设置位   
    // 清除EX4位
          INTCLKO &= ~EX4_BIT;// 使用位与操作和取反操作来清除位
          // 设置唤醒标志位
    wake_flag = 1;
}

乘风飞扬 发表于 2024-5-30 10:36:01

你的程序写的很乱啊。
初始化里面开启了INT4中断,但是没写INT4中断函数。
拉低P1.1的代码通过判断“wake_flag”标志位为1才会执行,
而“wake_flag”是在UART3_int函数里面设置的,
而这个函数后面的“interrupt 18”用的又是UART4的中断向量。
初始化里面又只对UART1进行初始化,
UART3、UART4都没有进行初始化。



神农鼎 发表于 2024-5-30 10:45:33

先用 STC8H8K64U实验箱中的演示程序学习

深圳国芯人工智能有限公司-实验箱 (stcai.com)

页: [1]
查看完整版本: 求求了,有善良可爱的网友可以帮忙看看为啥P1.1不能置为低电平吗