?? 340_ad.c
字號:
/********************************************************************
例程說明,本程序是對電位器產生的電壓值進行采樣,可以在寄存器觀察AD的
也可以通過LCD將值顯示出來,由于LV595芯片片選引腳與以太網控制芯片CP2200
中斷引腳復用因些在此例程中加入了將CP2200芯片中斷關閉的代碼。還要在建立
工程文件時將CP220X_REG.C文件包含到工程中.
********************************************************************/
#include "C8051F340.h"
#include "CP220x_REG.h"
#define uchar unsigned char
#define uint unsigned int
#define BASE_ADDRESS 0x10
#define EMIF_TIMING 0x9E
sfr16 TMR2RL = 0xca;
sfr16 TMR2 = 0xcc;
sbit LcdCs=P0^3;
sbit led=P2^4;
bit SPI_BUSY;
uchar xdata virt_port;
uchar xdata discode;//開關顯示、光標、閃爍的命令字暫存單元
uint Count1ms;
uint temp,ams,temp_l; // 存儲ADC的變量
uchar xdata tab1[16]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0x41,0x42,0x43,0x44
,0x45,0x46};
void ADC0_Init(void); //ADC初始化
void Timer2_Init(void); //定時器初始化
void Delay(void); //延時程序
void PORT_Init(void); //端口初始化
void delay(uint timer);
void writelcd(uchar spi_data);
void writecode(uchar ins);
void writedata(uchar d);
void lcdinit();
void lcdchar (void);
void lcdaddr(uchar DDRAM);
void cursoropen(uchar position);//開光標,關閃爍
void displayopen(void); //開顯示,光標、閃爍不變
void ether_reset_low();
void ether_reset_high (void);
void Delay1ms(uint T);
void EMIF_Init (void);
void SPI_Init();
/*********************************************************************/
// 主程序
/*********************************************************************/
void main()
{
uint j;
PCA0MD &= ~0x40; //關閉看門狗
OSCICN |= 0x03; //使用內部12M晶振
ADC0_Init(); //ADC初始化
Timer2_Init(); //定時器初始化
PORT_Init(); //端口初始化
EMIF_Init ();
ether_reset_low();
for(j=0;j<5000;j++);
ether_reset_high ();
INT0EN = 0x00;
INT1EN = 0x00;
SPI_Init();
lcdinit();
IE = 0x40;
EA=1; //開中斷
while(1)
{ lcdinit();
lcdchar();
} //取值此處加顯示
}
/*********************************************************************/
// 延時程序
/*********************************************************************
void Delay(void)
{
int x;
for(x = 0;x < 500;x)
x++;
}
/*********************************************************************/
// 定時器2中斷程序
/*********************************************************************/
void Timer2_ISR(void) interrupt 5
{
TF2H = 0; // 清除定時器2的中斷標志
}
/*********************************************************************/
// ADC初始化程序
/*********************************************************************/
void ADC0_Init(void)
{
REF0CN = 0x0F; //使用內部參考電壓VREF
AMX0P = 0x04; // P2.5為輸入端
AMX0N = 0x1F; // 單端輸入
ADC0CF = 0xFC; // 采樣值右對齊,如左對齊其值為FC
ADC0CN = 0xC2; //采用定時器2溢出后3個SAR時鐘開轉換
// 這時MCU功耗比較低
EIE1 |= 0x08; //使能ADC中斷
}
/*********************************************************************/
// ADC中斷程序
/*********************************************************************/
void ADC0_ConvComplete_ISR(void) interrupt 10
{
temp= ADC0H; //高字節
ams=ADC0L; //低字節
// temp_l=temp<<8+ams;
AD0INT = 0; //清中斷
}
/*********************************************************************/
// 端口初始化程序
/*********************************************************************/
void PORT_Init(void)
{
P2MDIN = 0xDF; // P2.5 為模擬輸入
P2SKIP = 0x20; // P2.5 跳過
XBR0 = 0x02;
XBR1 = 0x40; // 允許交叉開關
P0MDOUT =0xff;
P1MDOUT |= 0xD8; // /WR and /RD are push-pull P3MDOUT |= 0xFF; P4MDOUT |= 0xFF;
}
/*********************************************************************/
// 定時器2初始化程序
/*********************************************************************/
void Timer2_Init(void)
{
TMR2CN = 0x00; //清定時器2標志
CKCON &= ~0xF0; //選擇時基
TMR2RL = 0; // 重載值
TMR2 = 0xffff; //
ET2 = 1; // 使能定時器2 中斷
TR2 = 1; // 開啟定時器2
}
void delay1us(void)
{
uchar xdata i;
for(i=10;i>0;i--);
}
void SPI_Init()
{
SPI0CFG = 0x70;
SPI0CN = 0x01;
SPI0CKR = 0x3B;
}
void writelcd(uchar spi_data) //向LV595寫數據以SPI方式
{
uint xdata i,j;
for(i=0;i<3;i++)
LcdCs=0;
SPI_BUSY=1;
delay1us();
SPI0DAT=spi_data;
for(j=0;j<500;j++)
{
if(!SPI_BUSY)break;
}
delay1us();
LcdCs=1;
delay1us();
delay1us();
delay1us();
delay1us();
delay1us();
delay1us();
delay1us();
for(j=0;j<5000;j++);
}
void writecode(uchar ins) //向液晶控制模塊寫命令
{
uchar xdata t;
uint xdata j;
for(j=0;j<500;j++);
virt_port|=ins&0xf0; //取高字節向1602寫數據
writelcd(virt_port); //先寫高4位后再寫低4位 這是在選擇四個數據線的時候
virt_port|=0x08;
writelcd(virt_port); //使能E腳為高電平
virt_port&=~0x08;
writelcd(virt_port); //使能E腳為低電平
t=ins<<4;
virt_port|=t&0xf0; //取低字節向1602寫數據
writelcd(virt_port);
virt_port|=0x08; //使能E腳為高電平
writelcd(virt_port);
virt_port&=~0x08; //使能E腳為低電平
writelcd(virt_port);
virt_port=0;
writelcd(virt_port);
}
void writedata(uchar d) //寫數據到1602
{
uchar xdata i,t;
uint xdata j;
for(j=0;j<500;j++);
virt_port|=0x02; //光標返回指令
writelcd(virt_port);
virt_port|=d&0xf0; //取高字節數據給數據寄存器
writelcd(virt_port);
virt_port|=0x08; //使能E腳為高電平
writelcd(virt_port);
virt_port&=0xf7; //使能E腳為低電平
writelcd(virt_port);
virt_port&=0x07; //設置光標右移,文字也右移
writelcd(virt_port);
t=d&0x0f;
virt_port|=t<<4; //取低字節數據給數據寄存器
writelcd(virt_port);
virt_port|=0x08; //使能E腳為高電平
writelcd(virt_port);
for(i=3;i>0;i--);
virt_port&=0xf7; //使能E腳為低電平
writelcd(virt_port);
virt_port=0; //數據清零
writelcd(virt_port);
}
void lcdchar (void)
{
// uint xdata j,k;
uchar a,b,c,d;
temp=temp<<8;
temp_l=temp+ams;
temp_l=temp_l>>4;
a=temp_l/1000;d=temp_l%1000;
writedata(tab1[a]);
b=d/100;d=d%100;
writedata(tab1[b]);
c=d/10; d=d%10;
writedata(tab1[c]);
writedata(tab1[d]);
}
void lcdinit()
{
uint xdata i;
virt_port=0x00;
writelcd(virt_port); //RS腳為零寫指令寄存器 RW為低電平
virt_port|=0x08; //使能腳E=1
writelcd(virt_port);
virt_port&=0xf7; //使E=0使能腳發生跳變高--低
writelcd(virt_port);
for(i=1000;i>0;i--); //3次延時,重要!
for(i=8000;i>0;i--); //用延時代替查詢
virt_port=0; //寫入20碼,設置為4位總線方式,非常重要!!!!!
writelcd(virt_port);
virt_port|=0x20; //只寫高字節
writelcd(virt_port);
virt_port|=0x08; //E=1
writelcd(virt_port);
for(i=20;i>0;i--); //使命令執行
virt_port&=0xf7; //E=0
writelcd(virt_port);
for(i=10;i>0;i--);
virt_port=0;
writelcd(virt_port);
writecode(0x38); //設兩行顯示,4線方式5*10字符顯示
for(i=50;i>0;i--);
writecode(0x06); //光標右移
for(i=50;i>0;i--);
writecode(0x0c); //開顯示指令
for(i=50;i>0;i--);
writecode(0x02); //光標返回指令
for(i=50;i>0;i--);
writecode(0x01); //清屏指令
for(i=50;i>0;i--);
}
void spi_isr(void) interrupt 6
{
if(SPIF)
{
SPIF=0;
SPI_BUSY=0;
}
if(WCOL)
{
WCOL=0;
}
}
void ether_reset_low(){ P1 &= ~0x01; // Pull reset low}void ether_reset_high (void){ P1 |= 0x01; // Allow /RST to rise while(!(P1 & 0x01)); // Wait for /RST to go high}
void EMIF_Init (void){ EMI0CF = 0x1B; // non-muxed mode; split mode // with bank select EMI0TC = EMIF_TIMING; // This constant may be modified // according to SYSCLK to meet the // timing requirements for the CP2200 EMI0CN = BASE_ADDRESS; // Page of XRAM accessed by EMIF}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -