历史上的今天
返回首页

历史上的今天

今天是:2025年01月22日(星期三)

正在发生

2020年01月22日 | STM32—FMSC详解

2020-01-22 来源:eefocus

这个是从网上找到的,内容如下:

STM32 FMSC  LCD难点解析:

以下是网上和自己整理的:感觉应该可以把STM32 ----FSMC LCD中的关键RS说清楚~

----------------------------------------------------------------------------------------------------------------------

第一个角度理解STM32有FSMC(其实其他芯片基本都有类似的总线功能),FSMC的好处就是你一旦设置好之后,WR(写)、RD(读)、DB0-DB15这些控制线和数据线,

都是FSMC自动控制的。打个比方,当你在程序中写到:

 

*(volatile unsigned short int *)(0x60000000)=val;

 

那么FSMC就会自动执行一个写的操作,其对应的主控芯片的WE、RD这些脚,就会呈现出写的时序出来(即WE=0,RD=1),数据val的值也会通过DB0-15自动呈现出来(即FSMC-D0:FSMC-D15=val  )。地址0x60000000会被呈现在数据线上(即A0-A25=0,地址线的对应最麻烦,要根据具体情况来,好好看看FSMC手册)。

 

那么在硬件上面,我们需要做的,仅仅是MCU和LCD控制芯片的连接关系:

 

WE-WR,均为低电平有效

RD-RD,均为低电平有效

FSMC-D0-15接LCD DB0-15

FSMC_NE1--CS接PD7

 

连接好之后,读写时序都会被FSMC自动完成。但是还有一个很关键的问题,就是RS没有接因为在FSMC里面,根本就没有对应RS。怎么办呢?这个时候,有一个好方法,就是用某一根地址线来接RS。比如我们选择了A16这根地址线来接,那么当我们要写寄存器的时候,我们需要RS,也就是A16(RS为高)置高。软件中怎么做呢?也就是将FSMC要写的地址改成0x60020000,如下:

 

*(volatile unsigned short int *)(0x60020000)=val;

 

这个时候,A16在执行其他FSMC的同时会被拉高,因为A0-A18要呈现出地址0x60020000。0x60020000里面的Bit17=1,就会导致A16为1。


当要读数据时,地址由0x60020000改为了0x60000000,这个时候A16就为0了。

 

那么有朋友就会有疑问,第一,为什么地址是0x6xxxxxxx而不是0x0xxxxxxx;第二,CS怎么接;第三,为什么Bit17对应A16?

 

RS问题:RS为0表示;读写寄存器;RS为1,读写数据RAM;

 

先来看前两个问题,大家找到STM32的FSMC手册,在FSMC手册里面,我们很容易找到,FSMC将0x60000000-0x6fffffff的地址用作NOR/PRAM(共256M地址范围)。而这个存储块,又被分成了四部分,每部分64M地址范围。当对其中某个存储块进行读写时,对应的NEx就会置低。这里,就解决了我们两个问题,第一,LCD的操作时序,和NOR/PRAM是一样的(为什么一样自己找找NOR/PRAM的时序看看),所以我们选择0x6xxxxxxx这个地址范围(选择这个地址范围,操作这个地址时,FSMC就会呈现出NOR/PRAM的时序)。第二,我们可以将NEx连接到LCD的CS,只要我们操作的地址是第一个存储块内即可(即0-0x3ffffff地址范围)。 

 

第三个问题再来看一看FSMC手册关于存储器字宽的描述,我们发现,当外部存储器是16位时,硬件管脚A0-A24表示的是地址线A1-A25的值,所以我们要位移一下,Bit17的值,实际会被反应到A16这根IO来。关于数据宽度及位移的问题,初学的朋友可能会比较疑惑,当你接触了多NOR/PRAM这样的器件后,你会发现,很多芯片的总线,都是这样设计的,为的是节省地址线。

 

第二个角度理解:


FSMC总线上看,LCD只有2个地址.

Bank1_LCD_C是写寄存器,此时RS=1,告诉LCD我在总线上输出数据的是寄存器的地址

Bank1_LCD_D是写数据,此时RS=0,告诉LCD我在总线上输出地数据是寄存器的数据或者GRAM的数据.

 

写寄存器数据按2步来:

第一步先往Bank1_LCD_C (对应RS=1),送寄存器的地址:*(__IO uint16_t *) (Bank1_LCD_C)= index;  接着在Bank1_LCD_D这个地址(对应RS=0),写入刚指向的寄存器的数据: *(__IO uint16_t *) (Bank1_LCD_D)= val;

 

