2025-02-16
单片机
00

目录

STC12C5A60S2.h
串口
DS18B20
PWM
定时器0中断
NEC
eeprom flash

https://blog.csdn.net/lzxiaotu/article/details/119380425

STC12C5A60S2.h

h
#ifndef __STC12C5A60S2_H_ #define __STC12C5A60S2_H_ typedef unsigned int uint ; typedef unsigned char uchar ; ///////////////////////////////////////////////// //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机内核特殊功能寄存器 C51 Core SFRs // 7 6 5 4 3 2 1 0 Reset Value sfr ACC = 0xE0; //Accumulator 0000,0000 sfr B = 0xF0; //B Register 0000,0000 sfr PSW = 0xD0; //Program Status Word CY AC F0 RS1 RS0 OV F1 P 0000,0000 //----------------------------------- sbit CY = PSW^7; sbit AC = PSW^6; sbit F0 = PSW^5; sbit RS1 = PSW^4; sbit RS0 = PSW^3; sbit OV = PSW^2; sbit P = PSW^0; //----------------------------------- sfr SP = 0x81; //Stack Pointer 0000,0111 sfr DPL = 0x82; //Data Pointer Low Byte 0000,0000 sfr DPH = 0x83; //Data Pointer High Byte 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机系统管理特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr PCON = 0x87; //Power Control SMOD SMOD0 LVDF POF GF1 GF0 PD IDL 0001,0000 // 7 6 5 4 3 2 1 0 Reset Value sfr AUXR = 0x8E; //Auxiliary Register T0x12 T1x12 UART_M0x6 BRTR S2SMOD BRTx12 EXTRAM S1BRS 0000,0000 //----------------------------------- sfr AUXR1 = 0xA2; //Auxiliary Register 1 - PCA_P4 SPI_P4 S2_P4 GF2 ADRJ - DPS 0000,0000 /* PCA_P4: 0, 缺省PCA 在P1 口 1,PCA/PWM 从P1 口切换到P4 口: ECI 从P1.2 切换到P4.1 口, PCA0/PWM0 从P1.3 切换到P4.2 口 PCA1/PWM1 从P1.4 切换到P4.3 口 SPI_P4: 0, 缺省SPI 在P1 口 1,SPI 从P1 口切换到P4 口: SPICLK 从P1.7 切换到P4.3 口 MISO 从P1.6 切换到P4.2 口 MOSI 从P1.5 切换到P4.1 口 SS 从P1.4 切换到P4.0 口 S2_P4: 0, 缺省UART2 在P1 口 1,UART2 从P1 口切换到P4 口: TxD2 从P1.3 切换到P4.3 口 RxD2 从P1.2 切换到P4.2 口 GF2: 通用标志位 ADRJ: 0, 10 位A/D 转换结果的高8 位放在ADC_RES 寄存器, 低2 位放在ADC_RESL 寄存器 1,10 位A/D 转换结果的最高2 位放在ADC_RES 寄存器的低2 位, 低8 位放在ADC_RESL 寄存器 DPS: 0, 使用缺省数据指针DPTR0 1,使用另一个数据指针DPTR1 */ //----------------------------------- sfr WAKE_CLKO = 0x8F; //附加的 SFR WAK1_CLKO /* 7 6 5 4 3 2 1 0 Reset Value PCAWAKEUP RXD_PIN_IE T1_PIN_IE T0_PIN_IE LVD_WAKE _ T1CLKO T0CLKO 0000,0000B b7 - PCAWAKEUP : PCA 中断可唤醒 powerdown。 b6 - RXD_PIN_IE : 当 P3.0(RXD) 下降沿置位 RI 时可唤醒 powerdown(必须打开相应中断)。 b5 - T1_PIN_IE : 当 T1 脚下降沿置位 T1 中断标志时可唤醒 powerdown(必须打开相应中断)。 b4 - T0_PIN_IE : 当 T0 脚下降沿置位 T0 中断标志时可唤醒 powerdown(必须打开相应中断)。 b3 - LVD_WAKE : 当 CMPIN 脚低电平置位 LVD 中断标志时可唤醒 powerdown(必须打开相应中断)。 b2 - b1 - T1CLKO : 允许 T1CKO(P3.5) 脚输出 T1 溢出脉冲,Fck1 = 1/2 T1 溢出率 b0 - T0CLKO : 允许 T0CKO(P3.4) 脚输出 T0 溢出脉冲,Fck0 = 1/2 T1 溢出率 */ //----------------------------------- sfr CLK_DIV = 0x97; //Clock Divder - - - - - CLKS2 CLKS1 CLKS0 xxxx,x000 //----------------------------------- sfr BUS_SPEED = 0xA1; //Stretch register - - ALES1 ALES0 - RWS2 RWS1 RWS0 xx10,x011 /* ALES1 and ALES0: 00 : The P0 address setup time and hold time to ALE negative edge is one clock cycle 01 : The P0 address setup time and hold time to ALE negative edge is two clock cycles. 10 : The P0 address setup time and hold time to ALE negative edge is three clock cycles. (default) 11 : The P0 address setup time and hold time to ALE negative edge is four clock cycles. RWS2,RWS1,RWS0: 000 : The MOVX read/write pulse is 1 clock cycle. 001 : The MOVX read/write pulse is 2 clock cycles. 010 : The MOVX read/write pulse is 3 clock cycles. 011 : The MOVX read/write pulse is 4 clock cycles. (default) 100 : The MOVX read/write pulse is 5 clock cycles. 101 : The MOVX read/write pulse is 6 clock cycles. 110 : The MOVX read/write pulse is 7 clock cycles. 111 : The MOVX read/write pulse is 8 clock cycles. */ //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机中断特殊功能寄存器 //有的中断控制、中断标志位散布在其它特殊功能寄存器中,这些位在位地址中定义 //其中有的位无位寻址能力,请参阅 新一代 1T 8051系列 单片机中文指南 // 7 6 5 4 3 2 1 0 Reset Value sfr IE = 0xA8; //中断控制寄存器 EA ELVD EADC ES ET1 EX1 ET0 EX0 0x00,0000 //----------------------- sbit EA = IE^7; sbit ELVD = IE^6; //低压监测中断允许位 sbit EADC = IE^5; //ADC 中断允许位 sbit ES = IE^4; sbit ET1 = IE^3; sbit EX1 = IE^2; sbit ET0 = IE^1; sbit EX0 = IE^0; //----------------------- sfr IE2 = 0xAF; //Auxiliary Interrupt - - - - - - ESPI ES2 0000,0000B //----------------------- // 7 6 5 4 3 2 1 0 Reset Value sfr IP = 0xB8; //中断优先级低位 PPCA PLVD PADC PS PT1 PX1 PT0 PX0 0000,0000 //-------- sbit PPCA = IP^7; //PCA 模块中断优先级 sbit PLVD = IP^6; //低压监测中断优先级 sbit PADC = IP^5; //ADC 中断优先级 sbit PS = IP^4; sbit PT1 = IP^3; sbit PX1 = IP^2; sbit PT0 = IP^1; sbit PX0 = IP^0; //----------------------- // 7 6 5 4 3 2 1 0 Reset Value sfr IPH = 0xB7; //中断优先级高位 PPCAH PLVDH PADCH PSH PT1H PX1H PT0H PX0H 0000,0000 sfr IP2 = 0xB5; // - - - - - - PSPI PS2 xxxx,xx00 sfr IPH2 = 0xB6; // - - - - - - PSPIH PS2H xxxx,xx00 //----------------------- //新一代 1T 8051系列 单片机I/O 口特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr P0 = 0x80; //8 bitPort0 P0.7 P0.6 P0.5 P0.4 P0.3 P0.2 P0.1 P0.0 1111,1111 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; sfr P0M0 = 0x94; // 0000,0000 sfr P0M1 = 0x93; // 0000,0000 sfr P1 = 0x90; //8 bitPort1 P1.7 P1.6 P1.5 P1.4 P1.3 P1.2 P1.1 P1.0 1111,1111 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; sfr P1M0 = 0x92; // 0000,0000 sfr P1M1 = 0x91; // 0000,0000 sfr P1ASF = 0x9D; //P1 analog special function sfr P2 = 0xA0; //8 bitPort2 P2.7 P2.6 P2.5 P2.4 P2.3 P2.2 P2.1 P2.0 1111,1111 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; sfr P2M0 = 0x96; // 0000,0000 sfr P2M1 = 0x95; // 0000,0000 sfr P3 = 0xB0; //8 bitPort3 P3.7 P3.6 P3.5 P3.4 P3.3 P3.2 P3.1 P3.0 1111,1111 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; sfr P3M0 = 0xB2; // 0000,0000 sfr P3M1 = 0xB1; // 0000,0000 sfr P4 = 0xC0; //8 bitPort4 P4.7 P4.6 P4.5 P4.4 P4.3 P4.2 P4.1 P4.0 1111,1111 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; sfr P4M0 = 0xB4; // 0000,0000 sfr P4M1 = 0xB3; // 0000,0000 // 7 6 5 4 3 2 1 0 Reset Value sfr P4SW = 0xBB; //Port-4 switch - LVD_P4.6 ALE_P4.5 NA_P4.4 - - - - x000,xxxx sfr P5 = 0xC8; //8 bitPort5 - - - - P5.3 P5.2 P5.1 P5.0 xxxx,1111 sbit P50 = P5^0; sbit P51 = P5^1; sbit P52 = P5^2; sbit P53 = P5^3; sfr P5M0 = 0xCA; // 0000,0000 sfr P5M1 = 0xC9; // 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机定时器特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr TCON = 0x88; //T0/T1 Control TF1 TR1 TF0 TR0 IE1 IT1 IE0 IT0 0000,0000 //----------------------------------- sbit TF1 = TCON^7; sbit TR1 = TCON^6; sbit TF0 = TCON^5; sbit TR0 = TCON^4; sbit IE1 = TCON^3; sbit IT1 = TCON^2; sbit IE0 = TCON^1; sbit IT0 = TCON^0; //----------------------------------- sfr TMOD = 0x89; //T0/T1 Modes GATE1 C/T1 M1_1 M1_0 GATE0 C/T0 M0_1 M0_0 0000,0000 sfr TL0 = 0x8A; //T0 Low Byte 0000,0000 sfr TH0 = 0x8C; //T0 High Byte 0000,0000 sfr TL1 = 0x8B; //T1 Low Byte 0000,0000 sfr TH1 = 0x8D; //T1 High Byte 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机串行口特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr SCON = 0x98; //Serial Control SM0/FE SM1 SM2 REN TB8 RB8 TI RI 0000,0000 //----------------------------------- sbit SM0 = SCON^7; //SM0/FE sbit SM1 = SCON^6; sbit SM2 = SCON^5; sbit REN = SCON^4; sbit TB8 = SCON^3; sbit RB8 = SCON^2; sbit TI = SCON^1; sbit RI = SCON^0; //----------------------------------- sfr SBUF = 0x99; //Serial Data Buffer xxxx,xxxx sfr SADEN = 0xB9; //Slave Address Mask 0000,0000 sfr SADDR = 0xA9; //Slave Address 0000,0000 //----------------------------------- // 7 6 5 4 3 2 1 0 Reset Value sfr S2CON = 0x9A; //S2 Control S2SM0 S2SM1 S2SM2 S2REN S2TB8 S2RB8 S2TI S2RI 00000000B sfr S2BUF = 0x9B; //S2 Serial Buffer xxxx,xxxx sfr BRT = 0x9C; //S2 Baud-Rate Timer 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机看门狗定时器特殊功能寄存器 sfr WDT_CONTR = 0xC1; //Watch-Dog-Timer Control register // 7 6 5 4 3 2 1 0 Reset Value // WDT_FLAG - EN_WDT CLR_WDT IDLE_WDT PS2 PS1 PS0 xx00,0000 //----------------------- //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机PCA/PWM 特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr CCON = 0xD8; //PCA 控制寄存器。 CF CR - - - - CCF1 CCF0 00xx,xx00 //----------------------- sbit CF = CCON^7; //PCA计数器溢出标志,由硬件或软件置位,必须由软件清0。 sbit CR = CCON^6; //1:允许 PCA 计数器计数, 必须由软件清0。 //- //- sbit CCF1 = CCON^1; //PCA 模块1 中断标志, 由硬件置位, 必须由软件清0。 sbit CCF0 = CCON^0; //PCA 模块0 中断标志, 由硬件置位, 必须由软件清0。 //----------------------- sfr CMOD = 0xD9; //PCA 工作模式寄存器。 CIDL - - - CPS2 CPS1 CPS0 ECF 0xxx,x000 /* CIDL: idle 状态时 PCA 计数器是否继续计数, 0: 继续计数, 1: 停止计数。 CPS2: PCA 计数器脉冲源选择位 2。 CPS1: PCA 计数器脉冲源选择位 1。 CPS0: PCA 计数器脉冲源选择位 0。 CPS2 CPS1 CPS0 0 0 0 系统时钟频率 fosc/12。 0 0 1 系统时钟频率 fosc/2。 0 1 0 Timer0 溢出。 0 1 1 由 ECI/P3.4 脚输入的外部时钟,最大 fosc/2。 1 0 0 系统时钟频率, Fosc/1 1 0 1 系统时钟频率/4,Fosc/4 1 1 0 系统时钟频率/6,Fosc/6 1 1 1 系统时钟频率/8,Fosc/8 ECF: PCA计数器溢出中断允许位, 1--允许 CF(CCON.7) 产生中断。 */ //----------------------- sfr CL = 0xE9; //PCA 计数器低位 0000,0000 sfr CH = 0xF9; //PCA 计数器高位 0000,0000 //----------------------- // 7 6 5 4 3 2 1 0 Reset Value sfr CCAPM0 = 0xDA; //PCA 模块0 PWM 寄存器 - ECOM0 CAPP0 CAPN0 MAT0 TOG0 PWM0 ECCF0 x000,0000 sfr CCAPM1 = 0xDB; //PCA 模块1 PWM 寄存器 - ECOM1 CAPP1 CAPN1 MAT1 TOG1 PWM1 ECCF1 x000,0000 //ECOMn = 1:允许比较功能。 //CAPPn = 1:允许上升沿触发捕捉功能。 //CAPNn = 1:允许下降沿触发捕捉功能。 //MATn = 1:当匹配情况发生时, 允许 CCON 中的 CCFn 置位。 //TOGn = 1:当匹配情况发生时, CEXn 将翻转。 //PWMn = 1:将 CEXn 设置为 PWM 输出。 //ECCFn = 1:允许 CCON 中的 CCFn 触发中断。 //ECOMn CAPPn CAPNn MATn TOGn PWMn ECCFn // 0 0 0 0 0 0 0 0x00 未启用任何功能。 // x 1 0 0 0 0 x 0x21 16位CEXn上升沿触发捕捉功能。 // x 0 1 0 0 0 x 0x11 16位CEXn下降沿触发捕捉功能。 // x 1 1 0 0 0 x 0x31 16位CEXn边沿(上、下沿)触发捕捉功能。 // 1 0 0 1 0 0 x 0x49 16位软件定时器。 // 1 0 0 1 1 0 x 0x4d 16位高速脉冲输出。 // 1 0 0 0 0 1 0 0x42 8位 PWM。 //ECOMn CAPPn CAPNn MATn TOGn PWMn ECCFn // 0 0 0 0 0 0 0 0x00 无此操作 // 1 0 0 0 0 1 0 0x42 普通8位PWM, 无中断 // 1 1 0 0 0 1 1 0x63 PWM输出由低变高可产生中断 // 1 0 1 0 0 1 1 0x53 PWM输出由高变低可产生中断 // 1 1 1 0 0 1 1 0x73 PWM输出由低变高或由高变低都可产生中断 //----------------------- sfr CCAP0L = 0xEA; //PCA 模块 0 的捕捉/比较寄存器低 8 位。 0000,0000 sfr CCAP0H = 0xFA; //PCA 模块 0 的捕捉/比较寄存器高 8 位。 0000,0000 sfr CCAP1L = 0xEB; //PCA 模块 1 的捕捉/比较寄存器低 8 位。 0000,0000 sfr CCAP1H = 0xFB; //PCA 模块 1 的捕捉/比较寄存器高 8 位。 0000,0000 //----------------------- // 7 6 5 4 3 2 1 0 Reset Value sfr PCA_PWM0 = 0xF2; //PCA 模块0 PWM 寄存器。 - - - - - - EPC0H EPC0L xxxx,xx00 sfr PCA_PWM1 = 0xF3; //PCA 模块1 PWM 寄存器。 - - - - - - EPC1H EPC1L xxxx,xx00 //PCA_PWMn: 7 6 5 4 3 2 1 0 // - - - - - - EPCnH EPCnL //B7-B2: 保留 //B1(EPCnH): 在 PWM 模式下,与 CCAPnH 组成 9 位数。 //B0(EPCnL): 在 PWM 模式下,与 CCAPnL 组成 9 位数。 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机 ADC 特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr ADC_CONTR = 0xBC; //A/D 转换控制寄存器 ADC_POWER SPEED1 SPEED0 ADC_FLAG ADC_START CHS2 CHS1 CHS0 0000,0000 sfr ADC_RES = 0xBD; //A/D 转换结果高8位 ADCV.9 ADCV.8 ADCV.7 ADCV.6 ADCV.5 ADCV.4 ADCV.3 ADCV.2 0000,0000 sfr ADC_RESL = 0xBE; //A/D 转换结果低2位 ADCV.1 ADCV.0 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机 SPI 特殊功能寄存器 // 7 6 5 4 3 2 1 0 Reset Value sfr SPCTL = 0xCE; //SPI Control Register SSIG SPEN DORD MSTR CPOL CPHA SPR1 SPR0 0000,0100 sfr SPSTAT = 0xCD; //SPI Status Register SPIF WCOL - - - - - - 00xx,xxxx sfr SPDAT = 0xCF; //SPI Data Register 0000,0000 //-------------------------------------------------------------------------------- //新一代 1T 8051系列 单片机 IAP/ISP 特殊功能寄存器 sfr IAP_DATA = 0xC2; sfr IAP_ADDRH = 0xC3; sfr IAP_ADDRL = 0xC4; // 7 6 5 4 3 2 1 0 Reset Value sfr IAP_CMD = 0xC5; //IAP Mode Table 0 - - - - - MS1 MS0 0xxx,xx00 sfr IAP_TRIG = 0xC6; sfr IAP_CONTR = 0xC7; //IAP Control Register IAPEN SWBS SWRST CFAIL - WT2 WT1 WT0 0000,x000 //-------------------------------------------------------------------------------- ///////////////////////////////////////////////// #endif

