本帖最后由 10086333 于 2016-12-11 16:02 編輯
//1602液晶在stm8的基本顯示 //由于引腳不如51單片機多,采用位移方式 #include<stm8s003f3.h>
#define ucharunsigned char #define uintunsigned int /***************************************** 電路連接 D1~PD4低4位 ------DB0~DB3 (采用位左移一位方法取data 因為沒有D0) C1~C4 高4位 ------DB4~DB7 A2------RS A3------RW A1------E *****************************************/ //#define LCD_DB P1 _Bool LCD_RS@PA_ODR:2; _Bool LCD_RW@PA_ODR:3; _Bool LCD_E@PA_ODR:1; uchar ttt[]="Wyu University "; uchar kkk[]="Chen ying lun "; /******定義函數****************/
void LCD_init(void);//聲明初始化函數 void LCD_write_command(uchar command);//寫指令函數 void LCD_write_data(uchar dat);//寫數據函數 void LCD_disp_char(uchar x,uchar y,uchardat);//在某個屏 //個字符,X(0-16),y(1-2) //void LCD_check_busy(void);//檢查忙函數。我沒用到此函數 void delay_n40us(uint n);//延時函數 //******************************** //*******初始化函數*************** void LCD_init(void) { LCD_write_command(0x38);//設置8位格式,2行,5x7 LCD_write_command(0x0c);//整體顯示,關光標,不閃爍 LCD_write_command(0x06);//設定輸入方式,增量不移位-------------- LCD_write_command(0x01);//清除屏幕顯示---------------- delay_n40us(100);//實踐證明,我的LCD1602上,用for循環200 次就能可靠完成清屏指令。 } //******************************** //********寫指令函數***********設置先向1602寫入的數據為地址* void LCD_write_command(uchar dat) { PD_ODR=(dat<<1);
PC_ODR=dat; LCD_RS=0;//指令 LCD_RW=0;//寫入 LCD_E=1;//允許 LCD_E=0;//產生下降沿 delay_n40us(1);//實踐證明,我的LCD1602上,用for 循環1 次就 } //******************************* //********再寫數據函數*****向1602中寫入數據******** void LCD_write_data(uchar dat) { PD_ODR=(dat<<1);
PC_ODR=dat; LCD_RS=1;//數據 LCD_RW=0;//寫入 LCD_E=1;//允許 LCD_E=0;//產生下降沿 delay_n40us(1); } //寫入字符串函數 void writeDataLcd(unsigned char dat)// 寫入數據 { LCD_RS=1; //RS置1 (運算后P5=xxx1 xxxx,即RS=1其他位狀態不變) LCD_RW=0; //RW清0 (運算后P5=xx0x xxxx, 即RW=0其他位狀態不變) LCD_E=0; //使能E清0 (運算后P5=xxxx 0xxx, 即E=0其他位狀態不變)
PD_ODR=(dat<<1);
PC_ODR=dat; //送數據到P2OUT準備輸出進入1602,等E下降沿來即可進入1602. LCD_E=1; //使能E置1,P5OUT= xxxx xxxx+0000 1000=xxxx1xxx,E=1. delay_n40us(10); LCD_E=0; //使能E清0,這樣E從1變0,產生一個下降,寫入命令到1602。 }
//******************************** //*******顯示一個字符函數********* voidLCD_disp_char(uchar x,uchar y,uchar dat) { uchar address; if(y==1) address=0x80+x; else address=0xc0+x; LCD_write_command(address); LCD_write_data(dat); } //******************************** /*******檢查忙函數************* voidLCD_check_busy() //實踐證明,在我的LCD1602 上,檢查忙指令通 過率極低,以至于不能正常使用LCD。因此我沒有再用檢查忙函數。而使 do//用了延時的方法,延時還是非常好用的。我試了一下,用// {LCD_E=0; //for循環作延時,普通指令只要1次循就可完成。清屏指令 LCD_RS=0; //要用200次循環便能完成。 LCD_RW=1; LCD_DB=0xff; LCD_E=1; while(LCD_DB^7==1); } ******************************/ //********延時函數*************** voiddelay_n40us(uint n) { uint i; uchar j; for(i=n;i>0;i--) for(j=0;j<2;j++);//在這個延時循環函數中我只做了2次循環, } //實踐證明我的LCD1602 上普 //通的指令只需1次循環就能可靠完成。 //******************************* //*********主函數***************** void main() { PD_DDR=0XFF; PD_CR1=0XFF; PD_CR2=0XFF; PC_DDR=0XFF; PC_CR1=0XFF; PC_CR2=0XFF; PA_DDR=0XFF; PA_CR1=0XFF; PA_CR2=0XFF; PB_DDR=0XFF; PB_CR1=0XFF; PB_CR2=0XFF;
while(1) { uint i=0; LCD_init(); LCD_write_command(0x80);//0x80第一行從頂頭開始顯示,加n,后移n位顯示 for (i=0;i<25;i++) { writeDataLcd(ttt); delay_n40us(15000); } delay_n40us(30000); delay_n40us(30000); delay_n40us(50000); LCD_write_command(0x80+64);//0x80+64,第二行從頂頭開始顯示,加64再加n,第二行從n開始顯示。 for (i=0;i<23;i++) { writeDataLcd(kkk); delay_n40us(15000); } LCD_write_command(0x80); LCD_disp_char(2,1,' '); delay_n40us(20000); } } |