一、FSMC之我见
开端仅仅谈到他人对FSMC的了解,留意这儿只评论FSMC操控TFT,也便是在FSMC的NOR\PSRAM形式操控LCD,所以咱们以下的剖析都是根据这种形式的。
1、咱们之前经过运用GPIO来模仿8080/6800时序然后到达驱动彩屏的,相同需求理解的一点便是咱们也仅仅运用FSMC来模仿8080/6800时序,只不过这个读写速度有些快(运用了总线嘛),仅此而已!
简略一点便是:8080是经过“读使能(RE)”和“写使能(WE)”两条操控线进行读写操作。 6800是经过“总使能(E)”和“读写挑选(W/R)”两条操控线进行
2、那么了解到FSMC的三总线如下!
数据线:这个可以分为8位的和16位,这个不难了解,便是之指一次穿上红8位仍是16位数据,我的是16位的,8位的有一个懒得用。
地址线:已然咱们拜访的外NOR FLASH,那么必定会有相应的地址线,那么这些地址线在哪里呢?必定是经过GPIO引脚复用的。有A0 — A23 24根,可以操控拜访16M的空间,也便是一个子bank;
操控总线:它的操控总线只要三根:读使能信号,写使能信号,片选信号。所以这儿和咱们8080时序比较,少了复位信号线和数据/指令操控线,怎么办?持续看!
3、了解了FSMC会有这三总线的概念,那么接下来便是怎么转化为咱们需求的时序了。
比照一下FSMC拜访外nor flash和8080拜访时序如下
不同好像很小是吧,简略说便是在数据/指令挑选和复位信号上的差异。
4、在这儿咱们运用的软件方法来完善FSMC转化为8080的读写时序
在参考手册上的存储系统能找到,芯片留给咱们外扩的存储器(NOR FLASH、PSRAM这类可直接寻址的器
件)地址是从0x60000000开端的,意思便是当咱们拜访0x60000000的时分,那便是相当于拜访外部nor flash了(咱们只评论这种状况),那么他就会主动发生FSMC的时序
在这儿,咱们所需求增加的便是D/C挑选信号,怎么完成呢?咱们是经过,一根地址线来完成的,当咱们把A0多对应的GPIOF0(引脚默许复用)接到TFT的RS端,
然后履行拜访0x60000000的指令,那么RS是否便是低电平挑选为数据呢?又参加咱们拜访的地址是0x60000001的时分,那么RS是否便是高电平,然后挑选的便是指令传送呢?答案当然是必定的!所以咱们就处理了这个问题,复位信号就更好处理了,直接和开发板的复位引脚接在一同就好了,就这么简略!
二、说了这么久的理论,来个实例剖析愈加形象了,首要硬件连线要理解
在原理图或许开发手册上面可以确认引脚复用问题
地址引脚:
(A0-A5 :PF0 – PF5) (A6-A9: F12-F15 ) (A10-A15:PG0-PG5)
(A16-A18:PD11-PD13) (A19-A22:PE3-PE6) (A23-PE2)
片选信号(NEx:PG12)由于我挑选的是block4
写使能(NWR:PD5)
读使能(NOE:PD4)
至此操控引脚基本完成
下面便是数据引脚:
PD14-FSMC-D0 —-LCD-DB0
PD15-FSMC-D1 —-LCD-DB1
PD0-FSMC-D2 —-LCD-DB2
PD1-FSMC-D3 —-LCD-DB3
PE7-FSMC-D4 —-LCD-DB4
PE8-FSMC-D5 —-LCD-DB5
PE9-FSMC-D6 —-LCD-DB6
PE10-FSMC-D7 —-LCD-DB7
PE11-FSMC-D8 —-LCD-DB8
PE12-FSMC-D9 —-LCD-DB9
PE13-FSMC-D10 —-LCD-DB10
PE14-FSMC-D11 —-LCD-DB11
PE15-FSMC-D12 —-LCD-DB12
PD8-FSMC-D13 —-LCD-DB13
PD9-FSMC-D14 —-LCD-DB14
PD10-FSMC-D15 —-LCD-DB15
有了这些硬件连线是没有任何问题的
三、正式剖析程序
1、硬件引脚装备函数
voidLCD_CtrlLinesConfig(void)
{
GPIO_InitTypeDefGPIO_InitStructure;
/*EnableFSMC,GPIOD,GPIOE,GPIOF,GPIOGandAFIOclocks*/
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_FSMC,ENABLE);//使能FSMC
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD|RCC_APB2Periph_GPIOE|
RCC_APB2Periph_GPIOF|RCC_APB2Periph_GPIOG|
RCC_APB2Periph_AFIO,ENABLE);
//IO口复用功用时钟
/*SetPD.00(D2),PD.01(D3),PD.04(NOE),PD.05(NWE),PD.08(D13),PD.09(D14),
PD.10(D15),PD.14(D0),PD.15(D1)asalternate
funcTIonpushpull*/
/*D端口初始化*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_4|GPIO_Pin_5|
GPIO_Pin_8|GPIO_Pin_9|GPIO_Pin_10|GPIO_Pin_14|
GPIO_Pin_15;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOD,&GPIO_InitStructure);
/*E端口初始化*/
/*SetPE.07(D4),PE.08(D5),PE.09(D6),PE.10(D7),PE.11(D8),PE.12(D9),PE.13(D10),
PE.14(D11),PE.15(D12)asalternatefuncTIonpushpull*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_7|GPIO_Pin_8|GPIO_Pin_9|GPIO_Pin_10|
GPIO_Pin_11|GPIO_Pin_12|GPIO_Pin_13|GPIO_Pin_14|
GPIO_Pin_15;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOE,&GPIO_InitStructure);//将装备写入GPIOE管脚
/*A0地址线*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOF,&GPIO_InitStructure);
/*SetPG.12(NE4(LCD/CS))asalternatefuncTIonpushpull-CE3(LCD/CS)*/
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_12;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
GPIO_Init(GPIOG,&GPIO_InitStructure);