串口

uart.h

h
#ifndef UART_H_ #define UART_H_ #include <stc12c5a60s2.h> void UART_Initialize(void); void UART_SendString(char *str); #endif /* UART_H_ */

uart.c

c
#include "uart.h" /* 串口初始化,晶振11.0592,波特率9600 */ void UART_Initialize(void) //[email protected] { PCON &= 0x7F; // 波特率不倍速 SCON = 0x50; // 8位数据,可变波特率 AUXR |= 0x04; // 独立波特率发生器时钟为Fosc,即1T BRT = 0xDC; // 设定独立波特率发生器重装值 AUXR |= 0x01; // 串口1选择独立波特率发生器为波特率发生器 AUXR |= 0x10; // 启动独立波特率发生器 } /* 向串口发送单个字符 */ void UART_SendChar(char ch) { SBUF = ch; while (!TI); TI = 0; } /* 向串口发送字符串 */ void UART_SendString(char *str) { while (*str != '\0') { SBUF = *str++; while (!TI); TI = 0; } }

DS18B20

DS18B20.h

h
#ifndef DS18B20_H #define DS18B20_H #include <STC12C5A60S2.H> // DS18B20 IO设置 sbit DS18B20_DATA_PIN = P1 ^ 0; // 函数声明 extern void DS18B20_Delay(unsigned int delayTime); extern void DS18B20_WriteByte(unsigned char datax); extern unsigned char DS18B20_ReadByte(void); extern bit DS18B20_Init(void); extern unsigned int DS18B20_GetTemperature(void); // 变量声明 extern unsigned char temperatureFlag; #endif // DS18B20_H

