
/******************************************************************************( N( d2 x2 c$ ?8 T1 Y( ~9 P( W * 本文件实现串口发送功能(通过重构putchar函数,调用printf;或者USART_SendData()3 D/ z1 ^. F" _7 n3 Z$ z- @) S * 这里是一个用串口实现大量数据传输的例子,使用了DMA模块进行内存到USART的传输 * 每当USART的发送缓冲区空时,USART模块产生一个DMA事件,- M9 p9 ~ h4 M5 R7 o! l) I+ @ l4 T * 此时DMA模块响应该事件,自动从预先定义好的发送缓冲区中拿出下一个字节送给USART" w; O& j3 o- N! w * 整个过程无需用户程序干预,用户只需启动DMA传输传输即可 * 在仿真器调试时,可以在数据传输过程中暂停运行,此时DMA模块并没有停止 * 串口依然发送,表明DMA传输是一个独立的过程。 * 同时开启接收中断,在串口中断中将数据存入缓冲区,在main主循环中处理- x9 P" z9 p$ |4 U0 v * 作者:jjldc(九九)- `: H6 f. p& d% L# g * 代码硬件基于万利199元的EK-STM32F开发板,CPU=STM32F103VBT6 *******************************************************************************/# c9 V# q4 P( p0 _5 e. h6 y" \ /* Includes ------------------------------------------------------------------*/# `1 R4 q1 |7 e' R) k* D% P' ~ #include "stm32f10x_lib.h"& T& j! Q- R; x8 I' \0 S2 p! R #include "stdio.h" /* Private typedef -----------------------------------------------------------*/) J7 b% C$ |9 A9 j4 k. P. i9 p /* Private define ------------------------------------------------------------*/ #define USART1_DR_Base 0x40013804+ o9 m4 X8 A, [; g0 m0 `0 e" k /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/. [) i) J4 N# a( e6 |; u2 w #define SENDBUFF_SIZE 10240 vu8 SendBuff[SENDBUFF_SIZE];6 [" K" S7 ?! l8 Z0 H; s% D. Z vu8 RecvBuff[10]; vu8 recv_ptr;/ o5 G% v' f1 M! Q /* Private function prototypes -----------------------------------------------*/( ` N# h/ P y: C void RCC_Configuration(void);& F' _$ E3 ]1 ]7 p void GPIO_Configuration(void); void NVIC_Configuration(void); void DMA_Configuration(void); void USART1_Configuration(void); int fputc(int ch, FILE *f); void Delay(void); /* Private functions ---------------------------------------------------------*/" z- f: [9 L& }9 m8 v0 s) Q8 | /*******************************************************************************( O! |5 }+ v& A * Function Name : main * Description : Main program. \' c p5 @, y, G$ t' V * Input : None; Z5 T" Z8 e2 Q0 ]% L; `) a: Q * Output : None * Return : None *******************************************************************************/( G M' @& K+ T% R3 M# C7 b$ q int main(void)9 ]! l. K% |: ]8 K { u16 i;2 I' I6 U3 {3 o; {7 U# l; ~ #ifdef DEBUG" M# L6 h* N5 i+ M/ V' D; g debug();' V* e. G% M: x% M; U& | #endif recv_ptr = 0;2 Z- }" O- _# _6 ^9 o: j, I RCC_Configuration();( ^3 h- k# N: S* d GPIO_Configuration();. }) S1 L4 ?0 T6 [1 h NVIC_Configuration(); DMA_Configuration(); USART1_Configuration(); printf("\r\nSystem Start...\r\n"); printf("Initialling SendBuff... \r\n"); for(i=0;i<SENDBUFF_SIZE;i++) {. V" |0 L# m' O- l SendBuff = i&0xff; }/ b% F. B1 C8 ?, [# z printf("Initial success!\r\nWaiting for transmission...\r\n"); //发送去数据已经准备好,按下按键即开始传输* H- y, N; S" }3 K( T( e while(GPIO_ReadInputDataBit(GPIOD, GPIO_Pin_3));9 k5 o( |. x: t; H: w% @ printf("Start DMA transmission!\r\n");; |4 L- }1 c% e6 M+ K+ y- P //这里是开始DMA传输前的一些准备工作,将USART1模块设置成DMA方式工作 USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE); //开始一次DMA传输! DMA_Cmd(DMA1_Channel4, ENABLE);7 w+ G$ a# y- s //等待DMA传输完成,此时我们来做另外一些事,点灯 //实际应用中,传输数据期间,可以执行另外的任务 while(DMA_GetFlagStatus(DMA1_FLAG_TC4) == RESET)& l5 W, @5 t4 J- c7 x C3 H5 M {2 n0 t) D% P% n" O' k( j LED_1_REV; //LED翻转 Delay(); //浪费时间* {) l$ I z o- @ }) X% v9 h1 R# {& L3 T //DMA传输结束后,自动关闭了DMA通道,而无需手动关闭 //下面的语句被注释; T7 R l! x/ x$ e; R# R! _ //DMA_Cmd(DMA1_Channel4, DISABLE); printf("\r\nDMA transmission successful!\r\n"); /* Infinite loop */ while (1), V. S; r' f; L* _8 O# r { } }; l0 x$ T1 r6 ~* I9 j4 p. j* V /*******************************************************************************9 @/ H- ~8 @+ a6 `. b$ w * Function Name : 重定义系统putchar函数int fputc(int ch, FILE *f), m! n A- S- g6 e * Description : 串口发一个字节 * Input : int ch, FILE *f) ~7 f, M0 ^1 t4 q' q * Output :0 }+ x! z; G3 X; A& q3 i1 g: k * Return : int ch * 这个是使用printf的关键 *******************************************************************************/ int fputc(int ch, FILE *f) {- b \- B# \5 F) z) x6 k //USART_SendData(USART1, (u8) ch);9 Q# H1 d. `8 e USART1->DR = (u8) ch;- W7 Q2 P. ~) t! {3 {4 |; C, L /* Loop until the end of transmission */( S0 u1 ^+ [# g! a6 w while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET) { } return ch;7 \6 _( Q3 B# j0 K } /******************************************************************************* * Function Name : Delay * Description : 延时函数 * Input : None- s! }( A: P7 ]2 d i- O! B5 s * Output : None * Return : None2 u% K0 c% ^7 }& v: [ *******************************************************************************/ void Delay(void) {# R7 R9 d6 b% g u32 i; for(i=0;i<0xF0000;i++);& S9 j/ y7 t# q" Y7 w) @* Y return; } /******************************************************************************* * Function Name : RCC_Configuration * Description : 系统时钟设置 * Input : None, ? T; U0 B3 o * Output : None * Return : None *******************************************************************************/- X* R/ Y2 K2 A( Z void RCC_Configuration(void) { ErrorStatus HSEStartUpStatus; //使能外部晶振 RCC_HSEConfig(RCC_HSE_ON);2 n- ~- v2 D4 Y# p! B //等待外部晶振稳定2 u( S& @/ x* B4 I/ m6 w HSEStartUpStatus = RCC_WaitForHSEStartUp(); //如果外部晶振启动成功,则进行下一步操作 if(HSEStartUpStatus==SUCCESS) Y. V; [, v0 S! A M" F, }' k- V# S" L {( @+ b# t( u/ ^6 O5 t) C7 j) t //设置HCLK(AHB时钟)=SYSCLK, J O6 g' b3 l6 u& R RCC_HCLKConfig(RCC_SYSCLK_Div1);6 L4 o' V8 u. q1 A" p! d2 O/ Q //PCLK1(APB1) = HCLK/2 j" b1 @2 J2 w RCC_PCLK1Config(RCC_HCLK_Div2);) Z$ k/ ?% `6 L& w //PCLK2(APB2) = HCLK! \6 c) y# _; Q# F3 N/ f( I1 p0 R RCC_PCLK2Config(RCC_HCLK_Div1);: e( x$ Q4 J; F& k! @ //FLASH时序控制 //推荐值:SYSCLK = 0~24MHz Latency=0( i' R9 }, c7 z2 I // SYSCLK = 24~48MHz Latency=1 // SYSCLK = 48~72MHz Latency=2 FLASH_SetLatency(FLASH_Latency_2); //开启FLASH预取指功能# l: `; b5 P' |9 c4 W5 \ FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //PLL设置 SYSCLK/1 * 9 = 8*1*9 = 72MHz RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);$ }) O7 N5 m* S: L4 I //启动PLL4 l% m& h! g3 w" D" ?' f RCC_PLLCmd(ENABLE); //等待PLL稳定 while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //系统时钟SYSCLK来自PLL输出$ v1 b' H( R' W- L$ s" R: F5 u* Z- j RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //切换时钟后等待系统时钟稳定: q+ E- X9 r+ F/ P while(RCC_GetSYSCLKSource()!=0x08); /** N1 s4 e7 ^4 D( u# `% S0 J //设置系统SYSCLK时钟为HSE输入 RCC_SYSCLKConfig(RCC_SYSCLKSource_HSE); //等待时钟切换成功& ~9 q' Z. N |" j) P v while(RCC_GetSYSCLKSource() != 0x04); */$ g7 d7 s- k5 o& V e }! W8 t1 d0 _$ w/ P0 L+ F2 h //下面是给各模块开启时钟+ D" L4 ~+ W& c9 u: [2 o# i4 H //启动GPIO6 J6 e8 N2 ~2 L9 Z c$ d, L' l RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | \ RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD,\ ENABLE); //启动AFIO3 Z4 t5 C6 S$ u( ~( m, ?0 t/ ` RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);' ^- J5 i* }* Z4 n/ }% { //启动USART1 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);3 `% q& A2 ^, G/ x9 b w1 V8 o; b //启动DMA时钟7 {( u5 R: _( w1 E5 h$ Y q. Q RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); }. b/ D$ B* h8 {- i o1 |2 l0 { /******************************************************************************* * Function Name : GPIO_Configuration * Description : GPIO设置/ y0 x! [* K# j* J% c8 o: R5 m * Input : None( `9 s7 ]* q: x7 L2 u/ V * Output : None, j. Y" ], _1 Y* V" u0 q; }( L5 u( u! N * Return : None *******************************************************************************/- v' W2 D7 C3 S0 @# D void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure;; k; |8 @: q; v [2 Q3 Y5 W: o //PC口4567脚设置GPIO输出,推挽 2M: i* |8 H- P8 f3 P1 O( D GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_Init(GPIOC, &GPIO_InitStructure);9 K9 i( b$ O, H //KEY2 KEY3 JOYKEY //位于PD口的3 4 11-15脚,使能设置为输入 P$ I c$ C0 W GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_11 | GPIO_Pin_12 |\5 s9 ~- E0 p$ Z) A7 z" u! f GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOD, &GPIO_InitStructure);: {+ l/ l/ h) A1 C8 `4 J //USART1_TX h Q+ l3 ^9 r" P3 k2 r GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;$ W1 g( f+ h) }# v2 O, \ GPIO_Init(GPIOA, &GPIO_InitStructure);8 V+ |2 r/ z) s O5 W //USART1_RX2 C- k3 T6 R9 L( k; r! Z8 N, S GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;2 L% V- B' R8 t! }) z" f GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;5 a7 r% k% e+ o5 x( p3 K GPIO_Init(GPIOA, &GPIO_InitStructure); } /******************************************************************************* * Function Name : NVIC_Configuration' L( q& ^ ^5 N6 P1 k5 D * Description : NVIC设置 * Input : None * Output : None) E& P! Q. B( q. }+ ] * Return : None *******************************************************************************/$ A0 X- n. U8 \: S void NVIC_Configuration(void). A- k2 Y: ]& g+ f9 B { NVIC_InitTypeDef NVIC_InitStructure;% O2 v- l$ H; f4 H8 l9 J #ifdef VECT_TAB_RAM# p7 V* i: R3 b i- O+ u z // Set the Vector Table base location at 0x200000007 n) _" f8 {# L* I1 n) W! c, m NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);4 W3 U( t: M0 \- k" Q" j0 [+ j #else /* VECT_TAB_FLASH */6 @# [9 W1 X1 ?. |' n, ^- v' {1 q( D // Set the Vector Table base location at 0x080000007 {4 |; y; K1 I8 z NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0); #endif' h. ]1 M+ g( G1 |+ m& j //设置NVIC优先级分组为Group2:0-3抢占式优先级,0-3的响应式优先级+ D- }6 Q$ @/ I+ H NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //串口接收中断打开 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;9 y9 r: b' G5 {: y8 b7 L! q. S NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;& H3 S6 ]! R1 r' g7 h2 v/ R NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } /******************************************************************************* * Function Name : USART1_Configuration * Description : NUSART1设置 * Input : None+ t" @: n) Q+ u" b: o5 |" D * Output : None * Return : None *******************************************************************************/ void USART1_Configuration(void); E- M' I8 {4 \ {! G5 t% a! {; _/ z- z' m( p2 |( H0 } USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = 9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b;$ `" [0 ]8 m, d USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;/ }1 p- D& ^3 R! T USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; USART_Init(USART1, &USART_InitStructure);6 C# W$ N7 h- a- I USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); USART_Cmd(USART1, ENABLE); } void DMA_Configuration(void), l# p0 X" S [1 b8 h { DMA_InitTypeDef DMA_InitStructure;: D, R G' B& E3 \* p //DMA设置:+ H2 `! T+ x4 E* O //设置DMA源:内存地址&串口数据寄存器地址& l/ M* ?5 Q7 t5 b7 n0 @9 M //方向:内存-->外设; U2 {1 Q7 Z3 Y$ I+ Q //每次传输位:8bit8 I* Y& `( J1 D Z( n* H //传输大小DMA_BufferSize=SENDBUFF_SIZE5 L4 `1 n/ F% W& K3 y' U" S% { //地址自增模式:外设地址不增,内存地址自增1! L# C/ p6 G! T F' r //DMA模式:一次传输,非循环 //优先级:中 DMA_DeInit(DMA1_Channel4); DMA_InitStructure.DMA_PeripheralBaseAddr = USART1_DR_Base; DMA_InitStructure.DMA_MemoryBaseAddr = (u32)SendBuff; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;7 Q O& p; K! T DMA_InitStructure.DMA_BufferSize = SENDBUFF_SIZE; DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;+ Z3 G% I3 y# K% i DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;$ y: S7 D M! U/ T+ U) X% W4 i( g. C DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;; m- `3 l$ C/ e! J q! ]+ R- Q DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel4, &DMA_InitStructure);! h s4 |' ~% _' m# y0 U } |