为什么*(__IO uint16_t *) (Bank1_LCD_C)= index; 就是往 LCD 写寄存器呢?


这是一个16位的IO赋值操作,地址是Bank1_LCD_C,这个地址就是指向FSMC的 Bank1的NE1对应的地址空间。而LCD片选正是连接到NE1,具体地址要看RS接到哪一根地址线上。当CPU执行到这一条的时候,就会通过FSMC总线控制器在数据总线上进行一个地址为 Bank1_LCD_C的数据写操作,此操作自动完成CS信号,RD信号,WR信号,以及地址总线数据(RS信号)的输出以及数据总线数据的输出.

 

其他的操作都是这两个操作组合完成。也就是我上面所说的,"所有的寄存器地址和寄存器数据,以及 GRAM数据都是通过 IO0-IO15完成传输的,而不是FSMC的地址.这是容易搞混的一个地方.LCD的FSMC地址只有一根 ,就是RS."

----------------------------------------------------------------------------------------------------------------------第三个角度理解:

把TFT看做类似SRAM的存储器,只能接在 BANK1上。对应基地址是0x60000000.

而BANK1又有划分为四个片选,分别对应基地址:

NE1 0x600000000

NE2 0x640000000

NE3 0x680000000

NE4 0x6C0000000

所以每个NEx能寻址的空间大小为64M,也就是对应了FSMC的A0到A25 共26根地址线.

 

假如使用NE4接到为LCD的片选CS上,那么就对应基地址 0x6C000000,

如果RS接到地址线的 A0上,那么当 RS为0时对应的地址就是 LCD_REG = 0x6C000000,(其实你用0x6CFFFFF0是一样的,因为只用到一根地址线).

RS为1时对应的地址就是 LCD_RAM =0x6C000001,(0x6CFFFFF1一样对应 LCD_RAM,因为它一样对应 RS=1).

 

如果 RS接到 其他地址线上,情况是类似的。

比如接到 An上,那么

LCD_REG= 0x6C000000,

