stm32并口驱动12864,求大神看看我的程序错在哪了?编译通过但是屏幕上没显示~搞了两天了,头疼死我了、

stm32并口驱动12864,求大神看看我的程序错在哪了?编译通过但是屏幕上没显示~搞了两天了,头疼死我了、,第1张

我有51的程序,可供参考。

#include "lcd12864.h"

#include "ziku.h"键棚键

#include <string.h>

static void delay(uint j)//延时

{

uchar i

for(j!=0j--)

for(i=0i<100i++)

}

void busy(void)

{

uchar i

for(i=0i<50i++)

_nop_()

}

void wdata(uchar wdata)

{

busy() //忙提示

LCD_RW=0

LCD_DI=1

P0=wdata

LCD_EN=0

LCD_EN=1

LCD_EN=0

}

void wcode(uchar wcode)

{

busy()

LCD_RW=0

LCD_DI=0

P0=wcode

LCD_EN=0

LCD_EN=1

LCD_EN=0

}

void subinit()

{

delay(10)

wcode(0xc0)//设置显示初始行

}

//设置显示位置

void setxy(uchar x,uchar y)

{

if ((y>=0)&(y<=63))

{

LCD_CSA=0

LCD_CSB=1

}

else //if (y<=127)

{

LCD_CSA=1

LCD_CSB=0

}

wcode(0x40|(y%64))

wcode(0xb8|x)

P0=0xff

}

void wdram(uchar x,uchar y,uchar dd)

{

setxy(x,y)

wdata(dd)

P0=0xff

LCD_CSA=1

LCD_CSB=1

}

//复位.

void Lcd_RST(void)

{

//rst=0

LCD_REST=0

delay(50)

LCD_REST=1

Lcd_Clear(0,7,0,128)

wcode(0x3f)//开显示

}

//LCD初始化

void Lcd_Init(void)

{

LCD_POR=0

Lcd_RST()

LCD_CSA=0

LCD_CSB=1

wcode(0x3e)subinit()

LCD_CSA=1

LCD_CSB=0

wcode(0x3e)subinit()

Lcd_Clear(0,7,0,128)

LCD_CSA=0

LCD_CSB=1

wcode(0x3f)//开显和芦示

LCD_CSA=1

LCD_CSB=0

wcode(0x3f)//开显示

}

void Lcd_On(void)

{

LCD_CSA=0

LCD_CSB=1

wcode(0x3f)//开显示

LCD_CSA=1

LCD_CSB=0

wcode(0x3f)//开显示

}

/稿巧/LCD 清显示屏

void Lcd_Clear(uchar StartLine,uchar StopLine,uchar StartRow,uchar StopRow)

{

uchar x,y

for(x=StartLinex<StopLine+1x++)

{

for(y=StartRowy<StopRowy++)

{

wdram(x,y,0)

}

}

}

//显示一个汉字

void Lcd_DispOneChar(uchar x,uchar y,uchar * hz,uchar disp_mode,uchar Width)

{

uchar i

for(i=0i<Widthi++)

{

if(disp_mode==WHITE)

{

wdram(x,y+i,*(hz+i))

wdram(x+1,y+i,*(hz+Width+i))

}

else

{

wdram(x,y+i,0xff-*(hz+i))

wdram(x+1,y+i,0xff-*(hz+Width+i))

}

}

if(Width==12)

{

for(i=12i<14i++)

{

if(disp_mode==WHITE)

{

wdram(x,y+i,0)

wdram(x+1,y+i,0)

}

else

{

wdram(x,y+i,0xff)

wdram(x+1,y+i,0xff)

}

}

for(i=1i<4i++)

{

if(disp_mode==WHITE)

{

wdram(x,y-i,0)

wdram(x+1,y-i,0)

}

else

{

wdram(x,y-i,0xff)

wdram(x+1,y-i,0xff)

}

}

}

}

void Lcd_Disp_String(uchar x,uchar y,char *pString,uchar disp_mode)

{

uchar i,j

uchar LineDispCode[16]

//strlen(),为字符串长度测量。

memset(LineDispCode,0,16) //清零数组

strcpy(LineDispCode,pString) //字符串之间的相互复制。

for(i=0i<strlen(pString)i++)

{

LineDispCode[i]=*(pString+i)

}

i=0

while(LineDispCode[i]!=0)

{

if(LineDispCode[i]>=0xA0)

{

//显示的是汉字

for(j=0j<ZIMO_NUMj++)

{

if(GB_12[j].Index[0]==LineDispCode[i] &&

GB_12[j].Index[1]==LineDispCode[i+1])

{

//显示的是汉字

Lcd_DispOneChar(x,y,GB_12[j].Msk,disp_mode,12)

y+=16

break

}

}

i+=2

}

else

{

//显示的是ASCII编码

for(j=0j<ASC_NUMj++)

{

if(ASC_12[j].Index==LineDispCode[i])

{

//显示的是汉字

Lcd_DispOneChar(x,y,ASC_12[j].Msk,disp_mode,8)

y+=8

break

}

}

i++

}

if(i>=16)

{

break

}

}

}

