找回密码
 注册
关于网站域名变更的通知
查看: 213|回复: 3
打印 上一主题 下一主题

STM32的SYSTICK详解

[复制链接]

该用户从未签到

跳转到指定楼层
1#
发表于 2022-7-29 10:06 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

EDA365欢迎您登录!

您需要 登录 才可以下载或查看,没有帐号?注册

x
什么是SYStiCK:0 f; t$ h# q6 b4 f8 f6 m
这是一个24位的系统节拍定时器system tick timer,SysTick,具有自动重载和溢出中断功能,所有基于Cortex_M3处理器的微控制器都可以由这个定时器获得一定的时间间隔。

( T* ~4 h5 Z  K6 b" `
作用:
1 n8 }7 P9 Q- Z3 }% H- P
在单任务引用程序中,因为其架构就决定了它执行任务的串行性,这就引出一个问题:当某个任务出现问题时,就会牵连到后续的任务,进而导致整个系统崩溃。要解决这个问题,可以使用实时操作系统(RTOS).
$ w, I3 D6 N2 p9 D" a
因为RTOS以并行的架构处理任务,单一任务的崩溃并不会牵连到整个系统。这样用户出于可靠性的考虑可能就会基于RTOS来设计自己的应用程序。这样SYSTICK存在的意义就是提供必要的时钟节拍,为RTOS的任务调度提供一个有节奏的“心跳”。

