久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 6160|回復: 0
打印 上一主題 下一主題
收起左側

stm32 ARM之以太網通信底層接口

[復制鏈接]
跳轉到指定樓層
樓主
ID:82781 發表于 2015-6-13 16:52 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
  1. /*
  2.   ******************************************************************************
  3.   * 文件:   main.c
  4.   * 作者:   王均偉
  5.   * 固件庫版本 V3.5.0
  6.   * 時間日期    2012年3月9日
  7.   * 功能:APP
  8.   */
  9. /* Includes ------------------------------------------------------------------*/
  10. #include "stm32f10x.h"
  11. #include "main.h"
  12. #define SET_CS GPIOC->BSRR|=0x1000
  13. #define CLR_CS GPIOC->BRR|= 0x1000
  14. #define SET_EN GPIOD->BSRR|=0x0001
  15. extern unsigned char Receiver_date;//全局變量定義用于串口接收數據
  16. GPIO_InitTypeDef GPIO_InitStructure;//用于定義IO端口
  17. int main(void)
  18. {
  19.     unsigned char dat;
  20. SPI_PORT_INIT();
  21. //my_send_byte(0x88);
  22. //my_send_byte(0x88);
  23. my_send_byte(0x88);
  24. my_send_byte(0x88);
  25. SET_EN;
  26.     CLR_CS;
  27.     /**這個SPI我得介紹下了
  28.     這個SPI不是模擬的SPI,所以呢他是這樣工作的,就不能以模擬的方式來
  29.     看待這個借口,他是這樣的比如你要寫入一個字節數據,那么他在SCLK時鐘的控制下
  30.     一位一位的移向對方的緩存區,對方的數據也以為一位的移向你的buff,也就是說,8個
  31.     SCLK可以產生兩個操作,第一主機發送數據從MOSI一位一位的進入從機,第二從機的數據一位一位
  32.     的通過MISO移出來,而模擬就不是這樣了,模擬直接忽略了MISO的數據,所以發的發送接收是分開的
  33.     比如這個地方如下圖
  34.            =|=分界線
  35.    0   1   2    3  4   5   6   7   0   1   2   3   4   5   6   7  
  36.   SCLK___|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_|-|_
  37.   MOSI__________________________________|---------------------------  =====0x00,0xff:  ff是廢物,00才是地址
  38.     0  0    0   0  0  0  0   0   1   1   1   1   1   1   1      |
  39.                       |
  40.   MISO--------------------------------------------------|_____|--|____=====0xff,0xfa :FF是廢物fa才是數據
  41.     1  1   1    1  1  1  1    1   1  1   1    1  0    1   0
  42.       2012年3月9日
  43.       王均偉
  44.       于日照
  45.       12:08
  46.           =|=分解線
  47.           圖中——————表示低電平
  48.           ----------表示高電平
  49.           也就是說你在寫00的同時從機給你一個FF,
  50.           但是這不是我想要的,因為這個FF毫無意義
  51.           我就在寫一次,這次我寫進入一個毫無意義的
  52.           數FF,但是我接收的數就是我要的
  53.           所以這里一定注意!!
  54.           你要是只讀一次那肯定是不對的,
  55.    
  56.    
  57.     **/
  58.     dat=SPI_Write_Byte(0x00);
  59.     dat=SPI_Write_Byte(0xff);
  60.    //dat=SPI_Read_Byte();
  61.        SET_CS;
  62.    my_send_byte(dat);
  63.       
  64.    while(1);
  65. }
  66. /************
  67. SPI寫一個字節
  68. *************/
  69. uint16_t SPI_Write_Byte(uint16_t da)
  70. {
  71.      uint16_t b;
  72.     b=da;
  73.      my_send_byte(0x66);
  74.     my_send_byte(SPI1->SR);
  75.      while(1)
  76.   {
  77.   if((SPI1->SR&0x0080)==0x0000)break;
  78.     my_send_byte(0x01);
  79.   }//不忙?
  80.     my_send_byte(SPI1->SR);
  81.      while(1)
  82.   {
  83.   if((SPI1->SR&0x0002)==0x0002)break;
  84.     my_send_byte(0x02);
  85.   }//發送為空?
  86.     my_send_byte(SPI1->SR);
  87.   SPI1->DR=b;
  88.   //my_send_byte(SPI1->DR);
  89.      my_send_byte(SPI1->SR);
  90.    while(1)
  91.   {
  92.   if((SPI1->SR&0x0080)==0x0000)break;
  93.   my_send_byte(0x03);
  94.   }//不忙?
  95.       my_send_byte(SPI1->SR);
  96.      while(1)
  97.   {
  98.   if((SPI1->SR&0x0002)==0x0002)break;
  99.    my_send_byte(0x04);
  100.   }//發送為空?
  101.    my_send_byte(SPI1->SR);
  102.   b=SPI1->DR;
  103.   return(b);
  104. }
  105. /************
  106. SPI讀一個字節
  107. *************/
  108. /*uint16_t  SPI_Read_Byte()
  109. {
  110. uint16_t da;
  111.       my_send_byte(0x77);
  112.     my_send_byte(SPI1->SR);
  113.      while(1)
  114.   {
  115.   if((SPI1->SR&0x0080)==0x0000)break;
  116.    my_send_byte(0x11);
  117.   }//不忙?
  118.    my_send_byte(SPI1->SR);
  119.   while(1)
  120.   {
  121.   if((SPI1->SR&0x0001)==0x0001)break;
  122.       my_send_byte(0x12);
  123.   }//接收導數據?
  124.     my_send_byte(SPI1->SR);
  125. da=SPI1->DR;
  126.      my_send_byte(SPI1->SR);
  127.   while(1)
  128.   {
  129.   if((SPI1->SR&0x0080)==0x0000)break;
  130.   my_send_byte(0x13);
  131.   }//不忙?
  132.    my_send_byte(SPI1->SR);
  133.    while(1)
  134.   {
  135.   if((SPI1->SR&0x0001)==0x0000)break;
  136.    my_send_byte(0x14);
  137.   }//接收為空?
  138.    my_send_byte(SPI1->SR);
  139. return(da);

  140. }
  141. */
  142. /***************
  143. SPI接口初始化
  144. ***************/
  145. void SPI_PORT_INIT()
  146. {
  147. mysysinit();//系統時鐘初始化
  148. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能APB2的GPIO_A時鐘
  149. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);//使能APB2的GPIO_C時鐘
  150. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);//使能APB2的GPIO_D時鐘
  151.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,ENABLE);//使能APB2的SPI1時鐘
  152.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//使能APB2的USART1時鐘
  153. /*設置SPI1口的是SCK/MOSI =PA5\PA7,另設置一個IO設置為推挽復用相關引腳*/
  154.   GPIO_InitStructure.GPIO_Pin=GPIO_Pin_5|GPIO_Pin_7;                              
  155.   GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  156.   GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
  157.   GPIO_Init(GPIOA, &GPIO_InitStructure);
  158. /*設置SPI1口的嗎、MISO =PA6設置為上拉輸入引腳*/
  159.   GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6;                              
  160.   GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  161.   GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;
  162.   GPIO_Init(GPIOA, &GPIO_InitStructure);
  163.     /*設置PC12口的NET_CS =PC12設置為推挽輸出*/
  164.   GPIO_InitStructure.GPIO_Pin=GPIO_Pin_12;                              
  165.   GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  166.   GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
  167.   GPIO_Init(GPIOC, &GPIO_InitStructure);

  168. /*設置PD口用于控制;LED的為輸出*/
  169.   GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_8|GPIO_Pin_9 | GPIO_Pin_10| GPIO_Pin_11;                              
  170.   GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  171.   GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
  172.   GPIO_Init(GPIOD, &GPIO_InitStructure);
  173. my_USART_init();//串口初始化
  174.   SPI1->CR2=0;//禁止SPI中斷
  175.   SPI1->CR1=0x033c;//開啟SPI1,波特率設置為PCLK2/256=72M/256=0.28m
  176.   SPI1->I2SCFGR=0;//禁止I2S
  177.   SPI1->CR1=0x037c;//開啟SPI
  178. }
  179. /***********************************
  180. 發送一個字節函數通過串口
  181. ************************************/
  182. void my_send_byte(unsigned char send_date )
  183. {
  184.   while( (USART1->SR&0x00000080)!=0x80);//發送寄存器為空
  185.    USART1->DR=send_date;

  186. }
  187. /**********************************
  188.           初始化串口
  189. **********************************/
  190. void my_USART_init()
  191. {
  192. /*USART2的優先級設為5*/
  193. NVIC->IP[37]=5;
  194. /*開啟38號中斷即USART2,關閉其他所有外部的中斷*/
  195. NVIC->ISER[1]=0x00000020;   
  196. /*設置復用模式下的引腳模式為全雙工:TX輸出推挽復用,RX為輸入上拉模式,速度50MHZ*/
  197.             GPIOA->CRH=0x000008b0;
  198.   /* 1.開啟USART,
  199. *
  200. */
  201. USART1->CR1=0x2000;
  202. /* 1.關閉局域網模式
  203. * 2.1個停止位
  204. * 3.CK引腳禁能
  205. */
  206. USART1->CR2=0;
  207. /* 1.關閉調制解調模式
  208. * 2.關閉DMA模式
  209. * 3.關閉智能卡、紅外模式
  210. *   4.關閉錯誤中斷
  211. */
  212. USART1->CR3=0;
  213. /*     波特率設置
  214.      2011年8月11日
  215.         王均偉
  216.          天津第四項目部宿舍
  217.     BRR中的第四位(DIV_Fraction)作為小數,高12位(DIV_MANtissa)作為整數部分,
  218.   
  219.     1,根據公式:波特率=fck/16*usardiv,其中usardivBRR寄存器的值,所以變形得:USARDIV=fck/16*波特率
  220.     2.算出來BRR寄存器的值后就要把這個值變成16進制數據寫入BRR寄存器中,
  221.       遵循以下規則:
  222.       小數部分*16=DIV_Fraction或者取近似的值
  223.       整數部分直接=DIV_MANtissa
  224.     3.把這個16進制值寫入BRR寄存器
  225.     例如我要算波特率設成9600bps的BRR寄存器值,
  226.     1.先求USARDIV=36000000/16*9600=234.375
  227.     2.換成十六進制:DIV_Fraction=16*0.375=0x6
  228.                     DIV_MANtissa=234=0xea
  229.     3.組合并寫入寄存器
  230.                      USART2->BRR=0x0ea6;值得注意的是這里是16位半字操作,所以不要以為是32位。
  231. */
  232. USART1->BRR=0x0ea6;
  233. /* 1.開啟USART
  234. * 2.開啟接收完畢中斷
  235. * 3.開啟發送功能
  236. *   4.開啟接收功能
  237. */
  238. USART1->CR1=0x202c;

  239. }

  240. void mysysinit()//系統初始化程序
  241. {
  242.     ErrorStatus HSEStartUpStatus;//說明標志位
  243.      RCC_DeInit();//所有外設全部缺省設置
  244.    
  245.     /* Enable HSE */
  246.     RCC_HSEConfig(RCC_HSE_ON);
  247.     /* Wait till HSE is ready and if Time out is reached exit */
  248.     HSEStartUpStatus = RCC_WaitForHSEStartUp();
  249.     if(HSEStartUpStatus == SUCCESS)//啟動成功
  250.     {
  251.        /*這兩條FLASH指令必須加上,不知為啥?不加上就運行幾秒后出錯,參照系統初始化*/
  252.        /* Enable The Prefetch Buffer */
  253.         FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);//FLASH緩存開啟
  254.         /* Configure the Latency cycle: Set 2 Latency cycles */
  255.          FLASH_SetLatency(FLASH_Latency_2);  //設置FLASH這些位表示SYSCLK(系統時鐘)周期與閃存訪問時間的比例,為010:兩個等待狀態,當 48MHz < SYSCLK ≤ 72MHz
  256.         /* Set PLL clock output to 72MHz using HSE (8MHz) as entry clock */
  257.        RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);//外部時鐘為8M,PLL的輸入時鐘=8MHZ,倍頻系數9,
  258.       
  259.        /* Configure HCLK such as HCLK = SYSCLK */
  260.        RCC_HCLKConfig(RCC_SYSCLK_Div1);//設置了啦AHB分頻器的分頻系數=1,即HCLK=SYSCLK=72MHZ
  261.        /* Configure PCLK1 such as PCLK1 = HCLK/2 */
  262.        RCC_PCLK1Config(RCC_HCLK_Div2);//設置了APB1外設的時鐘頻率最大是36M這里是APB1的分頻器設為2,PCLK1=HCLK/2=72/2=36MHZ正好是最大值
  263.        /* Configure PCLK2 such as PCLK2 = HCLK */
  264.        RCC_PCLK2Config(RCC_HCLK_Div1);//設置PLCK2=HCLK=72MHZ,的APB2分頻器=1
  265.        /* Select the PLL as system clock source */
  266.        RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);//設置了SYSCLK的提供者為PLL,頻率由上面算出=72MHZ
  267.        /* disable PLL Ready interrupt */
  268.        RCC_ITConfig(RCC_IT_PLLRDY, DISABLE);//PLL中斷關閉
  269.        /* disable PLL Ready interrupt */
  270.        RCC_ITConfig(RCC_IT_HSERDY,DISABLE);//HSE中斷關閉
  271.        /* disable PLL Ready interrupt */
  272.        RCC_ITConfig(RCC_IT_HSIRDY, DISABLE); //HSI中斷關閉
  273.        /* disable PLL Ready interrupt */
  274.        RCC_ITConfig(RCC_IT_LSERDY, DISABLE); //LSE中斷關閉
  275.        /* disable PLL Ready interrupt */
  276.        RCC_ITConfig(RCC_IT_LSIRDY, DISABLE); //LSI中斷關閉
  277.       
  278.        /* PLL clock divided by 1.5 used as USB clock source */
  279.        RCC_USBCLKConfig(RCC_USBCLKSource_PLLCLK_1Div5);//設置USB的時鐘為=72、1.5=48mhz
  280.        /* Configure ADCCLK such as ADCCLK = PCLK2/2 */
  281.        RCC_ADCCLKConfig(RCC_PCLK2_Div2);//設置ADC時鐘=PCLK2/2= 36MHZ
  282.        /* disable the LSE */
  283.        RCC_LSEConfig(RCC_LSE_OFF);//外部低速晶振關閉
  284.       
  285.        /*DISable the RTC clock */
  286.        RCC_RTCCLKCmd(DISABLE);
  287.        /* DISable the Clock Security System */
  288.        RCC_ClockSecuritySystemCmd(DISABLE);
  289.        /* Enable the PLL */
  290.        RCC_PLLCmd(ENABLE);//使能PLL
  291.       
  292.       
  293.    
  294.     /* PLL ans system clock config */
  295.     }
  296.     else{/* Add here some code to deal with this error */}


  297. }