//显示数字.

void Lcd_Disp_OneNum(uchar x,uchar y,uchar num,uchar disp_mode)

{

switch(num)

{

case 0:{Lcd_Disp_String(x,y,"0",disp_mode)}break

case 1:{Lcd_Disp_String(x,y,"1",disp_mode)}break

case 2:{Lcd_Disp_String(x,y,"2",disp_mode)}break

case 3:{Lcd_Disp_String(x,y,"3",disp_mode)}break

case 4:{Lcd_Disp_String(x,y,"4",disp_mode)}break

case 5:{Lcd_Disp_String(x,y,"5",disp_mode)}break

case 6:{Lcd_Disp_String(x,y,"6",disp_mode)}break

case 7:{Lcd_Disp_String(x,y,"7",disp_mode)}break

case 8:{Lcd_Disp_String(x,y,"8",disp_mode)}break

case 9:{Lcd_Disp_String(x,y,"9",disp_mode)}break

default: break

}

}

//显示二位数。

void Disp_2num(uchar x,uchar y,uchar num,uchar disp_mode)

{

uchar ch[2]

ch[0]=num%10

ch[1]=num/10

Lcd_Disp_OneNum(x,y,ch[1],disp_mode)

Lcd_Disp_OneNum(x,y+8,ch[0],disp_mode)

}

//*****************************************************

//显示三位数。

void Disp_3num(uchar x,uchar y,uint num,uchar disp_mode)

{

uchar ch[2]

ch[0]=num/100

ch[1]=num%100

if(ch[0])

Lcd_Disp_OneNum(x,y, ch[0],disp_mode)

else

Lcd_Disp_String(x,y," ",disp_mode)

Disp_2num(x,y+8, ch[1],disp_mode)

}

//*****************************************************

//显示四位数。

void Disp_4num(uchar x,uchar y,uint num,uchar disp_mode)

{

uchar ch[4],tmp

tmp=num/100

ch[0]=tmp/10

ch[1]=tmp%10

tmp=num%100

ch[2]=tmp/10

ch[3]=tmp%10

Lcd_Disp_OneNum(x,y,ch[0],disp_mode)

Lcd_Disp_OneNum(x,y+8,ch[1],disp_mode)

Lcd_Disp_OneNum(x,y+16,ch[2],disp_mode)

Lcd_Disp_OneNum(x,y+24,ch[3],disp_mode)

}

void Lcd_DispIco2(uchar x,uchar y,uchar *pIco)//显示老肯图标

{

uchar i,j

for(i=0i<4i++)

{

for(j=0j<32j++)

{

wdram(x+i,y+j,*pIco)

pIco++

}

}

}

//*****************************************************

//显示多位数。 disp_mode&0x10==1时,进行即每位都显示,否则大于0的位置不显示。

void Disp_NumGB16(uchar x,uchar y,ulong Data,uchar num,uchar disp_mode)

{

uchar idata ch=0,i,tmp

for(i=0i<numi++)

{

tmp=Data%10

Data/=10

if((disp_mode&0x10)||tmp>0||Data>0||num<=2)

Lcd_Disp_OneNum(x,y+(num-i-1)*8,tmp,disp_mode%10)

else

Lcd_Disp_String(x,y+(num-i-1)*8," ",disp_mode%10)

}

}

可以。只是需要占用大量的碰槐老GPIO口,而且需要用其中的几根GPIO线模拟AD采样时序,具体跟你选择的AD芯片相关了。一般的ARM都有EMC静态存储器(NXP的ARM,三笑升星的名字不同),会预留明喊一些地址线或数据线,可以利用。