/ K0 t! _( g/ _7 H% o
微控制器的定时器资源一般比较丰富,比如STM32存在8个定时器,为啥还要再提供一个SYSTICK?原因就是所有基于ARM Cortex_M3内核的控制器都带有SysTick定时器,这样就方便了程序在不同的器件之间的移植。而使用RTOS的第一项工作往往就是将其移植到开发人员的硬件平台上,由于SYSTICK的存在无疑降低了移植的难度。
. Q. q( o. f1 k3 O! G+ E

3 |" l  j3 R" N  O7 p: I/ X
  SysTick定时器除了能服务于操作系统之外,还能用于其它目的:如作为一个闹铃,用于测量时间等。

: D, V' E$ y% E3 H; R7 @
要注意的是,当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作。
4 t! P* A* W& V4 j
% J+ o% T, B+ a) L: b7 N- J" n+ q
时钟的选择:

. g8 I" v9 K1 X  Y4 z+ v3 S
用户可以在位于Cortex_M3处理器系统控制单元中的系统节拍定时器控制和状态寄存器(SysTick control and status register ,SCSR)选择systick 时钟源。如将SCSR中的CLKSOURCE位置位,SysTick会在CPU频率下运行;而将CLKSOUCE位清除则SysTick会以CPU主频的1/8频率运行。
7 f6 S% @% o. N- u
3.5版本的库函数与以往的有所区别

( l, B6 b8 ~8 a3 R' B: }; z& A6 d
不存在stm32f10x_systick.c文件,故原来的一些函数也不存在,比如SysTick_SetReload(u32 reload);SysTick_ITConfig(FunctionalState NewState);等
! K+ ^* O1 s  ~, C+ u0 p1 U

$ {& S3 [! `% `6 T8 f1 z
在3.5版本的库函数中与systick相关的函数只有两个

* \$ x! Y: N* E0 P$ F2 ]
第一个,SysTick_Config(uint32_t ticks),在core_cm3.h头文件中进行定义的。
9 U# x" b6 ~2 G6 {( G4 @6 |
第二个,void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),在misc.c文件中定义的。

& G* f( l1 B% m, F9 v1 ~2 h3 }: H8 _8 d* d" \8 M
SysTick_Config(uint32_t ticks),在core_cm3.h
6 N; @' A* t% ]! f, Z+ o8 S3 Z
主要的作用:
9 K( D/ l, Z, E! A7 {9 i6 B
1、初始化systick
7 v3 S6 z& t  y$ o. n: d
2、打开systick

2 t1 _0 A/ f. A/ f% E1 `
3、打开systick的中断并设置优先级

3 l% F  c- K$ W: t/ n+ ?
4、返回一个0代表成功或1代表失败
  Y$ A2 N" [2 |' y0 n. x  k, [
注意:

$ z  F" m, w2 t
Uint32_t ticks  即为重装值,

% W7 L  _8 B) S1 N2 c
这个函数默认使用的时钟源是AHB,即不分频。
+ `) ^4 k2 b3 ~( ~( k! D
要想分频,调用void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),
. i! J( g- [6 X' q, c, o' p
但是要注意函数调用的次序,先SysTick_Config(uint32_t ticks),

- j- ]' K  q2 ?
后SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)

, g) G4 v  n' c" H" H. `, V  U; I0 o- O, w3 }/ d7 h" x
% j" q; \9 M9 S8 ]4 h0 a

6 N+ I4 @" ?: ~- ?& \
2 _: j2 o) T: w& {- a
函数说明:
# f" A: Q% H# S  X

: _3 Q- e+ G; c7 N- [" d; j( c
/**

; ]3 Y& ?: b( P5 t
* @brief  Initialize and start the SysTick counter and its interrupt.

% ]* }, }, }% F  a5 J$ y
*

: Q+ F% ^7 r1 m6 ]# y' X1 x
* @param   ticks   number of ticks between two interrupts
3 P% |- l  X) ~& m7 p" D; o
* @return  1 = faiLED, 0 = successful

- ], k7 @4 z9 b! `
*

+ K! _8 _0 S9 f1 J% X
* Initialise the system tick timer and its interrupt and start the
1 ?; M. c) {* ~. ]$ ~/ S
* system tick timer / counter in free running mode to generate

# K& ^! N! p+ ^9 Y: I' @/ g3 I# t5 ]
* periodical interrupts.

( E4 y2 I! a+ p) ~$ ?/ E5 ~0 ]
*/
  |6 y. F7 |4 G% ]' d8 S1 i, |
static __INLINE uint32_t SysTick_Config(uint32_t ticks)

# L) t2 N' ?( ]# }- S% j
{

# M/ s5 \% L6 h; t: C
  IF (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            

5 j1 E( k( M: a' h. _
  /* Reload value impossible */重装载值必须小于0XFF FFFF,为什么,这是一个24位的递减计数器。
+ F% g& z9 }+ o* ]) Q

2 V6 F, Z) V( Z6 c5 _0 I' _/ W
  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;
+ Y3 `5 U$ @$ K1 ]! T- [
     /* set reload register */设置重装载值,SysTick_LOAD_RELOAD_Msk定义见后面
0 n( w5 B, V5 ^+ z- M. a7 \
  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);

1 t( r8 Y" X5 n- U' q8 f" T+ ^! ~
/* set Priority for Cortex-M0 System Interrupts */
" T2 N7 f; Y* l7 \
  SysTick->VAL   = 0;

* P7 ?% R7 u! z% N( H
  /* Load the SysTick Counter Value */

% L) H$ v& g4 I$ K" |
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |
) [5 p8 Q% B( V
                   SysTick_CTRL_TICKINT_Msk   |

) i- p! ], r8 I- D- G* K4 _. H  ~
                   SysTick_CTRL_ENABLE_Msk;                  
" R( n8 Q9 _' M; w1 ~5 P& e8 V/ P
/* Enable SysTick IRQ and SysTick Timer */

  H+ I) W6 q% a! o
  return (0);
- s4 w5 N* q$ y: J& @' s
  /* Function successful */

- `5 V, c5 ?/ s1 V' X( J- C9 ?
}
8 [- `; j, L4 w
#endif

. H8 O* v) j5 x% C! ]# G
与systick相关的寄存器定义
$ u0 W, z$ o3 J4 c; B

1 W% |  M) B$ m( j$ @$ M
/** @addtogroup CMSIS_CM3_SysTick CMSIS CM3 SysTick

. ]* W" R. V4 w; F' a
  memory mapped structure for SysTick
! {& q/ y3 k* M
  @{

# L- L9 j: v% b( c" I1 N
*/

9 i$ _/ [9 D" m2 ~3 d* Z  I: E
typedef struct
& B$ |& i# I) Y. Y) g0 D
{

3 Z5 Y# q3 h  a& B8 z' u% h; [
  __IO uint32_t CTRL; /*!< Offset: 0x00  SysTick Control and Status Register */

8 O1 Q$ D. J1 K
  __IO uint32_t LOAD; /*!< Offset: 0x04  SysTick Reload Value Register       */
# R* H' Z" j% k( ~
  __IO uint32_t VAL; /*!< Offset: 0x08  SysTick Current Value Register      */
# n- P9 n2 V7 f
  __I  uint32_t CALIB; /*!< Offset: 0x0C  SysTick Calibration Register        */

5 o5 n% v2 x0 ^" V  i2 ?1 V6 d
} SysTick_Type;

! t/ k( i1 f0 H1 ]) }
& Y+ }9 G( G" @0 |5 h& K+ a
与systick寄存器相关的寄存器及位的定义
/ |* S1 B, S5 G1 R9 N  [) Q
. M3 q; `3 {5 k  E7 B* V: h" d# `
/* SysTick Control / Status Register Definitions */控制/状态寄存器

' Z+ P4 @8 B& r3 s9 U#define  SysTick_CTRL_COUNTFLAG_Pos  16      /*!< SysTick CTRL: COUNTFLAG Position */
" ]# O# t5 i/ U
#define SysTick_CTRL_COUNTFLAG_Msk         (1ul << SysTick_CTRL_COUNTFLAG_Pos)         
  b; {# b9 s4 \& z% A
/*!< SysTick CTRL: COUNTFLAG Mask */ 溢出标志位
& @1 |2 q+ m, ]1 i: B( s& j1 h
/ D3 K' V5 w9 Q2 k, Y5 |( [
#define SysTick_CTRL_CLKSOURCE_Pos   2       /*!< SysTick CTRL: CLKSOURCE Position */
$ b1 P2 u6 h9 q: Y$ u
#define SysTick_CTRL_CLKSOURCE_Msk         (1ul << SysTick_CTRL_CLKSOURCE_Pos)  

& v. e5 g1 @3 u/ W* h7 J& i( Z
/*!< SysTick CTRL: CLKSOURCE Mask */时钟源选择位,0=外部时钟;1=内核时钟
' z0 z" D% M% V1 B+ v% B' y

1 i& V# e/ Q1 z1 g- V% e4 ~3 F
#define SysTick_CTRL_TICKINT_Pos      1        /*!< SysTick CTRL: TICKINT Position */
& G7 |' Q6 P; ^8 V& u
#define SysTick_CTRL_TICKINT_Msk           (1ul << SysTick_CTRL_TICKINT_Pos)         

9 Q. w0 s2 G( W6 o0 K/ P+ E
/*!< SysTick CTRL: TICKINT Mask */异常请求位
% s5 t4 f. ~' j( E4 j! @2 X6 p
. _9 X7 z7 v1 g, A# B! s7 e+ w
#define SysTick_CTRL_ENABLE_Pos             0       /*!< SysTick CTRL: ENABLE Position */

2 E# O; I! C9 T3 O- r' n0 r5 F
#define SysTick_CTRL_ENABLE_Msk            (1ul << SysTick_CTRL_ENABLE_Pos)               
9 A& N( B  F9 |+ ?+ \2 g$ G9 @" ]" Y
/*!< SysTick CTRL: ENABLE Mask */使能位
0 v% X, y2 H; a4 s7 V3 d! h

  l" o5 Z+ K& g% u
/* SysTick Reload Register Definitions */
6 V2 \: @0 Z5 \7 U" {* x- X
#define SysTick_LOAD_RELOAD_Pos             0    /*!< SysTick LOAD: RELOAD Position */
* s0 ~9 ^7 V9 r& u: i: u" }0 ^' ?
#define SysTick_LOAD_RELOAD_Msk            (0xFFFFFFul << SysTick_LOAD_RELOAD_Pos)        

+ b' ^) A% J4 Y% B' t* Q
/*!< SysTick LOAD: RELOAD Mask */
7 o8 n( s) d: U, [  L8 `) g: R
5 N/ }0 M0 |7 U) K% B0 h# Z) q
/* SysTick Current Register Definitions */
5 F/ \+ D- k, }$ X& ?( B
#define SysTick_VAL_CURRENT_Pos             0       /*!< SysTick VAL: CURRENT Position */

. N2 [4 a1 T2 O# x) u
#define SysTick_VAL_CURRENT_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        
' g7 u  }; ^) t. }$ q
/*!< SysTick VAL: CURRENT Mask */

: x9 @+ @7 Y4 }: R* B+ ~% `
( v" ?- v" p- l( M) [
/* SysTick Calibration Register Definitions */
: b: b# \$ G3 f
#define SysTick_CALIB_NOREF_Pos            31      /*!< SysTick CALIB: NOREF Position */

, [" H  @5 |) R, X* G- o; d
#define SysTick_CALIB_NOREF_Msk            (1ul << SysTick_CALIB_NOREF_Pos)              

9 R! N! b' O; D0 P) W& a/ J# A
/*!< SysTick CALIB: NOREF Mask */

' Z% A5 H6 b" R! h8 M8 m* c; n4 N! R, o6 n, c4 a
#define SysTick_CALIB_SKEW_Pos             30       /*!< SysTick CALIB: SKEW Position */

$ k- M4 @, r, `! \. n  L
#define SysTick_CALIB_SKEW_Msk             (1ul << SysTick_CALIB_SKEW_Pos)               

9 P+ v- P. k/ r: @5 ?4 ]5 @: ]" F
/*!< SysTick CALIB: SKEW Mask */
6 c1 ]; }! b6 m$ c- g3 @, z
$ O. ]6 p' i6 i* q! A# A# [
#define SysTick_CALIB_TENMS_Pos             0       /*!< SysTick CALIB: TENMS Position */

0 F: `) u* c* C- L
#define SysTick_CALIB_TENMS_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        /*!< SysTick CALIB: TENMS Mask */
# I) o" s/ V" q) l
/*@}*/ /* end of group CMSIS_CM3_SysTick */

$ `+ v. d$ h1 B, v
  r( t+ N' ^6 Q* W3 q+ p
与systick相关的寄存器的说明
$ p5 {, O) ^$ D  Y; A4 U5 k # O* U% u" z2 H+ b  d- \  ]) t* L

/ j. g3 J! f* ?  }! S/ ?; F4 I6 C
& D/ j. j0 e6 Z* g0 V7 h% C , G- L& `2 {7 F

6 N# ?  c% f7 p! Y. l- N
9 `8 T9 c9 X) s
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
4 a3 c" v4 z- a9 G: p: p
作用:
) ^3 }2 B" \: y1 S$ R
选择systick的时钟源,AHB时钟或AHB的8分频

  L4 _6 x4 I) X  I% L. k
默认使用的是AHB时钟,即72MHz
" w( E, _1 {8 S" D4 }+ k
7 a& |/ j6 R$ o! h6 w" P0 w- M( ^! u
函数说明:
6 K& U8 `) S6 x5 U# b: P
/**

% _& V  g! `5 N& i
  * @brief  Configures the SysTick clock source.
, c4 J* Q' x+ t- b  X3 `
  * @param  SysTick_CLKSource: specifies the SysTick clock source.

6 q$ g' ?& C5 M; ~! }4 p& l
  *   This parameter can be one of the following values:
3 d/ d( b$ J; s
  *     @arg SysTick_CLKSource_HCLK_Div8: AHB clock divided by 8 selected as SysTick clock source.

. ^! L: A0 F+ h0 v
  *     @arg SysTick_CLKSource_HCLK: AHB clock selected as SysTick clock source.
+ f0 E: y4 \. s0 O- s+ U  h
  * @retval None

3 n6 C- ^  K( W/ s
  */

5 l" f) ^  q0 P2 w$ P( Y
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)

6 F- S; K: ~9 V5 l0 }
{

+ s( B4 x' K9 s) ~/ X
  /* Check the parameters */

* k$ k' g. p: q
  assert_param(IS_SYSTICK_CLK_SOURCE(SysTick_CLKSource));

# l- n  m. Q" T3 h9 \! g, K
  if (SysTick_CLKSource == SysTick_CLKSource_HCLK)
+ L( r( x$ z" P8 k  o
  {
+ Q" H  Y  [# Q) ~2 H
    SysTick->CTRL |= SysTick_CLKSource_HCLK;

* R/ v) R* }7 a8 m  b8 b5 O
  }
) L1 @! A& M8 G$ @3 X, f3 J
  else

& d) W* N5 u4 h1 M. D) l
  {
! g6 Q4 M! R- [' m
    SysTick->CTRL &= SysTick_CLKSource_HCLK_Div8;
% M# F1 |8 Z3 z; o5 d' w
  }

7 y& q/ A) S1 x. r) a/ }3 T
}
( |4 N7 K2 L8 o1 |6 P

9 @& m1 V! V- k, [) S1 \1 O * _* s  t( e- o; U$ ]9 \2 ~' d7 c) Z
Systick时钟源的定义:
* O+ O3 g: i% ^$ `
/** @defgroup SysTick_clock_source

, J+ H0 z6 Q; p2 b' B- o
  * @{
1 i# b% L4 ]: u  d
  */

+ V/ k: D, Q$ B6 h. C9 n9 u* G! k
) z! l1 B. B9 o, ^0 |( p
#define SysTick_CLKSource_HCLK_Div8    ((uint32_t)0xFFFFFFFB)//将控制状态寄存器的第二位置0,即用外部时钟源
! M. ?) D4 w& ^9 C* n6 k( n
#define SysTick_CLKSource_HCLK         ((uint32_t)0x00000004)//将控制状态寄存器的第二位置1,即用内核时钟
4 v  g! U. A6 }5 O- P( y& Y
#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SysTick_CLKSource_HCLK) || \

( Z7 L8 v' m% v* a' T/ S8 c$ |
                                       ((SOURCE) == SysTick_CLKSource_HCLK_Div8))
  o# I  o' c7 D6 ~* ~

+ a7 ^% C4 a1 S
Systick定时时间的设定:
/ ^  B' a5 M5 S! T- Z

/ |: T  o5 \3 L( b+ m8 W
重装载值=systick 时钟频率(Hz)X想要的定时时间(S)

/ x0 a  n0 ~# y2 r- |/ e
如:时钟频率为:AHB的8分频;AHB=72MHz那么systick的时钟频率为72/8MHz=9MHz;要定时1秒,则
1 ~, Y* R2 b- o7 f! J& G: ~
重装载值=9000000X1=9000000;
# n8 q9 H" g+ n6 a
定时10毫秒

! D; u- v1 ~( u3 l0 `, i2 f
重状态值=9000000X0.01=90000

' s: C8 f* }2 H6 p3 e+ N: B7 C
Systick的中断处理函数,

" B4 D+ V1 @7 w
在startup_stm32f10x_hd.s启动文件中有定义。
7 A9 m9 y  q" u- e9 ?1 C2 m/ l7 M
DCD     SysTick_Handler            ; SysTick Handler

: {7 E& v4 w) f1 z$ Z
根据需要直接编写中断处理函数即可:
) z) X* Z; K. _) j" s& ^& w
Void SysTick_Handler (void)
- s" F6 c1 r) V8 r" g0 E
{ ;}
- e" C4 h0 X8 r5 _" X7 C( U
注意:

" V0 |# ?3 u8 i% i, Z3 @
如果在工程中,加入了stm32f10x_it.c,而又在主函数中编写中断函数,则会报错。; i! U! x* w1 d- O
3 f' i/ `  I( d6 m4 k

3 N9 S* k8 f8 q/ D, @* W: ]
因为在stm32f10x_it.c文件中,也有这个中断函数的声明,只是内容是空的。
) `5 L) _' u7 g( @# }+ s
/**

9 e* z9 h" m2 Z7 [
  * @brief  This function handles SysTick Handler.
# a  L% D3 R1 @3 g+ s6 E+ E( Z
  * @param  None

" B5 A2 E5 i1 z3 F
  * @retval None
8 W, J, ?  G6 Q
  */
' ^4 }, P1 r! {; b5 `) r9 W6 u
void SysTick_Handler(void)

8 w4 X+ J& M. M$ }6 q
{
1 L3 g: [0 `& n+ I- F5 h' a6 V
}0 V7 q# |; `: x- R
" f% d( X4 [( y
中断优先级的修改

; |% G" ^2 @( q8 h! I3 V4 w0 i
在调用SysTick_Config(uint32_t ticks)之后,调用 void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)。这个函数在core_cm3.h头文件中。
. d( P- |; A1 v* Y9 M
具体内容如下:
4 h% P0 F4 ?: ^, ^& r" v
2 N% B4 l  Y* J2 U
/**

* n: }( a- n: z
* @brief  Set the priority for an interrupt

. r7 U% s* g3 P: w) @
*

% A. H& K3 J! f. [3 ]. o( p2 |, Q
* @param  IRQn      The number of the interrupt for set priority
5 g0 l0 u, Z0 |& J! s
* @param  priority  The priority to set
+ m% g& q" E0 O" k8 a. p5 H0 y
*

+ q3 c' d0 I. n* V) m) o
* Set the priority for the specified interrupt. The interrupt

$ ?. b% N# R3 E2 x
* number can be positive to specify an external (device specific)

: k! r7 ~5 s. Y) h
* interrupt, or negative to specify an internal (core) interrupt.
3 V" e( j! ]! H/ S4 K% `" N5 r
*
. q% H& H" X# X- m6 r7 M
* Note: The priority cannot be set for every core interrupt.

8 ~  A4 }7 i: _; W, W: D4 o  W+ q! d
*/
# m4 |0 r. [. M; @" x# a1 C/ `
3 K7 b) V( ]) I  N8 Y; s+ ?* U/ B
$ @3 @8 n, i: F9 N: e$ X
' a+ N: Q4 I) ]) [; Y1 x
7 s- _' f' C$ _" j5 o5 O* K1 ?
static __INLINE void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)
4 `+ F7 T9 J2 f0 `
{

) l3 F# t& ]$ j! v7 A
  if(IRQn < 0) {

) `& Y" I3 {) H) _( E4 ^
    SCB->SHP[((uint32_t)(IRQn) & 0xF)-4] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff); } /* set Priority for Cortex-M3 System Interrupts */

  x# ?% \4 w* J
  else {

8 A1 n1 ^7 _% l$ ]/ Z' C
    NVIC->IP[(uint32_t)(IRQn)] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff);    }        /* set Priority for device specific Interrupts  */
# ?8 h4 `: _3 t' n8 S. q: L
}

3 n, @) E( n7 E* G
" B$ s* x+ {3 p  Y! y+ `( M: w$ Y% l
8 T/ E0 s, _: M
下面以一个实例来说明:

$ @' i9 o+ i% ^; `- t. ?  P; E0 B: b" o
利用systick来实现以1秒的时间间隔,闪亮一个LED指示灯,指示灯接在GPIOA.8,低电平点亮。
" R  M% V) z/ Z2 F$ p% \0 z
2 \6 _: B& M' L2 E8 V) U
#include "stm32f10x.h"

0 v  U: z: D4 b1 {, z+ ]& Y5 C
//函数声明
$ n0 m. Y# P) i9 D0 C" m9 C# z- K
void GPIO_Configuration(void);//设置GPIOA.8端口

; Y* @( c  w# q, g: F9 d# a4 \4 ]
u32 t;//定义一个全局变量
: `+ D: \0 Z% z* `
int main(void)
{
// SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
       SysTick_Config(9000000);
       SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
       GPIO_Configuration();
       while(1);      
}

7 \* a% H. {4 I
//GPIOA.8设置函数
void GPIO_Configuration(void)
{
GPIO_InitTypeDef  GPIO_InitStruct;//定义一个端口初始化结构体
       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//打开GPIOA口时钟
       GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;//设置为推挽输出
       GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;//设置输出频率50M
       GPIO_InitStruct.GPIO_Pin=GPIO_Pin_8;//指定第8脚
       GPIO_Init(GPIOA,&GPIO_InitStruct);//初始化GPIOA.8      
       GPIO_SetBits( GPIOA,  GPIO_Pin_8);//置高GPIOA.8,关闭LED
}
//systick中断函数
void SysTick_Handler(void)
{
t++;
       if(t>=1)
       {
              if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==1)
              {GPIO_ResetBits( GPIOA, GPIO_Pin_8);}      
       }
       if(t>=2)
       {
              if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==0)
                     {GPIO_SetBits( GPIOA, GPIO_Pin_8);}
                     t=0;
       }
}

; H6 y! S* a! V8 x
模拟后的结果
1、8分频后结果
& D. R& s; u& l
  ?2 U6 J/ a2 R( \
2、直接调用SysTick_Config(9000000);即不分频的结果,间隔为1/8=0.125s
0 y5 U7 p% ^5 K& s
/ D% {6 J4 P) y: C- y5 l* S
总结:
1、要使用systick定时器,只需调用SysTick_Config(uint32_t ticks)函数即可,
   自动完成了,重装载值的装载,时钟源选择,计数寄存器复位,中断优先级的设置(最低),开中断,开始计数的工作。
2、要修改时钟源调用SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)。
3、要修改中断优先级调用
     void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)