LCD_RAM= 0x6C000000 | (1< 

注意这个地址不是唯一的,只要这个地址能寻址到 BANK1 的 NE4上而且使 RS=0,那么就是 LCD_REG,使 RS=1,就是LCD_RAM.

----------------------------------------------------------------------------------------------------------------------

对应Bank1_LCD_C 的地址,FSMC总线控制器在RS接的那根地址线输出的是 1,而对应Bank1_LCD_D,输出的0.


RS接的可不是GPIO,是FSMC地址总线的一根.FSMC进行读写操作的时候会在地址总线根据要读写的地址输出电平的.


RS接哪一根地址线虽然没有固定要求,但是一旦你确定要接哪一根,那么Bank_LCD_C和Bank_LCD_D也要随之确定,这可不是“自动的".

 

虽然没有手动操作GPIO来操作RS,但是你敲代码的时候可是手动指定 Bank1_LCD_C 或者 Bank1_LCD_D ,从而确定 RS的电平.


所谓的“自动”是指:不是通过操作GPIO来操作RS,而是直接根据地址总线地址的不同来完成操作RS,这两种方法的速度差别是非常大的.


如果是GPIO方式,先要通过操作GPIO 分别 输出 RS,CS,等的电平,然后再通过过GPIO操作输出数据,然后还要通过GPIO 再操作RD,WR,CS等的电平。


每操作一个GPIO都要好几个周期,加起来就非常慢了.

而FSMC是在一个FSMC写周期内就完成了这所有的动作。


*******************************************************************************

#define Bank1_LCD_R    ((uint32_t)0x60000000)    //disp Reg ADDR

#define Bank1_LCD_D    ((uint32_t)0x60020000)   //disp Data ADDR

 

从STM32 FMSC系统手册可以看到:

 

FSMC其实就相当于外部总线存储器和内部AHB总线的接口:而AHB是32位的,当外接NOR/LCD 时,而外部存储器的数据宽度可以选择8位和16位的,这时候就存在一个地址转换的问题即32位和8位或者16位地址转换的问题。解决这个问题STM32采用的HADDR[25:0],它的作用就是将外部存储器地址转换为AHB地址线。

推荐阅读

史海拾趣

埃派克森微电子(Apexone)公司的发展小趣事

在埃派克森微电子的发展过程中,著名投资机构多尔基金和华登国际的风险投资起到了重要的推动作用。这些投资不仅为公司提供了资金支持,还带来了宝贵的市场资源和管理经验。借助这些资源,埃派克森加快了国际化步伐,积极拓展海外市场,与全球范围内的客户和合作伙伴建立了紧密的合作关系。

Astema公司的发展小趣事

随着技术的不断成熟,Astema开始积极拓展市场。公司制定了一系列市场策略,包括与大型电子设备制造商建立合作关系、参加国际电子展等,以扩大品牌影响力。通过这些努力,Astema逐渐打开了国际市场的大门,其产品远销海外,市场份额稳步提升。

EMC Technology RF Labs公司的发展小趣事

随着对射频技术的深入研究,RF Labs在多个领域取得了技术突破。他们成功开发了一系列高性能的射频电阻、衰减器、定向耦合器等关键元件,这些产品不仅具有优异的性能,而且能够满足各种复杂的应用场景需求。同时,RF Labs还不断推出创新产品,如智能探测器温度测量端子等,这些产品为客户提供了更加便捷和高效的解决方案。

Advanced Power Solutions公司的发展小趣事

Advanced Power Solutions公司非常重视人才培养和企业文化建设。公司注重员工的职业发展和培训,为员工提供了广阔的职业发展空间和学习机会。同时,公司还倡导开放、创新的企业文化,鼓励员工提出新的想法和建议。这种积极向上的企业文化吸引了大量优秀人才的加入,也为公司的持续发展提供了强大的动力。

以上五个故事框架仅用于说明Advanced Power Solutions公司可能的发展路径和成就,并不代表公司的实际历史。在实际撰写时,需要根据公司的具体情况进行调整和补充。

GE Solid State公司的发展小趣事
由于声音信号的模糊性和主观性,可能无法精确反映曝光量的细微变化。
Conexcon Group公司的发展小趣事

在电子行业的发展过程中,环保问题日益受到关注。Conexcon Group积极响应国家环保政策,将绿色环保理念贯穿于产品研发、生产和销售的全过程。公司采用了环保材料和工艺,减少了对环境的污染和破坏。同时,公司还加强了废旧电子产品的回收和处理工作,推动了电子行业的可持续发展。这些举措不仅赢得了消费者的认可和支持,也为公司树立了良好的社会形象。

问答坊 | AI 解惑

求救:tornado 2.2 powerpc c++驱动问题。

下面是编译信息: vxrm *.o *.rpo ctdt.c symTbl.c vxApp* *.out *.pl vxrm ..\\prjComps.h ..\\prjParams.h ..\\prjConfig.c ..\\linkSyms.c vxrm ..\\libs.nm ..\\libs.size F:\\tornado2.2\\host\\gnu\\3.3\\x86-win32\\bin\\ccppc -g -mcpu ...…

查看全部问答>

2410 usb 电路请教

怎么发布了电路图啊? 我的usb调试部出来想让帮忙看看 怎么发电路图啊…

查看全部问答>

关于WDB和系统启动

小弟是新接触vxWorks系统的新手,请问有哪位哥哥姐姐有关于WDB的资料么?还有,请问谁有关于vxWorks启动时的资料啊,比如系统先加载什么,然后运行什么,这方面的资料,很急,先谢谢大家了!…

查看全部问答>

想自己制作一块51的开发板,却不知道如何下手?请教师兄们

准备学习单片机,本想买一块开发板,不过想到自己搭的话可以熟悉硬件部分,不过我还没有实战过实际电路,所以感到一头雾水,请教高手具体步骤怎么干?谢谢啊…

查看全部问答>

新建一超级群:26993639,欢迎加入

本群主题:研究led阻容降压,探讨led驱动电源等技术话题--------   格式:职业+公司+姓.男+R女+S   欢迎加入…

查看全部问答>

求助:自己设计的IIR滤波器

昨天自己设计了一个IIR滤波器,写了个测试程序:滤波器是一个低通的滤波器,要求滤除20KHZ以上的频率段, 程序如下: #include \"filter.h\" #include \"iir.h\" #include \"math.h\" float xn[400],yn[400],zn[400],rn[400]; #pragma DATA_S ...…

查看全部问答>

ADI实验室电路

ADI实验室电路 下载…

查看全部问答>

大家开发LPC1114的时候都是用什么开发模式呢?

大家开发LPC1114的时候都是用什么开发模式呢?从编程环境到编译器再到仿真器…

查看全部问答>

谈F28M35的M3侧的blink例程

从简单的谈起吧。     1、程序先进行了HWREG(SYSCTL_MWRALLOW) =  0xA5A5A5A5;  因为 //必须写入0xA5A5A5A5,之后才能够改写一些被MWRALLOW保护的寄存器   关于哪些寄存器被保护可以参考1.13.1 System Control, Conf ...…

查看全部问答>