DS18B20.c

c
#include "ds18b20.h" #include <intrins.h> // 定义变量 unsigned char temperatureFlag = 0; //**************************************************** // DS18B20延时函数 //**************************************************** void DS18B20_Delay(unsigned int delayTime) { unsigned int i; for (i = 0; i < delayTime; i++) { _nop_(); } } //**************************************************** // DS18B20写1字节 //**************************************************** void DS18B20_WriteByte(unsigned char datax) { unsigned char i; for (i = 0; i < 8; i++) { DS18B20_DATA_PIN = 0; _nop_(); // 延时>1us _nop_(); DS18B20_DATA_PIN = datax & 0x01; // 先写低位 datax >>= 1; DS18B20_Delay(70); // 延时60~120us DS18B20_DATA_PIN = 1; // 释放总线 _nop_(); // 延时>1us _nop_(); } } //**************************************************** // DS18B20读1字节 //**************************************************** unsigned char DS18B20_ReadByte(void) { unsigned char datax, i; for (i = 0; i < 8; i++) { DS18B20_DATA_PIN = 0; _nop_(); // 延时>1us _nop_(); DS18B20_DATA_PIN = 1; // 释放总线 _nop_(); // 延时>1us _nop_(); datax >>= 1; if (DS18B20_DATA_PIN == 1) { datax |= 0x80; } else { datax &= 0x7F; } DS18B20_Delay(70); // 延时60~120us } return datax; } //**************************************************** // DS18B20初始化 //**************************************************** bit DS18B20_Init(void) { bit flagExist = 1; DS18B20_DATA_PIN = 1; // 释放总线 _nop_(); // 延时>1us _nop_(); DS18B20_DATA_PIN = 0; DS18B20_Delay(500); // 延时480~960us DS18B20_DATA_PIN = 1; // 释放总线 DS18B20_Delay(30); // 延时15~60us flagExist = DS18B20_DATA_PIN; DS18B20_Delay(70); // 延时60~240us DS18B20_DATA_PIN = 1; // 释放总线 return flagExist; } //********************************************************** // 读取温度函数,返回温度的绝对值,并标注temperatureFlag,temperatureFlag=1表示负,temperatureFlag=0表示正 //********************************************************** unsigned int DS18B20_GetTemperature(void) { float temperature; unsigned char lsb, msb; unsigned int tempValue; if (DS18B20_Init() == 0) // 初始化 { DS18B20_WriteByte(0xCC); // 忽略ROM指令 DS18B20_WriteByte(0x44); // 温度转换指令 if (DS18B20_Init() == 0) // 初始化 { DS18B20_WriteByte(0xCC); // 忽略ROM指令 DS18B20_WriteByte(0xBE); // 读暂存器指令 lsb = DS18B20_ReadByte(); // 读取到的第一个字节为温度LSB msb = DS18B20_ReadByte(); // 读取到的第一个字节为温度MSB tempValue = msb; // 先把高八位有效数据赋于tempValue tempValue <<= 8; // 把以上8位数据从tempValue低八位移到高八位 tempValue |= lsb; // 两字节合成一个整型变量 if (tempValue > 0xFFF) { temperatureFlag = 1; // 温度为负数 tempValue = (~tempValue) + 1; } else { temperatureFlag = 0; // 温度为正或者0 } temperature = tempValue * 0.0625; // 得到真实十进制温度值 tempValue = temperature * 10 + 0.5; // 放大十倍,同时进行四舍五入 } } return tempValue; }

