📄 xram.#1
字号:
//----------------------------------------------------------------
// C8051F020实验核心模块测试程序
// XDATA RAM实验
//
// 实现功能:通过交叉开关,将UART0配置到P00,P01引脚上,实现:
// 1.通过计算机向模块发送命令控制底板上的LED灯
// 2.串口通信协议是基于字符的,可使用超级终端进行调试。支持BACKSPACE
// 键,但是不支持其它功能键和方向键。
//
// XRAM说明:
// 1.底板上外边扩展的XRAM地址空间分布:
// 62LV256: 0-0x7FFF,32K
// CCP2201: 0x8000-0xBFFF
// 大容量RAM,可通过核心板上跳线选择使用RTC的电池作为长期存储设备;
// 62LV2568:0xC000-0xFFFF
// HRAM3:HRAM0,共16页
// 当使用62LV1024时共有8页
//
// 实验条件:
// 1.底板上JP1的1,2,4,6跳线插上跳线帽接通
//
// 说明:
// 本模块中配套的测试程序都是以消息驱动的,如果对程序实现思想不甚清楚,
// 可与公司联系
//
// scj,scj@sina.com,2007/04
//---------------------------------------------------------------
// 修改记录
// 版本 时间 人员 内容
// 1.0 2007/4/30 scj 建立文件,实现基本功能
// 1.1 2007/6/4 scj 根据底板V122修改功能
//
//---------------------------------------------------------------
//----------------------------------------------------------------------------------------------------
// 命令说明:
// 字符通信协议,所有的命令以回车符结束
// 命令字符不区分大小写
// 键盘控制字符支持Backspace,其它控制键不支持
//
// 应答:命令操作成功,最后都将返回\r\nOK\r\n
// 命令错误,返回\r\nError\r\n
// 具体命令:
// 1.控制LED
// 命令格式:setled=灯序号,开关状态 \r
// 参数: 灯序号: 2,LED2
// 开关状态:ON,打开灯;OFF,关闭灯
// 应答: \r\nOK\r\n
// 2.测试SRAM,U2
// 命令格式:testsram\r
// 参数: 无
// 应答: 发送测试的结果;如果测试中发现错误,可能出现了芯片焊接问题或损坏问题
// 3.测试TRAM,U4
// 命令格式:testtram\r
// 参数: 无
// 应答: 发送每个页面的测试结果;测试过程中会修改RAM中保存的内容
// 4.向指定页面写入字符串
// 命令格式:writetram=页面\r
// 参数: 页面,0-15
// 应答: \r\nOK\r\n;\r\nError\r\n
// 说明: 写入的字符串均在页面的0偏移处开始写入,即0xC000地址空间;用户发送的字符串长度不要超过接收缓冲区的长度
// 5.读取指定页面从0偏移处开始的指定长度数据
// 命令格式:readtram=页面,长度\r
// 参数: 页面序号:欲写入的页面,0-15;
// 长度:要读出的字节个数
// 应答: 16进制的方式显示264个字节的内容;同时每个16进制后的括号内显示该字符;字节间用空格隔开
// \r\nOK\r\n
#include "F020.h"
#include <string.h>
#include <ctype.h>
#include <stdlib.h>
#define T62LV2568 0
#define T62LV1024 1
// 根据核心板上焊接的芯片选择相应的TRAM类型(芯片U4)
#define TRAM_TYPE T62LV2568
//#define TRAM_TYPE T62LV1024
#if(TRAM_TYPE==T62LV2568)
#define RAM_PAGE 16
#else
#define RAM_PAGE 8
#endif
//--------------------------------------------
// 数据类型定义
//--------------------------------------------
// 基本数据类型
typedef unsigned char BYTE; // 8位无符号数
typedef unsigned int WORD; // 16位无符号数
typedef unsigned long DWORD; // 32位无符号数
// 结构体
typedef struct
{
BYTE base; // DIDA次数
BYTE second; // 秒
BYTE minute; // 分
}S_TIME;
//--------------------------------------------
// 引脚定义
//--------------------------------------------
#define P_LED2 P31
//--------------------------------------------
// 宏定义
//--------------------------------------------
// 系统消息定义
#define SECD_SIGNAL 0x01
#define MINU_SIGNAL 0x02
#define UART0RECV_SIGNAL 0x04
// 常量定义
//定时器0定时,使用SYSCLK/12作为时钟源
//定时计算方法,TH0:TL0= 65536-t*SYSCLK/12
//定时25ms,TH0:TL0=0x4C 00
#define TMR0H 0x4C
#define TMR0L 0x00
#define TICKS_PER_SEC 40
#define RECVBUF_LEN 50
code BYTE ASCII_CODE[]={"0123456789ABCDEF"};
code BYTE TESTDATA[]={0x00,0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0xFF,0xFE,0xFC,0xFA,0xF8,0xF4,0xF2};
//--------------------------------------------
// 函数声明
//--------------------------------------------
void CommandProc(char *pRecvString);
void SendString(char *pSendString);
void SendHex(BYTE nData);
void SendWordHex(WORD wData);
void SendBuf(BYTE buf);
BYTE TestSRAM();
BYTE TestTRAM();
char strncmps(char *pString1,char *pString2,int iLen);
//--------------------------------------------
// 全局变量定义
// 全局变量以g_开头
// 位变量: g_b
// 无符号字节变量: g_c
// 无符号双字节变量: g_w
// 无符号4字节变量: g_dw
// 有符号数在类型前加s,比如char型位g_sc
// 结构体为g_s后为大写
//--------------------------------------------
data BYTE g_cMainSignal; // 主消息变量
data S_TIME g_sTime; // 主时钟
data BYTE g_cRecvPointer; // 接收指针
data BYTE g_cRecvLength; // 接收数据长度
idata BYTE g_cRecvBuffer[RECVBUF_LEN]; // 接收缓冲区,idata
//-----------------------------------------------
// 初始化函数
//
//-----------------------------------------------
void Timer_Init()
{
PCON = 0;
//.......................................
// 设置定时器的时钟信号
// - T4M T2M T1M T0M - - -
// 0 1 0 1 0 0 0 0
// 1:使用SYSCLK;0:使用SYSCLK/12
// 定时器0,1,2,4均使用SYSCLK
//.......................................
CKCON = 0x70;
//......................................
// 定时0工作在16位定时器模式
// 定时器1工作在8位重载模式
//......................................
TMOD = 0x21;
//....................................
// T0作为系统节拍时钟
// 定时计算方法= 65536-t*SYSCLK
//.....................................
TH0 = TMR0H;
TL0 = TMR0L;
//.......................................
// T1作为串口0时钟
// 串口0波特率=SYSCLK/16*(256-TMR1LOAD)
// T1波特率=9600
//.......................................
TH1 = 0xB8;
TL1 = 0xB8;
TCON = 0x00;
//..................................................
// T2 and T3 not used
//..................................................
RCAP2H = 0x00; // Timer 2 Capture Register High Byte
RCAP2L = 0x00; // Timer 2 Capture Register Low Byte
TH2 = 0x00; // Timer 2 High Byte
TL2 = 0x00; // Timer 2 Low Byte
T2CON = 0x00; // Timer 2 Control Register
TMR3RLL = 0x00; // Timer 3 Reload Register Low Byte
TMR3RLH = 0x00; // Timer 3 Reload Register High Byte
TMR3H = 0x00; // Timer 3 High Byte
TMR3L = 0x00; // Timer 3 Low Byte
TMR3CN = 0x00; // Timer 3 Control Register
//.....................................................
// T4 作为串口1波特率发生器
// T4波特率=SYSCLK/32*(65536-RCAP4H:RCAP4L)
// T4波特率=9600
//.....................................................
RCAP4H = 0xFF; // Timer 4 Capture Register High Byte
RCAP4L = 0xDC; // Timer 4 Capture Register Low Byte
TH4 = 0xFF; // Timer 4 High Byte
TL4 = 0xDC; // Timer 4 Low Byte
T4CON = 0x30; // Timer 4 Control Register
}
//...........................................................
// 功能:UART0初始化
// 输入参数:无
// 输出参数:无
// 注意:1.应该首先使用交叉开关为UART0分配相应的引脚
// 2.初始化定时器1作为波特率发生器
//..........................................................
void UART0_Init()
{
SCON0 = 0x50; // 8位波特率异步通信模式
REN0 = 1; // 使能UART0接收
TR1 = 1; // 启动定时器1
}
//--------------------------------------------------
// 只使用外部RAM
//------------------------------------------------
void EMI_Init()
{
EMI0CF = 0x2F;
//EMI0TC = 0x8A; // 系统时钟为22。1184,地址建立时间和WR,RD时间不能太短,否则可能导致访问RAM错误
// 具体时间参数可以参考数据手册
EMI0TC = 0xFF;
}
//......................................................
// 功能:配置交叉开关和端口方向
// 输入参数:无
// 输出参数:无
// 注意:该函数使能交叉开关
// TX0,RX0被分配到P00,P01
//.....................................................
void Port_IO_Init()
{
// P0.0 - TX0 (UART0), Open-Drain, Digital
// P0.1 - RX0 (UART0), Open-Drain, Digital
// P0.2 - SCK (SPI0), Open-Drain, Digital
// P0.3 - MISO (SPI0), Open-Drain, Digital
// P0.4 - MOSI (SPI0), Open-Drain, Digital
// P0.5 - NSS (SPI0), Open-Drain, Digital
// P0.6 - SDA (SMBus), Open-Drain, Digital
// P0.7 - SCL (SMBus), Open-Drain, Digital
// P1.0 - Skipped, Open-Drain, Analog
// P1.1 - Skipped, Open-Drain, Analog
// P1.2 - Skipped, Open-Drain, Analog
// P1.3 - Skipped, Open-Drain, Analog
// P1.4 - Skipped, Open-Drain, Analog
// P1.5 - Skipped, Open-Drain, Analog
// P1.6 - Skipped, Open-Drain, Analog
// P1.7 - Skipped, Open-Drain, Analog
// P2.0 - TX1 (UART1), Open-Drain, Digital
// P2.1 - RX1 (UART1), Open-Drain, Digital
// P2.2 - INT0 (Tmr0), Open-Drain, Digital
// P2.3 - INT1 (Tmr1), Open-Drain, Digital
// P2.4 - Unassigned, Open-Drain, Digital
// P2.5 - Unassigned, Open-Drain, Digital
// P2.6 - Unassigned, Open-Drain, Digital
// P2.7 - Unassigned, Open-Drain, Digital
// P3.0 - Unassigned, Open-Drain, Digital
// P3.1 - Unassigned, Push-Pull, Digital
// P3.2 - Unassigned, Open-Drain, Digital
// P3.3 - Unassigned, Open-Drain, Digital
// P3.4 - Unassigned, Open-Drain, Digital
// P3.5 - Unassigned, Open-Drain, Digital
// P3.6 - Unassigned, Open-Drain, Digital
// P3.7 - Unassigned, Open-Drain, Digital
P1MDIN = 0x00;
P3MDOUT = 0x02;
XBR0 = 0x07;
XBR1 = 0x14;
XBR2 = 0x44;
}
void Oscillator_Init()
{
int i = 0;
OSCXCN = 0x67;
for (i = 0; i < 3000; i++); // Wait 1ms for initialization
while ((OSCXCN & 0x80) == 0);
OSCICN = 0x08;
}
void Interrupts_Init()
{
//..................................
// 初始化中断
// EA IEGF0 ET2 ES0 ET1 EX1 ET0 EX0
//..................................
IE = 0;
ET0 = 1; // 打开定时器0中断
ES0 = 1; // 使能串口中断
}
//-------------------------------
// 功能: 初始化单片机内部资源
// 输入参数:无
// 输出参数:无
//-------------------------------
void Init_Device(void)
{
EMI_Init();
Timer_Init();
Port_IO_Init();
Oscillator_Init();
UART0_Init();
Interrupts_Init();
}
//------------------------------------------------
// 函数名称:InitVariable()
// 函数功能:初始化系统全局变量
//------------------------------------------------
void InitVariable()
{
g_cMainSignal = 0;
g_sTime.base = 0;
g_sTime.second= 0;
g_sTime.minute= 0;
g_cRecvPointer =0;
g_cRecvLength =0;
}
//-------------------------------------------------------------
// 函数名称:定时器0中断入口函数,完成计时功能,定时时间为25ms
// 输入参数: 无
// 输出参数: 无
//-------------------------------------------------------------
void timer0() interrupt 1
{
TH0=TMR0H;
TL0=TMR0L;
// 程序时基信号
if(++g_sTime.base>=TICKS_PER_SEC)
{// 1s到
g_sTime.base = 0;
g_cMainSignal|=SECD_SIGNAL; // 产生秒信号
if(++g_sTime.second>59)
{// 1分钟到
g_sTime.second = 0;
g_cMainSignal|=MINU_SIGNAL; // 产生分信号
if(++g_sTime.minute>59) g_sTime.minute=0;
}
}
}
//..................................................
// 串口0中断服务程序
//
//.................................................
void Uart0() interrupt 4
{
BYTE cRecvData;
if(RI0) // 如果收到数据
{
RI0=0; // 清除收到数据标志
cRecvData = SBUF0; // 读取数据
SBUF0 = cRecvData; // 回显
if(cRecvData=='\r')
{// 回车符作为结束标志
g_cRecvBuffer[g_cRecvPointer]=0; // 接收结束
g_cRecvPointer = 0;
g_cMainSignal|=UART0RECV_SIGNAL; // 接收完成信号
}
else if(cRecvData==0x08)
{// CTRL+H控制键,退格
if(g_cRecvPointer>0) g_cRecvPointer--;
}
else
{
if(g_cRecvPointer<RECVBUF_LEN)
{
if(isupper(cRecvData))
{// 大写字符转换为小写
cRecvData = _tolower(cRecvData);
}
g_cRecvBuffer[g_cRecvPointer++]=cRecvData;
}
else
{// 数据长度错误
g_cRecvPointer = 0;
}
}
}
if(TI0) // 如果数据发送结束
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -