本帖最后由 XinLiYF 于 2018-4-14 18:02 编辑
( A& c4 O9 R& N5 x Z# M# T2 o# s6 p/ b) n
ARM CMSIS Driver 学习 之 USART+ i4 ]. t. m4 |6 v* o
) U6 g+ g" A( `$ _- p' p. N* n/ H: p. j
最近把 MDK 升级到了 V5.25 ,发现 Managing Run-Time Environment 中已经有好多好多的库。相比之前已经好了太多太多,从底层驱动,到上层协议栈,常用的有不常用的也有。发现 ARM 对这套系统的更新速度加快了一些,觉得有必要学习一下。从驱动开始学起,先学 USART API 详细介绍见 CMSIS Driver USART API
, Q- u J% n2 H5 ^' Q1 X. X8 i' r2 T: g! ~. O: g& u" I
USART 把收到的数据再发出去程序
. T- e2 V/ L7 ]5 y- /**$ k _$ f8 E3 W0 N4 q
- ******************************************************************************
0 w U( G5 Q$ T - * @file main.c
4 d" s7 o. I9 X8 b$ @; ]3 A( I! y% Q - * @author XinLi( G( x! M7 a) t+ }& p& r9 h
- * @version v1.0/ d1 P" v, k0 D6 d) J4 W
- * @date 20-March-2018& Q5 t# H" h z2 F' C( S c
- * @brief Main program body.3 g* _$ R) V* g+ J
- ******************************************************************************
9 n8 v, k) r1 o4 I* Y - * @attention+ o1 Y6 p r4 n; Y7 T& Q
- *' I; k: d* G9 g# m' N
- * <h2><center>Copyright © 2018 XinLi</center></h2>
) m- c/ b. p& ?& q; N - *
) D# i$ y1 Z+ N, h: a( ] - * This program is free software: you can redistribute it and/or modify
8 W7 p( X" N; n; r7 P+ ^! e - * it under the terms of the GNU General Public License as published by4 q: y1 ^0 \5 }& }
- * the Free Software Foundation, either version 3 of the License, or" R# r% X/ Q+ j7 x& k
- * (at your option) any later version.' e$ W* `) Y. I) ~) J& M" r
- *
& P" o- q e: t+ n7 ^ - * This program is distributed in the hope that it will be useful,
. ]$ v5 U. @8 Z) Q% f5 {! K - * but WITHOUT ANY WARRANTY; without even the implied warranty of! f5 A* z; Z6 j; E
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the$ ^& p& H L2 M. P7 b8 A8 B
- * GNU General Public License for more details.
4 M& ~" T# s' z5 e: O0 t2 L - *
+ d& S0 F6 G$ t" X* G - * You should have received a copy of the GNU General Public License
( |9 U- b% d) G" q" h - * along with this program. If not, see <http://www.gnu.org/licenses/>.
; Y. L% t# x+ Z5 ]& [" ?; N - *
! O* |- v6 \) \ U - ******************************************************************************# W* j+ B2 p; |; |. C m
- */) f% e9 B3 Q% P1 ~
- : g* V& O. O' {2 X# M! s5 }
- /* Header includes -----------------------------------------------------------*/
; N4 ?: w/ n6 x ^ - #include "stm32f4xx.h"% f) K1 _! R& p: O# d) Y$ V
- #include "Driver_USART.h"
5 o2 y( @- y3 S" a. P& v7 X) x - #include <string.h>
7 O2 J7 l, n1 R' u. D
" p4 O& u/ @9 p- /* Macro definitions ---------------------------------------------------------*/
' b/ U' v7 _# m k7 ~7 T( q - /* Type definitions ----------------------------------------------------------*/
9 Z2 V* r1 @ y* W' G9 Y* y - /* Variable declarations -----------------------------------------------------*/
' f1 r: I( E) i# x - extern ARM_DRIVER_USART Driver_USART1;
- B1 ^- [* P$ l - % b: M3 ~: A2 i2 \
- /* Variable definitions ------------------------------------------------------*/- n2 ~' E# h& F: G% l7 [
- static uint8_t rxBuffer[1024] = {0};
' e" r3 p9 a* ?6 G - static uint8_t txBuffer[1024] = {0};
; [; _2 Y% J3 m# Y; X
4 I# G3 y3 U/ K% M7 z- /* Function declarations -----------------------------------------------------*/
/ f* c9 n/ E) j9 ^2 z+ X6 Z - static void USART1_Callback(uint32_t event);
B' Q; y, A+ J/ u+ \ - static void SystemClock_Config(void);' ]; I: N1 G% e9 x- S* \" B2 Y
1 v$ C3 J: z5 o' H- /* Function definitions ------------------------------------------------------*/* s: g+ L3 i0 R/ I6 k4 U
- 3 V Y/ C S8 M$ i7 s6 P
- /**
- L* a7 Z8 e# G1 }8 w5 I6 n7 Z8 Z - * @brief Main program.% p( y: |1 g& s! G4 t. d
- * @param None.+ \9 v5 w, L8 W# c) V/ } z) E3 p9 v
- * @return None.+ s4 `' W F9 I
- */1 D8 E6 A! Z8 Q% D, ?* _
- int main(void)
( @8 |; @. E9 E. L( M - {4 d, v' }& j* a, s6 [
- /* STM32F4xx HAL library initialization:8 f& ~, l$ F v+ A- h' O% c
- - Configure the Flash prefetch, instruction and Data caches& g9 b' i0 X. Y* S# |
- - Configure the Systick to generate an interrupt each 1 msec
1 a, ^ T5 Y; {. y- n; U1 L - - Set NVIC Group Priority to 4
3 @& |& x! P7 ~4 m( s% W% i - - Global MSP (MCU Support Package) initialization; y. a/ @! L( W# d+ P4 C8 E! Q
- */
5 P; x% o! h7 j+ ~8 M2 O2 V* G2 T - HAL_Init();9 G$ H& C. H! |
-
1 N" @. g( Y! ~' T( ^% _" L" t: Y, n - /* Configure the system clock to 168 MHz */
+ S. e* `, i9 B) d, C; D - SystemClock_Config();3 ^1 P2 Z; z. A5 J5 L) ^
- 6 C+ f: }7 u5 q& z
- Driver_USART1.Initialize(USART1_Callback);
2 a4 \. S# S1 F8 w# q0 @( m - Driver_USART1.PowerControl(ARM_POWER_FULL);
. S" h5 e) ?5 v, C8 u( b* Q" X' v: J - Driver_USART1.Control(ARM_USART_MODE_ASYNCHRONOUS |
- U& u2 K4 w; w: y: y4 @9 w6 }3 z- M - ARM_USART_DATA_BITS_8 |- o/ J$ g/ V* T' O
- ARM_USART_PARITY_NONE |" M: p" D, G+ t0 A. D
- ARM_USART_STOP_BITS_1 |" ] A8 Z# F0 f& D( o
- ARM_USART_FLOW_CONTROL_NONE, 115200);
' t/ O& j/ x0 s% l! A - Driver_USART1.Control(ARM_USART_CONTROL_TX, 1);
! _; p. G0 D* |2 k' W9 \ - Driver_USART1.Control(ARM_USART_CONTROL_RX, 1);+ `: v) g3 @0 A8 t9 R( g' |
-
) X {" ^# |- ? - Driver_USART1.Receive(rxBuffer, sizeof(rxBuffer));3 \( i; c. G( a' g4 P g
-
2 m. j j4 ]+ }6 g) U - for(;;)) I, B8 \% y( o* D. s! \ D# @9 m+ z* ?
- {
4 _/ @& V" I& P0 c9 A -
) A; ^- v0 Q. i6 Y) a$ z - }
! X5 ]' {& e" J7 O8 p - }. W# _. R5 z- N' T
9 f7 Q8 v9 Z e* e" [- /**
% { w% u$ p# x- x* J' \4 _! G - * @brief USART1 callback function.
& T; O/ v) f9 S3 z" R - * @param event: USART events notification mask.4 H4 o2 d- G( L" O' v( f* h
- * @return None.
3 p* ^) m7 ~, Q2 l% n2 f - */; L& A& V+ l; ?. h" W) f
- static void USART1_Callback(uint32_t event)
% b" H/ [( \+ F# P$ l. t - {
6 A* y0 p/ R( ^, |/ e+ C - if(event & ARM_USART_EVENT_RX_TIMEOUT)
# `. @7 S' R: A& m" g7 @ - {
2 ]( S0 V9 C/ [# Z* }8 }- O - Driver_USART1.Control(ARM_USART_ABORT_RECEIVE, 1);3 C+ l8 d- [1 N/ q: s5 T
- * g& _6 v- S3 ], Q- h$ h7 L
- uint32_t length = Driver_USART1.GetRxCount();9 S/ m9 p* A5 q9 X
-
) C2 i* Y9 h L) X# p - memcpy(txBuffer, rxBuffer, length);
' _: K5 P( g' n* e/ F; }5 d, `3 o! g4 P -
9 Q$ o2 o9 n& U& Y0 Q d! i- S - Driver_USART1.Send(txBuffer, length);5 K* w, H( ^3 l' o! g% o
- Driver_USART1.Receive(rxBuffer, sizeof(rxBuffer));
) j* w( D1 ]) ]6 s5 h4 o7 c: X - }( i G3 ~9 n2 H: k& x/ `- o
- }! J J$ m* W& a7 P! o1 T0 R$ Q
- + }/ A( K$ [; f
- /**
* x5 H6 s+ i1 |3 P: X7 A7 g, J - * @brief System Clock Configuration
$ Y+ q( L+ d o - * The system Clock is configured as follow :
! \" k: E. }, `- w - * System Clock source = PLL (HSE)9 P6 m8 Q* `5 V& Q. B# T1 J
- * SYSCLK(Hz) = 168000000/ F; Y$ T! ~. i# C
- * HCLK(Hz) = 1680000006 |0 v" ?1 Z% X1 m% p9 g1 G
- * AHB Prescaler = 1' s8 f4 E3 q& T/ |
- * APB1 Prescaler = 4( U) B/ [& }/ [+ U* U; h( m
- * APB2 Prescaler = 2
_% G1 g& s1 l. I& I' c- j+ W - * HSE Frequency(Hz) = 8000000/ O* a7 g' Y1 [7 d' w
- * PLL_M = 8
- D6 G4 [+ D& X - * PLL_N = 336 k. ~# T/ h( G) a$ F. ~# G
- * PLL_P = 2, Y, q) f6 @" V6 D
- * PLL_Q = 7
1 }! q2 E5 d4 |* l% z - * VDD(V) = 3.3. P. |) Y( Q0 _
- * Main regulator output voltage = Scale1 mode3 m6 g* b1 j- C3 ^$ C# d$ A
- * Flash Latency(WS) = 5
1 e0 `, D5 t9 V: F ]8 o - * @param None
6 I; z# [- J. u2 Z$ X5 M* q - * @retval None+ p. X' O2 j' K% f, C/ v m1 u
- */ Z X2 J* V* m k
- static void SystemClock_Config(void)* |; G9 t% l+ Z" u6 Q5 o5 I; G# n: G
- {
/ Z4 w6 s A8 z: E' @; s - RCC_ClkInitTypeDef RCC_ClkInitStruct;6 x/ R( X$ L* h2 z- i, s
- RCC_OscInitTypeDef RCC_OscInitStruct;- c u5 v& K5 C6 A# T
- v* t4 ]2 y y) t
- /* Enable Power Control clock */
- o4 |! p& ^. R - __HAL_RCC_PWR_CLK_ENABLE(); @) [2 q9 d6 Z# } E- o
- # s8 S4 w# P# x, n
- /* The voltage scaling allows optimizing the power consumption when the device is
( ]# B. d+ J1 e - clocked below the maximum system frequency, to update the voltage scaling value . p4 Q0 ~7 t1 f
- regarding system frequency refer to product datasheet. */3 k' O2 @" T2 K- v9 a4 T
- __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);
& }, G% p6 V! [" ?0 {2 G3 b
# z6 Y2 ` a ~, X0 F q- /* Enable HSE Oscillator and activate PLL with HSE as source */5 U. E$ D2 A. R7 r7 i! o
- RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
: Y: y$ O/ j/ Z+ c1 f9 q - RCC_OscInitStruct.HSEState = RCC_HSE_ON;
- u9 c% e, ]1 U/ R8 ~! C X - RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
) m9 Z$ O! d2 w0 H# z - RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
% s% }1 r3 h( d) q - RCC_OscInitStruct.PLL.PLLM = 8;1 H' W; x) v4 P; g8 Q+ e
- RCC_OscInitStruct.PLL.PLLN = 336;
6 }; _! e& P# N+ G9 q' i - RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
" Z- F0 z) ~0 O- D - RCC_OscInitStruct.PLL.PLLQ = 7;
# i3 o$ y4 v# w( u) S# e - HAL_RCC_OscConfig(&RCC_OscInitStruct);
3 u! P- _7 x/ B- H) v# j$ b -
+ d8 M5 e+ d- N3 f - /* Select PLL as system clock source and configure the HCLK, PCLK1 and PCLK2 5 Y$ n U8 l/ g' ^/ N' Q8 t
- clocks dividers */ _* v' F4 t1 q8 x9 H" l, `
- RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2);; K+ W# G% |' j! G* P( w& O: r
- RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;$ O; G3 v: K1 R% o9 y- \4 F& g
- RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
2 q; y0 }8 p- }% p l' R z - RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
) S$ P4 o/ d' ?# [ - RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
) \% b. Q, w2 y/ R/ o - HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5);
0 ?& f. ~4 A0 O) f; m% m; n - " |0 F) Q% i) U1 r
- /* STM32F405x/407x/415x/417x Revision Z devices: prefetch is supported */: |) n; g' T$ s/ k, [3 Y3 Q
- if (HAL_GetREVID() == 0x1001)1 _7 e2 T6 x: X# H
- {- s/ p: m5 b$ [+ {
- /* Enable the Flash prefetch */5 V$ f# j* u" ~! A9 [; i/ g
- __HAL_FLASH_PREFETCH_BUFFER_ENABLE();4 z4 l B5 s7 [) l& b3 {5 c! ]8 O
- }2 a( s* M+ r! t8 H8 R
- }
& Y! H; w% v- \0 d8 j6 _; V
复制代码 - O/ ?9 }9 j2 @1 a
归档链接" r* }; J5 R/ z% y
ARM CMSIS Driver 学习 之 SPI) l, s) y1 Q4 O( j- a1 t+ ~ p, v/ Z
5 N% T/ @6 ]( h4 p# H
|
这个出来有几年了,只不过之前没有现在全,现在可以用到项目里了,底层驱动 ARM 已经实现了跨平台,之后产品换 MCU 也比较方便。
这个趋势很好啊,避免碎片化,抽空看看
是啊,MCU 开发就是缺少这样大一统的框架啊
https://www.stmcu.org.cn/module/forum/thread-615497-1-1.html
{0 C* O4 f+ N1 F& ~+ E6 Z1 d8 O
// UART1_Comm_Init(115200);& h+ c% Q1 t0 b9 y8 Q# A
UART2_Comm_Init(9600);0 Z* n5 w H9 L
Driver_USART2.Send("http://www.cmsoft.cn", sizeof("http://www.cmsoft.cn"));
while(1);- n% |2 g/ o7 y& H, N/ W, _2 i
}. S% t6 r; [! C8 b5 X
程序运行到Driver_USART2.Send就卡在这一行 无法运行到while那里 同时发送的内容也没发出去' r/ F/ A: f& O$ D' T3 G
不知道咋回事????
{- s ^1 l+ v0 w$ o0 e: `" k- h
if(event & ARM_USART_EVENT_RX_TIMEOUT)3 q7 {8 x$ `6 T3 T P6 s
{
Driver_USART2.Control(ARM_USART_ABORT_RECEIVE, 1);
2 I' W7 j' [, Z2 t2 S" U4 @- D# ?
uint32_t length = Driver_USART2.GetRxCount();: S3 C2 \& X3 S L) v E
( A# ^( `* ]) N# l6 w/ o0 o
// memcpy(txBuffer, rxBuffer, length);
//
// Driver_USART1.Send(txBuffer, length);2 \( E2 q, @, }* I* g6 y$ U
Driver_USART2.Receive(USART2_RxBfr, sizeof(USART2_RxBfr));
} ' M6 k4 l( \* ~( A5 B
else if(event & ARM_USART_EVENT_SEND_COMPLETE) [( z5 {& N: I3 w
{5 G6 g5 @* N9 ^
__NOP();
}
else if(event & ARM_USART_EVENT_RECEIVE_COMPLETE)7 w; H/ o8 e7 R1 |5 F
{$ a6 Z' ^# F3 q
__NOP();
}) b6 `$ e w" H; e
}) F$ Y+ Z$ ?3 L" ^8 `9 [& Z
从电脑端的串口工具发送内容 也没产生串口回调事件 不知道咋回事?????
{//usb) \0 J3 L6 k8 O" F
Driver_USART2.Initialize(USART2_Callback);2 o8 D. b3 [" [ q# b- r
Driver_USART2.PowerControl(ARM_POWER_FULL);, F+ W" P" x/ `2 v% @
Driver_USART2.Control(ARM_USART_MODE_ASYNCHRONOUS |* m" P8 P5 `3 E" i. |
ARM_USART_DATA_BITS_8 |
ARM_USART_PARITY_NONE |
ARM_USART_STOP_BITS_1 |
ARM_USART_FLOW_CONTROL_NONE, Brate
);% ?& a+ M8 M4 r' r; c
/* Enable Receiver and Transmitter lines */
Driver_USART2.Control(ARM_USART_CONTROL_TX, 1);
Driver_USART2.Control(ARM_USART_CONTROL_RX, 1);
/* Begin to receive */7 ]4 \1 J* f: t$ Z6 e5 f- n
Driver_USART2.Receive(USART2_RxBfr, sizeof(USART2_RxBfr));1 G4 f! w* Q+ b4 M
}
这是串口初始化