找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7048|回复: 0
打印 上一主题 下一主题
收起左侧

stm32 NAND闪存成功读取ID程序

[复制链接]
跳转到指定楼层
楼主
ID:82781 发表于 2015-6-14 01:32 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
今天对我意义很重大,终于把NAND闪存的ID号给读出来了,这就意味着我的时序是没有问题的,操作也是没有问题的,就这么干以后,当我读出数据:AD F1 80 1D,沸腾了。也许这就是我想要的,这样的快乐,知足了。
这是我用IO驱动外部器件的第一步,也是很值得纪念的日子,
下面是程序:(这里我采用的是库代码与裸寄存器操作)

  1. #include "stm32f10x.h"
  2. #include "stm32f10x_flash.h"
  3. #define SET_CLE GPIO_SetBits(GPIOD, GPIO_Pin_6)
  4. #define CLR_CLE GPIO_ResetBits(GPIOD, GPIO_Pin_6)
  5. #define SET_ALE GPIO_SetBits(GPIOD, GPIO_Pin_5)
  6. #define CLR_ALE GPIO_ResetBits(GPIOD, GPIO_Pin_5)
  7. #define SET_CE GPIO_SetBits(GPIOD, GPIO_Pin_7)
  8. #define CLR_CE GPIO_ResetBits(GPIOD, GPIO_Pin_7)
  9. #define SET_WE GPIO_SetBits(GPIOD, GPIO_Pin_14)
  10. #define CLR_WE GPIO_ResetBits(GPIOD, GPIO_Pin_14)
  11. #define SET_RE GPIO_SetBits(GPIOD, GPIO_Pin_15)
  12. #define CLR_RE GPIO_ResetBits(GPIOD, GPIO_Pin_15)
  13. void mysysinit(void);//系统时钟初始
  14. void my_USART_init(void);//初始化
  15. void my_send_byte(unsigned char send_date); //发送一个字节
  16. GPIO_InitTypeDef GPIO_InitStructure;
  17. void READ_NAND_ID(void);
  18. void delay_1us(uint32_t);
  19. void my_NAND_INIT(void ); //nand flash 初始化
  20. void wait_NAND_readay(void);

  21. int main()
  22. {
  23.      


  24. mysysinit();//RCC初始化,系统时钟设置72MHZ
  25.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能APB2的GPIO_D时钟
  26.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);//使能APB2的GPIO_D时钟
  27.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//使能APB2的GPIO_B时钟
  28. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE, ENABLE);//使能APB2的GPIO_E时钟
  29. RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);//使能APB2的GPIO_E时钟
  30.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//使能APB1的USART2时钟
  31.   /* led*/
  32.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9| GPIO_Pin_10| GPIO_Pin_11;
  33.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  34.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  35.   GPIO_Init(GPIOD, &GPIO_InitStructure);
  36.   /* 设置PB.5用于控制NAND的为上拉输入 */
  37.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
  38.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  39.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  40.   GPIO_Init(GPIOB, &GPIO_InitStructure);
  41.    my_USART_init();
  42.             
  43.    
  44.    
  45.   

  46.         wait_NAND_readay();
  47.     GPIO_SetBits(GPIOD, GPIO_Pin_8);

  48.                 my_send_byte(0);
  49.           my_send_byte(1);
  50.        my_send_byte(2);
  51.     my_send_byte(3);

  52.    
  53. READ_NAND_ID();
  54.   while(1);

  55. }
  56. /*等待芯片不忙*/
  57. void wait_NAND_readay()
  58. {

  59. uint8_t wait=0;//忙闲信号
  60. //等待芯片不忙
  61.    do
  62.    { wait=GPIO_ReadOutputDataBit(GPIOB, GPIO_Pin_5);//读取忙闲引脚
  63.    }
  64.     while(0x00==wait);

  65. }
  66. /*读取NAND闪存的ID序列号串口发送*/
  67. void READ_NAND_ID()
  68. {

  69. uint8_t a=0,b=0,c=0,d=0;

  70.   
  71.    /* 设置PD口用于控制NAND的为输出 */
  72.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6| GPIO_Pin_7| GPIO_Pin_14| GPIO_Pin_15;
  73.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  74.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  75.   GPIO_Init(GPIOD, &GPIO_InitStructure);
  76. /*设置PE口得低八位为输出*/
  77.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1| GPIO_Pin_2| GPIO_Pin_3| GPIO_Pin_4|GPIO_Pin_7 | GPIO_Pin_6| GPIO_Pin_7;
  78.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  79.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  80.   GPIO_Init(GPIOE, &GPIO_InitStructure);
  81. //delay_1us(1000000);
  82.   CLR_CE;//开启片选
  83.   SET_CLE;//命令锁存开启
  84.   CLR_WE;//写使能
  85.   CLR_ALE;//地址锁存关闭
  86.   SET_RE;//读关闭
  87.   GPIO_Write(GPIOE, 0x90); //读ID命令
  88.   SET_WE;//关闭写
  89.   CLR_CLE;
  90.   SET_ALE;
  91.   CLR_WE;
  92.   GPIO_Write(GPIOE, 0x00); //地址00
  93.   SET_WE;




  94.   /*设置PE口得低八位为输入*/
  95.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1| GPIO_Pin_2| GPIO_Pin_3| GPIO_Pin_4|GPIO_Pin_7 | GPIO_Pin_6| GPIO_Pin_7;
  96.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  97.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  98.   GPIO_Init(GPIOE, &GPIO_InitStructure);

  99.   CLR_ALE; //注意啊,这里一定把他放在下面应为是有时间要求的;时序图TAR=15纳秒之后才是RE的操作;
  100.        //读出的结果是AD F1 80 1D ,2011年8月26日,与天津第四项目部宿舍
  101.   CLR_RE;

  102.    a=(GPIOE->IDR);
  103.   SET_RE;

  104.     CLR_RE;
  105.   b=(GPIOE->IDR);
  106.   SET_RE;

  107.     CLR_RE;
  108.    c=(GPIOE->IDR);
  109. SET_RE;

  110.   CLR_RE;
  111.     d=(GPIOE->IDR);//delay_1us(1);
  112. SET_RE;


  113.   SET_CE;//关闭片选

  114.                 my_send_byte(a);
  115.           my_send_byte(b);
  116.        my_send_byte(c);
  117.     my_send_byte(d);
  118. }


  119. /********************
  120.   延时函数
  121.   形参:uint8_t time
  122.   功能:延时数值为time值

  123. *********************/
  124. void delay_1us(uint32_t time)
  125. {   uint32_t b,c;
  126.         for(c=time;c;c--) //定时=time*3*333.6=time*1000.8ns
  127.          {     
  128.    for(b=8;b;b--); //8*13.9*3= 333.6     
  129.          }



  130. }
  131. /***********************************

  132. 发送一个字节函数通过串口

  133. ************************************/
  134. void my_send_byte(unsigned char send_date )
  135. {

  136.   while( (USART1->SR&0x00000080)!=0x80);//发送寄存器为空
  137.    USART1->DR=send_date;


  138. }
  139. /**********************************
  140.           初始化串口

  141. **********************************/
  142. void my_USART_init()
  143. {

  144. /*USART2的优先级设为5*/
  145. NVIC->IP[37]=5;
  146. /*开启38号中断即USART2,关闭其他所有外部的中断*/
  147. NVIC->ISER[1]=0x00000020;   
  148. /*设置复用模式下的引脚模式为全双工:TX输出推挽复用,RX为输入上拉模式,速度50MHZ*/
  149.             GPIOA->CRH=0x000008b0;
  150.   /* 1.开启USART,
  151. *
  152. */
  153. USART1->CR1=0x2000;
  154. /* 1.关闭局域网模式
  155. * 2.1个停止位
  156. * 3.CK引脚禁能
  157. */
  158. USART1->CR2=0;
  159. /* 1.关闭调制解调模式
  160. * 2.关闭DMA模式
  161. * 3.关闭智能卡、红外模式
  162. *   4.关闭错误中断

  163. */
  164. USART1->CR3=0;
  165. /*     波特率设置

  166.      2011年8月11日
  167.         王均伟
  168.          天津第四项目部宿舍

  169.     BRR中的第四位(DIV_Fraction)作为小数,高12位(DIV_MANtissa)作为整数部分,
  170.   
  171.     1,根据公式:波特率=fck/16*usardiv,其中usardivBRR寄存器的值,所以变形得:USARDIV=fck/16*波特率
  172.     2.算出来BRR寄存器的值后就要把这个值变成16进制数据写入BRR寄存器中,
  173.       遵循以下规则:
  174.       小数部分*16=DIV_Fraction或者取近似的值
  175.       整数部分直接=DIV_MANtissa
  176.     3.把这个16进制值写入BRR寄存器
  177.     例如我要算波特率设成9600bps的BRR寄存器值,
  178.     1.先求USARDIV=36000000/16*9600=234.375
  179.     2.换成十六进制:DIV_Fraction=16*0.375=0x6
  180.                     DIV_MANtissa=234=0xea
  181.     3.组合并写入寄存器
  182.                      USART2->BRR=0x0ea6;值得注意的是这里是16位半字操作,所以不要以为是32位。

  183. */
  184. USART1->BRR=0x0ea6;

  185. /* 1.开启USART
  186. * 2.开启接收完毕中断
  187. * 3.开启发送功能
  188. *   4.开启接收功能
  189. */
  190. USART1->CR1=0x202c;


  191. }


  192. void mysysinit()//系统初始化程序
  193. {
  194. ErrorStatus HSEStartUpStatus;//说明标志位
  195. RCC_DeInit();//所有外设全部缺省设置

  196. /* Enable HSE */
  197. RCC_HSEConfig(RCC_HSE_ON);
  198. /* Wait till HSE is ready and if Time out is reached exit */
  199. HSEStartUpStatus = RCC_WaitForHSEStartUp();
  200. if(HSEStartUpStatus == SUCCESS)//启动成功
  201. {
  202. /*这两条FLASH指令必须加上,不知为啥?不加上就运行几秒后出错,参照系统初始化*/
  203. /* Enable The Prefetch Buffer */
  204. FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);//FLASH缓存开启
  205. /* Configure the Latency cycle: Set 2 Latency cycles */
  206.   FLASH_SetLatency(FLASH_Latency_2);  //设置FLASH这些位表示SYSCLK(系统时钟)周期与闪存访问时间的比例,为010:两个等待状态,当 48MHz < SYSCLK ≤ 72MHz
  207. /* Set PLL clock output to 72MHz using HSE (8MHz) as entry clock */
  208. RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);//外部时钟为8M,PLL的输入时钟=8MHZ,倍频系数9,

  209. /* Configure HCLK such as HCLK = SYSCLK */
  210. RCC_HCLKConfig(RCC_SYSCLK_Div1);//设置了啦AHB分频器的分频系数=1,即HCLK=SYSCLK=72MHZ
  211. /* Configure PCLK1 such as PCLK1 = HCLK/2 */
  212. RCC_PCLK1Config(RCC_HCLK_Div2);//设置了APB1外设的时钟频率最大是36M这里是APB1的分频器设为2,PCLK1=HCLK/2=72/2=36MHZ正好是最大值
  213. /* Configure PCLK2 such as PCLK2 = HCLK */
  214. RCC_PCLK2Config(RCC_HCLK_Div1);//设置PLCK2=HCLK=72MHZ,的APB2分频器=1
  215. /* Select the PLL as system clock source */
  216. RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);//设置了SYSCLK的提供者为PLL,频率由上面算出=72MHZ
  217. /* disable PLL Ready interrupt */
  218. RCC_ITConfig(RCC_IT_PLLRDY, DISABLE);//PLL中断关闭
  219. /* disable PLL Ready interrupt */
  220. RCC_ITConfig(RCC_IT_HSERDY,DISABLE);//HSE中断关闭
  221. /* disable PLL Ready interrupt */
  222. RCC_ITConfig(RCC_IT_HSIRDY, DISABLE); //HSI中断关闭
  223. /* disable PLL Ready interrupt */
  224. RCC_ITConfig(RCC_IT_LSERDY, DISABLE); //LSE中断关闭
  225. /* disable PLL Ready interrupt */
  226. RCC_ITConfig(RCC_IT_LSIRDY, DISABLE); //LSI中断关闭

  227. /* PLL clock divided by 1.5 used as USB clock source */
  228. RCC_USBCLKConfig(RCC_USBCLKSource_PLLCLK_1Div5);//设置USB的时钟为=72、1.5=48mhz
  229. /* Configure ADCCLK such as ADCCLK = PCLK2/2 */
  230. RCC_ADCCLKConfig(RCC_PCLK2_Div2);//设置ADC时钟=PCLK2/2= 36MHZ
  231. /* disable the LSE */
  232. RCC_LSEConfig(RCC_LSE_OFF);//外部低速晶振关闭

  233. /*DISable the RTC clock */
  234. RCC_RTCCLKCmd(DISABLE);
  235. /* DISable the Clock Security System */
  236. RCC_ClockSecuritySystemCmd(DISABLE);
  237. /* Enable the PLL */
  238. RCC_PLLCmd(ENABLE);//使能PLL







  239. /* PLL ans system clock config */
  240. }
  241. else
  242. {
  243. /* Add here some code to deal with this error */
  244. }
  245. }
复制代码




分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表