一、 STM32的输入输出管脚有下面8种(4输入 2输出 2复用输出)可能的配置:\x0d\x0a\x0d\x0a① 浮空输入_IN_FLOATING\x0d\x0a\x0d\x0a② 带上拉输入_IPU \x0d\x0a\x0d\x0a③ 带下拉输入_IPD \x0d\x0a\x0d\x0a④ 模拟输入_AIN\x0d\x0a\x0d\x0a⑤ 开漏输出_OUT_OD \x0d\x0a\x0d\x0a⑥ 推挽输出_OUT_PP\x0d\x0a\x0d\x0a⑦ 复用功能的推挽输出_AF_PP\x0d\x0a\x0d\x0a⑧ 复用功能的开漏输出_AF_OD\x0d\x0a\x0d\x0a1.1 I/O口的输出模式下,有3种输出速度可选(2MHz、10MHz和50MHz),这个速度是指I/O口驱动电路的响应速度而不是输出信号的速度,输出信号的速度与程序有关(芯片内部在I/O口 的输出部分安排了多个响应速度不同的输出驱动电路,用户可以根据自己的需要选择合适的驱动电路)。通过选择速度来选择不同的输出驱动模块,达到最佳的噪声控制和降低功耗的目的。高频的驱动电路橘链孝,噪声也高,当不需要高的输出频率时,请选用低频驱动电路,这样非常有利于提高系统的EMI性能。当然如果要输出较高频率的信号,但却选用了较低频率的驱动模块,很可能会得到失真的输出信号。\x0d\x0a\x0d\x0a输出速度又称输出驱动电路的响应速度,可理解为:输出驱动电路的带宽,即一个驱动电路可以不失真地通过信号的最大频率。\x0d\x0a\x0d\x0a如果一个信号的频率超过了驱动电路的响应速度,就有可能信号失真。如果信号频率为10MHz,而你配置了2MHz的带宽,则10MHz的方波很可能就变成了正弦波。就好比是公路的设计时速,汽车速度低于设计时速时,可以平稳地运行,如果超过设计时速就会颠簸,甚至翻车。\x0d\x0a\x0d\x0a关键是: GPIO的引脚速度跟应用相匹配,速度配置越高,噪声越大,功耗越大。\x0d\x0a\x0d\x0a带宽速度高的驱动器耗电大、噪声也大,带宽低的驱动器耗电小、噪声也小。使用合适的驱动器可以降低功耗和噪声。\x0d\x0aGPIO的引脚速度跟应用匹配(推荐10倍以上)。比如:\x0d\x0a\x0d\x0a1.1.1 对于串口,假如最大波特率只需115.2k,那么用2M的GPIO的引脚速度就够了,既省电也噪声小。\x0d\x0a\x0d\x0a1.1.2 对于I2C接口,假如使用400k波特率,若想把余量留大些,那么用2M的GPIO的引脚速度或许不够,这时可以选用10M的GPIO引脚速度。\x0d\x0a\x0d\x0a1.1.3 对于SPI接口,假如使用18M或9M波特率,用10M的GPIO的引脚速度显然不够了,需要选用50M的GPIO的引脚速度。\x0d\x0a\x0d\x0a 1.2 GPIO口设为输入时,输出驱动电路与端口是断开,所以输出速度配置无意义。\x0d\x0a\x0d\x0a 1.3 在复位期间和刚复位后,复用功能未开启,I/O端口被配置成浮空输入模式。\x0d\x0a\x0d\x0a 1.4 所有端口都有外部中断能力。为了使用外部中断线,端口必须配置成输入模式。\x0d\x0a\x0d\x0a 1.5 GPIO口的配置具有上锁功能,当配置好GPIO口后,可以通过程序锁住配置组唤渗合,直到下次芯片复位才能解锁。\x0d\x0a\x0d\x0a二、GPIO的翻转速度指:输入/输出寄存器的0 ,1 值反映到外部引脚(APB2上)高低电圆稿平的速度.手册上指出GPIO最大翻转速度可达18MHz。通过简单的程序测试,用示波器观察到的翻转时间是综合的时间,包括取指令的时间、指令执行的时间、指令执行后信号传递到寄存器的时间(这其中可能经过很多环节,比如AHB、APB、总线仲裁等),最后才是信号从寄存器传输到引脚所经历的时间。如有上拉电阻,其阻值越大,RC延时越大,即逻辑电平转换的速度越慢,功耗越大。 \x0d\x0a\x0d\x0a三、在STM32中如何配置片内外设使用的IO端口\x0d\x0a\x0d\x0a首先,一个外设经过 ①配置输入的时钟和 ②初始化后即被激活(开启);③如果使用该外设的输入输出管脚,则需要配置相应的GPIO端口(否则该外设对应的输入输出管脚可以做普通GPIO管脚使用);④再对外设进行详细配置。\x0d\x0a\x0d\x0a对应到外设的输入输出功能有下述三种情况:\x0d\x0a\x0d\x0a① 外设对应的管脚为输出:需要根据外围电路的配置选择对应的管脚为复用功能的推挽输出或复用功能的开漏输出。\x0d\x0a② 外设对应的管脚为输入:则根据外围电路的配置可以选择浮空输入、带上拉输入或带下拉输入。\x0d\x0a③ ADC对应的管脚:配置管脚为模拟输入。\x0d\x0a\x0d\x0a如果把端口配置成复用输出功能,则引脚和输出寄存器断开,并和片上外设的输出信号连接。将管脚配置成复用输出功能后,如果外设没有被激活,那么它的输出将不确定。\x0d\x0a\x0d\x0a四、 通用IO端口(GPIO)初始化\x0d\x0a\x0d\x0a 4.1 GPIO初始化\x0d\x0a\x0d\x0a41.1 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | B | C, ENABLE):使能APB2总线外设时钟;\x0d\x0a\x0d\x0a41.2 RCC_ APB2PeriphResetCmd (RCC_APB2Periph_GPIOA | B | C, DISABLE):释放GPIO复位。\x0d\x0a\x0d\x0a 4.2 置各个PIN端口(模拟输入_AIN、输入浮空_IN_FLOATING、输入上拉_IPU、输入下拉_IPD、开漏输出_OUT_OD、推挽式输出_OUT_PP、推挽式复用输出_AF_PP、开漏复用输出_AF_OD)。\x0d\x0a\x0d\x0a 4.3GPIO初始化完成。\x0d\x0a\x0d\x0a五、 的GPIO *** 作函数\x0d\x0a\x0d\x0auint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)//读GPIO某一位的输入\x0d\x0a\x0d\x0auint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx)//读GPIO的输入\x0d\x0a\x0d\x0auint8_t GPIO_ReadOutputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)//读GPIO某一位的输出\x0d\x0a\x0d\x0auint16_t GPIO_ReadOutputData(GPIO_TypeDef* GPIOx)//读GPIO的输出\x0d\x0a\x0d\x0avoid GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)//将GPIO的某个位置位\x0d\x0a\x0d\x0avoid GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)//将GPIO的某个位复位\x0d\x0a\x0d\x0avoid GPIO_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, BitAction BitVal)//写GPIO的某个位\x0d\x0a\x0d\x0avoid GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal)//写GPIO\x0d\x0a\x0d\x0a 六、管脚的复用功能 重映射\x0d\x0a\x0d\x0a1、复用功能:内置外设是与I/O口共用引出管脚(不同的功能对应同一管脚)\x0d\x0a\x0d\x0aSTM32 所有内置外设的外部引脚都是与标准GPIO引脚复用的,如果有多个复用功能模块对应同一个引脚,只能使能其中之一,其它模块保持非使能状态。\x0d\x0a\x0d\x0a2、重映射功能:复用功能的引出脚可以通过重映射,从不同的I/O管脚引出,即复用功 能的引出脚位是可通过程序改变到其他的引脚上!\x0d\x0a\x0d\x0a直接好处:PCB电路板的设计人员可以在需要的情况下,不必把某些信号在板上绕一大圈完成联接,方便了PCB的设计同时潜在地减少了信号的交叉干扰。\x0d\x0a\x0d\x0a如:USART1: 0: 没有重映像(TX/PA9,RX/PA10); 1: 重映像(TX/PB6,RX/PB7)。\x0d\x0a\x0d\x0a(参考AFIO_MAPR寄存器介绍)[0,1为一寄存器的bit值]\x0d\x0a\x0d\x0a【注】 下述复用功能的引出脚具有重映射功能:\x0d\x0a\x0d\x0a - 晶体振荡器的引脚在不接晶体时,可以作为普通I/O口\x0d\x0a\x0d\x0a - CAN模块; - JTAG调试接口;- 大部分定时器的引出接口; - 大部分USART引出接口\x0d\x0a\x0d\x0a - I2C1的引出接口; - SPI1的引出接口;\x0d\x0a\x0d\x0a举例:对于STM32F103VBT6,47引脚为PB10,它的复用功能是I2C2_SCL和 USART3_TX,表示在上电之后它的默认功能为PB10,而I2C2的SCL和USART3的TX为它的复用功能;另外在TIM2的引脚重映射后,TIM2_CH3也成为这个引脚的复用功能。\x0d\x0a\x0d\x0a(1)要使用STM32F103VBT6的47、48脚的USART3功能,则需要配置47脚为复用推挽输出或复用开漏输出,配置48脚为某种输入模式,同时使能USART3并保持I2C2的非使能状态。\x0d\x0a\x0d\x0a(2)使用STM32F103VBT6的47脚作为TIM2_CH3,则需要对TIM2进行重映射,然后再按复用功能的方式配置对应引脚。


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/12375742.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-23
下一篇2023-05-23

发表评论

登录后才能评论

评论列表(0条)

    保存