STC8H 外部中断 1 结构图
上升沿或下降沿触发外部中断 1
工程结构
原理图
源代码
main.c 文件:
sfr TCON = 0x88;
sfr P1 = 0x90; /* Port 1 Register */
sfr P1M1 = 0x91; /* Port 1 Configuration Register 1 */
sfr P1M0 = 0x92; /* Port 1 Configuration Register 0 */
sfr IE = 0xA8;
sfr P3M1 = 0xB1; /* Port 3 Configuration Register 1 */
sfr P3M0 = 0xB2; /* Port 3 Configuration Register 0 */
/** TCON Register Bits Definition */
#define INT1_TCON_IT1 (0x04)
/** P1M1 Register Bit Definition */
#define P12M1 (0x04) /* P1.2 Mode Selection Bit 1 */
/** P1M0 Register Bit Definition */
#define P12M0 (0x04) /* P1.2 Mode Selection Bit 0 */
/** IE Register Bits Definition */
#define IE_EA (0x80)
#define INT1_IE_EX1 (0x04)
/** P3M1 Register Bit Definition */
#define P33M1 (0x08) /* P3.3 Mode Selection Bit 1 */
/** P3M0 Register Bit Definition */
#define P33M0 (0x08) /* P3.3 Mode Selection Bit 0 */
#define setRegisterBit(r, b) r |= b
#define clearRegisterBit(r, b) r &= ~b
#define enableInterrupts() setRegisterBit(IE, IE_EA)
#define disableInterrupts() clearRegisterBit(IE, IE_EA)
sbit LED = P1^2; /* LED Control Bit. 0: On, 1: Off */
/**
* External Interrupt 1
*/
void main() {
/**
* External Interrupt 1 Initialization
*/
clearRegisterBit(TCON, INT1_TCON_IT1); // 上升沿或下降沿触发外部中断 1
setRegisterBit(IE, INT1_IE_EX1); // 允许外部中断 1 请求中断
/**
* IO Initialization
*/
// 将 LED 的控制引脚设置为推挽输出
clearRegisterBit(P1M1, P12M1);
setRegisterBit(P1M0, P12M0);
// 将外部中断 1 所在的输入引脚设置为准双向口
clearRegisterBit(P3M1, P33M1);
clearRegisterBit(P3M0, P33M0);
LED = 1; // LED Off
/**
*
*/
enableInterrupts();
/**
* Do nothing...
*/
while(1) {}
}
void externalInterrupt0InterruptService() interrupt 2 {
LED = !LED;
}
STC-ISP 下载选项
测试
编译工程之后,将程序下载到单片机。然后使用示波器分别测量 LED 的控制引脚(黄色信号波形)和外部中断的输入引脚(天蓝色信号波形)。
- 第一次按下按键 K1,此时产生一个下降沿触发外部中断 1;进入外部中断程序之后,LED 的控制引脚被反转(见下图第一个红色箭头);
- 松开按键,此时产生一个上降沿,再次触发外部中断 1;进入外部中断程序之后,LED 的控制引脚再次被反转(见下图第二个红色箭头)。
模块化
将所有的东西的写在一个文件中,是非常不好的习惯。为了方便代码维护,应当是根据单片机的外设,合理地将其划分、模块化。
工程结构
源文件
新建 stc8h1k08.h 文件,在其中声明寄存器,以及其他:
个人观点:由于历史原因,8051 内核单片机的寄存器分布极其混乱。因此,在声明寄存器时,我是根据寄存器地址来排列的。
#ifndef __STC8H1K08_H
#define __STC8H1K08_H
sfr TCON = 0x88;
sfr P1 = 0x90; /* Port 1 Register */
sfr P1M1 = 0x91; /* Port 1 Configuration Register 1 */
sfr P1M0 = 0x92; /* Port 1 Configuration Register 0 */
sfr IE = 0xA8;
sfr P3M1 = 0xB1; /* Port 3 Configuration Register 1 */
sfr P3M0 = 0xB2; /* Port 3 Configuration Register 0 */
/** TCON Register Bits Definition */
#define INT1_TCON_IT1 (0x04)
/** P1M1 Register Bit Definition */
#define P12M1 (0x04) /* P1.2 Mode Selection Bit 1 */
/** P1M0 Register Bit Definition */
#define P12M0 (0x04) /* P1.2 Mode Selection Bit 0 */
/** IE Register Bits Definition */
#define IE_EA (0x80)
#define INT1_IE_EX1 (0x04)
/** P3M1 Register Bit Definition */
#define P33M1 (0x08) /* P3.3 Mode Selection Bit 1 */
/** P3M0 Register Bit Definition */
#define P33M0 (0x08) /* P3.3 Mode Selection Bit 0 */
typedef enum {
false = 0,
true = !false
} boolean;
#define setRegisterBit(r, b) r |= b
#define clearRegisterBit(r, b) r &= ~b
#define enableInterrupts() setRegisterBit(IE, IE_EA)
#define disableInterrupts() clearRegisterBit(IE, IE_EA)
#endif
新建 int1.h 文件,用于管理外部中断 1。在其中声明一些常量,并使用宏定义封装对外部中断 1 的部分操作:
注:对于不常用的操作,我习惯使用宏定义封装。当然,也可以使用方法封装。
#ifndef __STC8H1K08_INT_1_H
#define __STC8H1K08_INT_1_H
#include "stc8h1k08.h"
typedef enum {
EXTERNAL_INTERRUPT_1_SENSITIVITY_FALL = (uint8_t)0x00,
EXTERNAL_INTERRUPT_1_SENSITIVITY_RISE_AND_FALL = (uint8_t)0x01
} ExternalInterrupt1Sensitivity;
#define int1SetSensitivity(sensitivity) { \
if(sensitivity == EXTERNAL_INTERRUPT_1_SENSITIVITY_FALL) { \
setRegisterBit(TCON, INT1_TCON_IT1); \
} else { \
clearRegisterBit(TCON, INT1_TCON_IT1); \
} \
}
#define int1EnableInterrupt(enable) { \
if(enable == false) { \
clearRegisterBit(IE, INT1_IE_EX1); \
} else { \
setRegisterBit(IE, INT1_IE_EX1); \
} \
}
#endif
新建 int1.c 文件,这里面没什么的,只是一条包含头文件的语句:
注:因为没有使用方法封装对外部中断 1 的操作,所以,该文件是非必须的。
#include "int1.h"
新建 config.h 配置文件,其内容如下:
#ifndef __CONFIG_H
#define __CONFIG_H
#include "stc8h1k08.h"
#include "int1.h"
sbit LED = P1^2; /* LED Control Bit. 0: On, 1: Off */
#endif
修改 main.c 文件:
#include "config.h"
/**
* External Interrupt 1
*/
void main() {
/**
* External Interrupt 1 Initialization
*/
int1SetSensitivity(EXTERNAL_INTERRUPT_1_SENSITIVITY_RISE_AND_FALL); // 上升沿或下降沿均可触发外部中断 1
int1EnableInterrupt(true); // 允许外部中断 1 请求中断
/**
* IO Initialization
*/
// 将 LED 的控制引脚设置为推挽输出
clearRegisterBit(P1M1, P12M1);
setRegisterBit(P1M0, P12M0);
// 将外部中断 1 所在的输入引脚设置为准双向口
clearRegisterBit(P3M1, P33M1);
clearRegisterBit(P3M0, P33M0);
LED = 1; // LED Off
/**
*
*/
enableInterrupts();
/**
* Do nothing...
*/
while(1) {}
}
void int1InterruptService() interrupt 2 {
LED = !LED;
}