使用:

wendu_now = DS18B20_GetTemperature();//获取温度 扩大了10倍 memset(disp, 0, sizeof(disp)); sprintf(disp, "%d.%dC", wendu_now / 10, wendu_now % 10); OLED_ShowString(0, 0, disp);

PWM

c
/********************************** *函数名称:PWM_Init(void) *输入 :无 *输出 :无 *调用说明:外部调用 *函数说明:PWM模块初始化 ***********************************/ void PWM_Init(void) { CCON = 0; //PCA初始化 CMOD = 0x00; //空闲时不计数,不产生中断,时钟源为Sysclk/12,PWM频率大约为4KHz CL = 0x00; //PCA低8位清零 CH = 0x00; //PCA高8位清零 CCAPM0 = 0x42; //8位PWM模式,无中断 CCAP0H = 0xc0; //PWM0占空比(调节此处值调节PWM占空比) CCAP0L = 0xc0; //PWM0占空比(调节此处值调节PWM占空比) CCAPM1 = 0x42; //8位PWM模式,无中断 CCAP1H = 0x40; //PWM1占空比(调节此处值调节PWM占空比) CCAP1L = 0x40; //PWM1占空比(调节此处值调节PWM占空比) CR = 1; //启动PCA计数器 } /********************************** *函数名称:PWM_SetDutyCycle(unsigned char dutyCycle0, unsigned char dutyCycle1) *输入 :占空比输入1 unsigned char dutyCycle0(0-255),占空比输入2 unsigned char dutyCycle1(0-255) *输出 :无 *调用说明:外部调用 *函数说明:占空比设置 ***********************************/ void PWM_SetDutyCycle(unsigned char dutyCycle0, unsigned char dutyCycle1) { dutyCycle0 = ~dutyCycle0; dutyCycle1 = ~dutyCycle1; CCAP0H = dutyCycle1; //设置比较值 CCAP0L = dutyCycle1; CCAP1H = dutyCycle0; //设置比较值 CCAP1L = dutyCycle0; }

定时器0中断

c
void Timer0Init(void) /* 50毫秒@11.0592MHz */ { AUXR &= 0x7F; /* 定时器时钟12T模式 */ TMOD &= 0xF0; /* 设置定时器模式 */ TMOD |= 0x01; /* 设置定时器模式 */ TL0 = 0x00; /* 设置定时初值 */ TH0 = 0x4C; /* 设置定时初值 */ TF0 = 0; /* 清除TF0标志 */ ET0 = 1; /* 允许定时器0中断 */ TR0 = 0; /* 定时器0开始计时 */ } void time0_isr() interrupt 1 { TL0 = 0x00; /* 设置定时初值 */ TH0 = 0x4C; /* 设置定时初值 */ }

NEC

image.png

nec.h

h
#ifndef NEC_IR_H #define NEC_IR_H #include <stc12c5a60s2.h> #include <intrins.h> // 宏定义和常量 #define MAIN_FOSC 11059200L // 变量声明 extern unsigned char ir_time; // 检测红外高电平持续时间(脉宽) extern unsigned char ir_code[4]; // 用于存储分离出来的4个字节的数据(用户码2个字节+键值码2个字节) extern unsigned char ir_data[33]; // 用于存储红外的33位数据(第一位为引导码用户码16+键值码16) extern bit ir_process_complete, ir_ok; // 用于标识红外接收和脉宽检测的状态 // 定时器0初始化函数 void Timer0_Init(void); // 外部中断0初始化函数 void ExternalInterrupt0_Init(void); // 提取红外数据并进行解码 void DecodeIR(void); #endif // NEC_IR_H

nec.c

c
#include "nec.h" sbit IR_RECEIVER = P3 ^ 2; // 红外接收器 // 变量定义 unsigned char ir_time; // 检测红外高电平持续时间(脉宽) unsigned char ir_code[4]; // 用于存储分离出来的4个字节的数据 unsigned char ir_data[33]; // 用于存储红外的33位数据 bit ir_process_complete, ir_ok; // 用于标识红外接收和脉宽检测的状态 // 定时器0初始化函数 void Timer0_Init(void) { TMOD = 0x22; // 定时器0和定时器1工作方式2,8位自动重装 TH0 = 0x00; TL0 = 0x00; ET0 = 1; // 定时器0中断 TR0 = 1; // 启动定时器0 } // 外部中断0初始化函数 void ExternalInterrupt0_Init(void) { IT0 = 1; // 设置外部中断0为跳沿触发方式,来一个下降沿触发一次 EX0 = 1; // 启动外部中断0 } // 提取红外数据并进行解码 void DecodeIR(void) { unsigned char i, j, k, cord, value = 0; // 初始化变量 k = 1; // 从第一位脉宽开始取,丢弃引导码脉宽 for (i = 0; i < 4; i++) { for (j = 0; j < 8; j++) { cord = ir_data[k]; // 把脉宽存入cord if (cord > 5) { // 如果脉宽大于某个值,判断为1 value |= 0x80; // 设置最高位为1 } if (j < 7) { value >>= 1; // value右移 } k++; // 每执行一次,脉宽位加1 } ir_code[i] = value; // 存入数组 value = 0; // 清零value } ir_process_complete = 1; // 完成标志 } // 定时器0中断处理函数 void Timer0Interrupt() interrupt 1 { ir_time++; } // 外部中断0处理函数 void ExternalInterrupt0() interrupt 0 { static unsigned char i = 0; // 声明静态变量 static bit start_flag = 0; // 开始储存脉宽标志位 if (start_flag) { // 开始接收脉宽检测 if ((ir_time < 53) && (ir_time >= 32)) { i = 0; // 如果是引导码 } ir_data[i] = ir_time; // 存入数组 ir_time = 0; // 清零 i++; // 计数脉宽 if (i == 33) { // 是否存入34次 ir_ok = 1; // 脉宽检测完毕 i = 0; // 清零 } } else { ir_time = 0; // 引导码开始 start_flag = 1; // 标志位置1 } }

使用注意: 初始化:

c
CCAPM1 = 0x42; //8位PWM模式,无中断 CCAP1H = 0x40; //PWM1占空比(调节此处值调节PWM占空比) CCAP1L = 0x40; //PWM1占空比(调节此处值调节PWM占空比) //nec Timer0_Init(); //定时器0初始化 ExternalInterrupt0_Init(); //外部中断0初始化

在while1里解码:

c
while (1) { if (ir_ok) //判断脉宽是否检测完毕 { DecodeIR(); //根据脉宽解码出4个字节的数据 ir_ok = 0; //重新等待脉宽检测 if (ir_process_complete) //判断是否解码完毕 { UART_SendChar(ir_code[0]); UART_SendChar(ir_code[1]); UART_SendChar(ir_code[2]); UART_SendChar(ir_code[3]); ir_process_complete = 0; } } }

eeprom flash

有1K:

image.png

eeprom.h

h
#ifndef __eeprom_H__ #define __eeprom_H__ #include <stc12c5a60s2.h> #include <intrins.h> //void Delay(uchar n); void IapIdle(); uchar IapReadByte(uint addr); void IapProgramByte(uint addr, uchar dat); void IapEraseSector(uint addr); #endif

eeprom.c

c
#include "EEPROM.H" /*Define ISP/IAP/EEPROM command*/ #define CMD_IDLE 0 //Stand-By #define CMD_READ 1 //Byte-Read #define CMD_PROGRAM 2 //Byte-Program #define CMD_ERASE 3 //Sector-Erase /*Define ISP/IAP/EEPROM operation const for IAP_CONTR*/ //#define ENABLE_IAP 0x80 //if SYSCLK<30MHz //#define ENABLE_IAP 0x81 //if SYSCLK<24MHz //#define ENABLE_IAP 0x82 //if SYSCLK<20MHz #define ENABLE_IAP 0x83 //if SYSCLK<12MHz //#define ENABLE_IAP 0x84 //if SYSCLK<6MHz //#define ENABLE_IAP 0x85 //if SYSCLK<3MHz //#define ENABLE_IAP 0x86 //if SYSCLK<2MHz //#define ENABLE_IAP 0x87 //if SYSCLK<1MHz /*---------------------------- Software delay function --------------------------- void Delay(uchar n) { uint x; while (n--) { x = 0; while (++x); } } -*/ /*---------------------------- Disable ISP/IAP/EEPROM function Make MCU in a safe state 该函数用于禁用ISP/IAP/EEPROM操作,确保MCU处于安全状态。 ----------------------------*/ void IapIdle() { IAP_CONTR = 0; // 关闭IAP功能 IAP_CMD = 0; // 清除命令,使其进入空闲状态 IAP_TRIG = 0; // 清除触发寄存器 IAP_ADDRH = 0x80; // 数据指针指向非EEPROM区域 IAP_ADDRL = 0; // 清除IAP地址以防止误用 } /*---------------------------- Read one byte from ISP/IAP/EEPROM area Input: addr (ISP/IAP/EEPROM address) Output:Flash data 该函数用于从ISP/IAP/EEPROM区域读取一个字节的数据。 ----------------------------*/ uchar IapReadByte(uint addr) { uchar dat; IAP_CONTR = ENABLE_IAP; // 打开IAP功能,并设置等待时间 IAP_CMD = CMD_READ; // 设置ISP/IAP/EEPROM读取命令 IAP_ADDRL = addr; // 设置ISP/IAP/EEPROM地址低位 IAP_ADDRH = addr >> 8; // 设置ISP/IAP/EEPROM地址高位 IAP_TRIG = 0x5a; // 发送触发命令1 (0x5a) IAP_TRIG = 0xa5; // 发送触发命令2 (0xa5) _nop_(); // MCU将在此保持,直到ISP/IAP/EEPROM操作完成 dat = IAP_DATA; // 读取ISP/IAP/EEPROM数据 _nop_(); _nop_(); IapIdle(); // 关闭ISP/IAP/EEPROM功能 return dat; // 返回Flash数据 } /*---------------------------- Program one byte to ISP/IAP/EEPROM area Input: addr (ISP/IAP/EEPROM address) dat (ISP/IAP/EEPROM data) Output:- 该函数用于向ISP/IAP/EEPROM区域写入一个字节的数据。 ----------------------------*/ void IapProgramByte(uint addr, uchar dat) { IAP_CONTR = ENABLE_IAP; // 打开IAP功能,并设置等待时间 IAP_CMD = CMD_PROGRAM; // 设置ISP/IAP/EEPROM编程命令 IAP_ADDRL = addr; // 设置ISP/IAP/EEPROM地址低位 IAP_ADDRH = addr >> 8; // 设置ISP/IAP/EEPROM地址高位 IAP_DATA = dat; // 写入ISP/IAP/EEPROM数据 IAP_TRIG = 0x5a; // 发送触发命令1 (0x5a) IAP_TRIG = 0xa5; // 发送触发命令2 (0xa5) _nop_(); _nop_(); IapIdle(); // 关闭ISP/IAP/EEPROM功能 } /*---------------------------- Erase one sector area Input: addr (ISP/IAP/EEPROM address) Output:- 该函数用于擦除一个扇区的数据。 ----------------------------*/ void IapEraseSector(uint addr) { IAP_CONTR = ENABLE_IAP; // 打开IAP功能,并设置等待时间 IAP_CMD = CMD_ERASE; // 设置ISP/IAP/EEPROM擦除命令 IAP_ADDRL = addr; // 设置ISP/IAP/EEPROM地址低位 IAP_ADDRH = addr >> 8; // 设置ISP/IAP/EEPROM地址高位 IAP_TRIG = 0x5a; // 发送触发命令1 (0x5a) IAP_TRIG = 0xa5; // 发送触发命令2 (0xa5) _nop_(); _nop_(); IapIdle(); // 关闭ISP/IAP/EEPROM功能 } /* void main() { uint i; P1 = 0xfe; //1111,1110 System Reset OK Delay(10); //Delay IapEraseSector(IAP_ADDRESS); //Erase current sector for (i=0; i<512; i++) //Check whether all sector data is FF { if (IapReadByte(IAP_ADDRESS+i) != 0xff) goto Error; //If error, break } P1 = 0xfc; //1111,1100 Erase successful Delay(10); //Delay for (i=0; i<512; i++) //Program 512 bytes data into data flash { IapProgramByte(IAP_ADDRESS+i, (uchar)i); } P1 = 0xf8; //1111,1000 Program successful Delay(10); //Delay for (i=0; i<512; i++) //Verify 512 bytes data { if (IapReadByte(IAP_ADDRESS+i) != (uchar)i) goto Error; //If error, break } P1 = 0xf0; //1111,0000 Verify successful while (1); Error: P1 = 0xff; //0xxx,xxxx IAP operation fail while (1); } */

main里面使用:

c
#define IAP_ADDRESS1 0x0100 /* 起始地址 */ IapEraseSector(IAP_ADDRESS1); IapProgramByte(IAP_ADDRESS1, 0x37); /* 写入数据 */ delayms(100); datx = IapReadByte(IAP_ADDRESS1); /* 读取用户密码个数 */ UART_SendChar(datx);

注意了,写入之前要擦除:

IapEraseSector(IAP_ADDRESS1); IapProgramByte(IAP_ADDRESS1, 0x37); /* 写入数据 */

读取就随时读取:

delayms(100); datx = IapReadByte(IAP_ADDRESS1); /* 读取用户密码个数 */
如果对你有用的话,可以打赏哦
打赏
ali pay
wechat pay

本文作者:Dong

本文链接:

版权声明:本博客所有文章除特别声明外,均采用 CC BY-NC。本作品采用《知识共享署名-非商业性使用 4.0 国际许可协议》进行许可。您可以在非商业用途下自由转载和修改,但必须注明出处并提供原作者链接。 许可协议。转载请注明出处!