" ?4 ]6 K8 P1 R6 \; ~% i& ^
应用说明:
1、因systick是一个24位的定时器,故重装值最大值为2的24次方=16 777 215,
   要注意不要超出这个值。
2、systick是cortex_m3的标配,不是外设。故不需要在RCC寄存器组打开他的时钟。
3、每次systick溢出后会置位计数标志位和中断标志位,计数标志位在计数器重装载后被清除,而中断标志位也会随着中断服务程序的响应被清除,所以这两个标志位都不需要手动清除。
4、采用使用库函数的方法,只能采用中断的方法响应定时器计时时间到,如要采用查询的方法,那只能采用设置systick的寄存器的方法,具体操作以后再做分析。

; {$ p: s; H7 d0 r1 E4 H

该用户从未签到

2#
发表于 2022-7-29 11:07 | 只看该作者
好长的文章啊。大佬发表的帖子,感觉很厉害,很棒

该用户从未签到

3#
发表于 2022-7-29 13:53 | 只看该作者
学习学习一下,谢谢楼主

该用户从未签到

4#
发表于 2022-7-29 14:44 | 只看该作者
这个,实在是太高端了。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

关闭

推荐内容上一条 /1 下一条

EDA365公众号

关于我们|手机版|EDA365电子论坛网 ( 粤ICP备18020198号-1 )

GMT+8, 2025-6-16 11:45 , Processed in 0.109375 second(s), 23 queries , Gzip On.

深圳市墨知创新科技有限公司

地址:深圳市南山区科技生态园2栋A座805 电话:19926409050

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