復制代碼


分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 久草中文在线 | 日本在线观看视频 | 成年人在线观看 | 久久久新视频 | 午夜精品一区二区三区在线视频 | 午夜无码国产理论在线 | 日韩在线中文字幕 | 超碰在线人 | 99九色 | 日韩欧美一区在线 | 国产在线播放一区二区三区 | 亚洲成人福利 | 欧美午夜精品久久久久免费视 | 国产一区二区中文字幕 | 国产资源一区二区三区 | 欧美伊人久久久久久久久影院 | 国产精品国产成人国产三级 | 亚洲视频一区二区三区 | 国产综合精品一区二区三区 | 午夜私人影院在线观看 | 日韩免费一区二区 | 男女深夜网站 | 最新中文字幕久久 | 超碰伊人久久 | 日韩午夜在线播放 | 成人高清在线视频 | 91免费在线视频 | 久久精品屋 | 欧美日韩在线精品 | 一区视频在线播放 | 中文字幕国产精品 | 欧美亚洲综合久久 | 亚洲高清久久 | 日韩最新网址 | 在线免费av观看 | 国产精品揄拍一区二区 | 国产激情一区二区三区 | 亚洲区一区二区 | 日韩中文字幕 | 国产一二三区在线 | 欧美精品久久久久久久久久 |