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

STM32的SYSTICK详解

[复制链接]

该用户从未签到

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

EDA365欢迎您登录!

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

x
什么是SYStiCK:' o9 g2 F( s8 k$ w' L# D
这是一个24位的系统节拍定时器system tick timer,SysTick,具有自动重载和溢出中断功能,所有基于Cortex_M3处理器的微控制器都可以由这个定时器获得一定的时间间隔。
6 W0 Q, T. X, i3 K  s
作用:
3 |$ O, e3 f) [* m3 ~
在单任务引用程序中,因为其架构就决定了它执行任务的串行性,这就引出一个问题:当某个任务出现问题时,就会牵连到后续的任务,进而导致整个系统崩溃。要解决这个问题,可以使用实时操作系统(RTOS).
' i% i- ?, l4 |' o; r& C& j7 F
因为RTOS以并行的架构处理任务,单一任务的崩溃并不会牵连到整个系统。这样用户出于可靠性的考虑可能就会基于RTOS来设计自己的应用程序。这样SYSTICK存在的意义就是提供必要的时钟节拍,为RTOS的任务调度提供一个有节奏的“心跳”。

3 ]4 ~" _1 r0 a- _' v' [3 o
微控制器的定时器资源一般比较丰富,比如STM32存在8个定时器,为啥还要再提供一个SYSTICK?原因就是所有基于ARM Cortex_M3内核的控制器都带有SysTick定时器,这样就方便了程序在不同的器件之间的移植。而使用RTOS的第一项工作往往就是将其移植到开发人员的硬件平台上,由于SYSTICK的存在无疑降低了移植的难度。
" W! C& D' W! n) t4 Z
5 i, Y, E% p. V0 B2 {- F
  SysTick定时器除了能服务于操作系统之外,还能用于其它目的:如作为一个闹铃,用于测量时间等。
4 ^+ H* V9 d% \) q& c( L9 h% R
要注意的是,当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作。
$ ~$ X5 I0 U, H' \+ ]* i2 R/ h

9 e+ t0 l: W1 k% Z
时钟的选择:
& M' S& m, d: J9 _
用户可以在位于Cortex_M3处理器系统控制单元中的系统节拍定时器控制和状态寄存器(SysTick control and status register ,SCSR)选择systick 时钟源。如将SCSR中的CLKSOURCE位置位,SysTick会在CPU频率下运行;而将CLKSOUCE位清除则SysTick会以CPU主频的1/8频率运行。
& M& l( ^4 t, B# z& N
3.5版本的库函数与以往的有所区别

+ E5 A: g& ?2 ^* y5 G; S
不存在stm32f10x_systick.c文件,故原来的一些函数也不存在,比如SysTick_SetReload(u32 reload);SysTick_ITConfig(FunctionalState NewState);等

8 s4 K# N7 d( L
+ Y/ w, V+ {8 d' z6 i0 U
在3.5版本的库函数中与systick相关的函数只有两个

- A+ K  q0 }- e" z8 s
第一个,SysTick_Config(uint32_t ticks),在core_cm3.h头文件中进行定义的。
" S) H& R4 K  Q( l" m+ Z
第二个,void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),在misc.c文件中定义的。
8 C8 W% d9 I* z+ j( t. g- N

$ o( _  ?% e) z& h& w$ h" U# O! d1 l
SysTick_Config(uint32_t ticks),在core_cm3.h
: f' a5 @2 a3 Y( P. w- L. U
主要的作用:
/ w1 w# k; |# r7 n8 n4 Q% K" M
1、初始化systick
; T: a( r$ G. o+ _; L2 d5 N; D3 v2 U
2、打开systick
% y0 g+ M5 j' z; k6 L8 S
3、打开systick的中断并设置优先级

9 Y4 r0 Q$ Q8 U, X9 C) {! P
4、返回一个0代表成功或1代表失败
" f4 Y. `2 G9 [( i4 {& p
注意:

) v3 X" G$ x' r8 e4 x; ~
Uint32_t ticks  即为重装值,

, Q% Y. N' {7 L) O$ U" O  M
这个函数默认使用的时钟源是AHB,即不分频。
$ v3 u& F2 F& a. p4 J
要想分频,调用void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),

! i  G5 s8 ?7 B$ J9 N: P
但是要注意函数调用的次序,先SysTick_Config(uint32_t ticks),

, e% s3 [6 J- m' o, a6 {4 ~# F
后SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)

! W# d: q  I+ Q) Z" J  v
3 r" x. Z; U: N; N4 @ # E: `2 {1 r# x9 l# K% V5 F4 Y' O
, r* ~& X. r% N9 P

( `$ }6 i* H. G, y9 U; h
函数说明:
* b9 r" ]- y- p
( t2 D+ y7 x# l/ J; s4 `
/**
/ m- i& R$ h9 J6 E
* @brief  Initialize and start the SysTick counter and its interrupt.
6 ~9 Z6 n' u, Z! G
*
" B1 q) Q  `6 t  i8 s
* @param   ticks   number of ticks between two interrupts
, Y$ K' ?- i* n, o3 R, T
* @return  1 = faiLED, 0 = successful
! m9 y$ F" g" a# G7 ^
*
; }  `3 Y3 b# N
* Initialise the system tick timer and its interrupt and start the
6 U1 N  b4 Y# M% ~" V0 |4 y' M
* system tick timer / counter in free running mode to generate

' p7 }% y4 z/ k- o5 l7 b9 o
* periodical interrupts.

: A5 s2 g; ?+ g7 K- m
*/

# u  D) J7 ~8 Y5 T
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
. C: m2 Y' o- b! @) g( k& e2 b5 a
{

& i9 ~; |6 I$ ~& R$ y7 c! t
  IF (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            

7 `) ^+ q' d# j* A7 h3 r
  /* Reload value impossible */重装载值必须小于0XFF FFFF,为什么,这是一个24位的递减计数器。

7 c5 b# a9 T% @. W4 e% m* w$ V( ?, d+ Q# r" g
  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;

* E) p) n% O5 {+ w9 e# K
     /* set reload register */设置重装载值,SysTick_LOAD_RELOAD_Msk定义见后面

3 g' f: ^4 k& O6 i
  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);

' _3 ?- Z' [; T9 T# \- E2 k7 G5 a
/* set Priority for Cortex-M0 System Interrupts */
1 y! J% R  F' j! N, H0 ^
  SysTick->VAL   = 0;

& H* _+ t* Q+ |% C6 H% V
  /* Load the SysTick Counter Value */
. Y+ f' S4 a3 t8 n* D6 C
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |
6 L0 G* J! S3 w$ R# A
                   SysTick_CTRL_TICKINT_Msk   |

( Z% N4 V& H/ E) I& t6 y. [
                   SysTick_CTRL_ENABLE_Msk;                  
5 y! m1 T! `2 }1 _: v: T
/* Enable SysTick IRQ and SysTick Timer */
% `9 H$ ]7 y) ^8 T8 c8 y. m  V
  return (0);

( ?) S) r3 P6 R# v. @  p
  /* Function successful */

* x8 t! P( c- O* Z% `9 A! ^4 v
}

8 j% Z. \$ P! N4 k, ^  C& d
#endif

7 O( D; P! L4 F
与systick相关的寄存器定义
2 _' ?; K, t; U9 X5 e. j' C. ?

- e: ?! \: p  i5 J) E
/** @addtogroup CMSIS_CM3_SysTick CMSIS CM3 SysTick
1 W# ^  h; s& H/ d  M
  memory mapped structure for SysTick
$ z7 W/ v5 T' P8 _% ~% ]" Y
  @{

) D8 u1 {$ N, A, ]4 A
*/

: Z& P( }9 C' ?9 C4 M2 \* b: w6 h1 x) i
typedef struct

* G' |$ W  s) o+ F/ Z
{
1 |4 p* x, _8 K( z( ^, F/ F/ d
  __IO uint32_t CTRL; /*!< Offset: 0x00  SysTick Control and Status Register */
! h* q4 a3 S1 l6 y+ V, P
  __IO uint32_t LOAD; /*!< Offset: 0x04  SysTick Reload Value Register       */
, ^' X6 B( u# {" o
  __IO uint32_t VAL; /*!< Offset: 0x08  SysTick Current Value Register      */

1 p/ ]. |9 `- c- w
  __I  uint32_t CALIB; /*!< Offset: 0x0C  SysTick Calibration Register        */
& q' t! m' G6 l
} SysTick_Type;
1 O7 I1 E: T2 h/ u, b
$ O6 u  ~! j  d  }/ I
与systick寄存器相关的寄存器及位的定义

6 A" ?" u+ x) I# K4 X3 O$ \# ]4 \$ p  u8 C
/* SysTick Control / Status Register Definitions */控制/状态寄存器

9 C5 c1 U/ p6 L& k, g0 A#define  SysTick_CTRL_COUNTFLAG_Pos  16      /*!< SysTick CTRL: COUNTFLAG Position */
, W- c( f# M: a, z- N
#define SysTick_CTRL_COUNTFLAG_Msk         (1ul << SysTick_CTRL_COUNTFLAG_Pos)         
# u! u5 p1 N$ p  _) m( ?
/*!< SysTick CTRL: COUNTFLAG Mask */ 溢出标志位

: {3 J1 N0 h+ s2 P- m  a% U! N/ S* e' [
#define SysTick_CTRL_CLKSOURCE_Pos   2       /*!< SysTick CTRL: CLKSOURCE Position */
8 }, B( F; N% O
#define SysTick_CTRL_CLKSOURCE_Msk         (1ul << SysTick_CTRL_CLKSOURCE_Pos)  
2 A4 N/ a- ?+ m9 g# w; V/ ]
/*!< SysTick CTRL: CLKSOURCE Mask */时钟源选择位,0=外部时钟;1=内核时钟

6 t) K2 n$ L$ [+ Z+ j' R, V6 N' }0 C5 D, c' D1 K# l
#define SysTick_CTRL_TICKINT_Pos      1        /*!< SysTick CTRL: TICKINT Position */
+ |  I& ~4 W9 _" m
#define SysTick_CTRL_TICKINT_Msk           (1ul << SysTick_CTRL_TICKINT_Pos)         
$ w9 O( |! P% e$ d
/*!< SysTick CTRL: TICKINT Mask */异常请求位

5 h$ q2 Q. p& U5 X2 R& m6 I2 e% h0 [  C6 L
#define SysTick_CTRL_ENABLE_Pos             0       /*!< SysTick CTRL: ENABLE Position */

( {( p1 e, X9 W: U! z
#define SysTick_CTRL_ENABLE_Msk            (1ul << SysTick_CTRL_ENABLE_Pos)               

6 n: \, y! n  \( Y1 [5 m2 {/ {& P( ]
/*!< SysTick CTRL: ENABLE Mask */使能位

( g6 D- }! k* M$ A& O; h/ U, L3 z: Z$ s) e+ e
/* SysTick Reload Register Definitions */

8 ]3 R+ L7 S4 V$ z0 `
#define SysTick_LOAD_RELOAD_Pos             0    /*!< SysTick LOAD: RELOAD Position */

( r, ]% ~4 ^! Y6 Z! F
#define SysTick_LOAD_RELOAD_Msk            (0xFFFFFFul << SysTick_LOAD_RELOAD_Pos)        

$ n+ H! \4 |6 a$ F
/*!< SysTick LOAD: RELOAD Mask */

# F) D3 e4 {7 O/ |' H0 R& p% ]  |+ Y) b6 n/ `' V. f+ q
/* SysTick Current Register Definitions */

/ |% l' W2 f& v2 X# S+ T
#define SysTick_VAL_CURRENT_Pos             0       /*!< SysTick VAL: CURRENT Position */

. _: g2 v1 [4 n' u
#define SysTick_VAL_CURRENT_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        

/ H" L( H6 q. X+ {$ e0 u2 v
/*!< SysTick VAL: CURRENT Mask */

2 P1 l  E9 I4 ]8 ]- ^
9 @/ Y: f7 Z6 e  i
/* SysTick Calibration Register Definitions */
# h9 C$ R3 v$ i$ v4 l4 N! k5 e; X
#define SysTick_CALIB_NOREF_Pos            31      /*!< SysTick CALIB: NOREF Position */

8 l- b- {. y4 y. Z
#define SysTick_CALIB_NOREF_Msk            (1ul << SysTick_CALIB_NOREF_Pos)              
7 q: J; M$ X& V3 @; q
/*!< SysTick CALIB: NOREF Mask */

0 R& H2 _* n. d# B5 D% Z
5 P7 w2 i7 G+ K* l( k3 J+ K
#define SysTick_CALIB_SKEW_Pos             30       /*!< SysTick CALIB: SKEW Position */
2 Z9 n  |* G  ?: b) e' p  z. g
#define SysTick_CALIB_SKEW_Msk             (1ul << SysTick_CALIB_SKEW_Pos)               
) v- B' s. S+ S0 @
/*!< SysTick CALIB: SKEW Mask */

$ l% J1 B5 E# t2 Z. q% K
/ p- f6 w6 t, W: p5 z$ P
#define SysTick_CALIB_TENMS_Pos             0       /*!< SysTick CALIB: TENMS Position */

( W, X/ e9 M. F: _
#define SysTick_CALIB_TENMS_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        /*!< SysTick CALIB: TENMS Mask */

  M/ k( B6 o0 U* M% P5 G7 B
/*@}*/ /* end of group CMSIS_CM3_SysTick */

, J# z' @$ i4 u- K, L+ Q: x* D% D5 k- I7 ^( h6 P, m
与systick相关的寄存器的说明8 x# U' ~) V% _. s; q% N5 M

; K/ F0 M! I( m! Z
' k' C( {% c* Z# J . \/ x6 ~6 n; k. a" H3 ?3 I7 A- P& q

+ m2 g4 Y8 O. y% e. ~4 x
! [5 n1 s8 a( ]- t# x
" m4 F$ C# L% J* c2 {' h" }# w
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
; U- [6 o4 p/ A
作用:
& k& c2 s- w# g- U6 i- _+ _+ J, f
选择systick的时钟源,AHB时钟或AHB的8分频

1 F! P' }( c( o3 y4 N& N  _
默认使用的是AHB时钟,即72MHz
( F) O: V$ m- m" [  i
1 [( H: e2 s, J: t  R* w+ e
函数说明:
& U  Q* S; F$ R! }0 j! W
/**

( X* F2 e; U. X) d6 J
  * @brief  Configures the SysTick clock source.
. I2 H& u0 W9 k% `
  * @param  SysTick_CLKSource: specifies the SysTick clock source.
, J0 d' k" C" t6 n9 V- T0 z2 X
  *   This parameter can be one of the following values:
" z* B4 o9 ]5 Y# s4 j
  *     @arg SysTick_CLKSource_HCLK_Div8: AHB clock divided by 8 selected as SysTick clock source.

0 Q; R* ^2 E) G; L
  *     @arg SysTick_CLKSource_HCLK: AHB clock selected as SysTick clock source.
4 E- W4 ~) M! e( i5 U  Z
  * @retval None

/ f3 @: e" l- J9 j1 F$ F1 R5 o2 Y
  */
& ~9 j3 |# A8 J
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)

: {/ |6 ], ^7 F0 t
{

! N8 q( |6 w9 f% i1 r1 o
  /* Check the parameters */

0 K, _: I- B# e* G/ ^. w
  assert_param(IS_SYSTICK_CLK_SOURCE(SysTick_CLKSource));
3 k! O/ J4 Y) y% x) G4 ]
  if (SysTick_CLKSource == SysTick_CLKSource_HCLK)
5 x- ^, }% [; I
  {

( y; _/ O* G3 q( ^4 b
    SysTick->CTRL |= SysTick_CLKSource_HCLK;
1 b) {) {) J% e' S5 V5 n/ K
  }

. G2 [) G. B; i) J. @  ?4 L( p
  else

# }' m) |' B" l. l; l4 x
  {

2 h2 C" c0 H$ a
    SysTick->CTRL &= SysTick_CLKSource_HCLK_Div8;

6 ]1 n9 ^4 C3 {; q6 d! d2 A
  }

) w9 V) `6 A& l  K, [! r
}

, c4 J; ^# @7 v2 F, P* T( A1 a
! t' a* Q7 i/ Q
$ L! G6 O  `" o" z3 W. ?
Systick时钟源的定义:
% h7 \! Z- O! b1 Q6 V% s9 R
/** @defgroup SysTick_clock_source

4 w- V$ ]# ^* L8 V* `
  * @{

4 {% ~" |2 B% F4 O! X. G' d  T, w
  */
" G7 B: }9 @' z' s0 `4 N& z' W
0 l! J3 q+ g4 K# M, G5 M
#define SysTick_CLKSource_HCLK_Div8    ((uint32_t)0xFFFFFFFB)//将控制状态寄存器的第二位置0,即用外部时钟源

# @4 J* L: O. N1 j: [
#define SysTick_CLKSource_HCLK         ((uint32_t)0x00000004)//将控制状态寄存器的第二位置1,即用内核时钟

' ^) c% o; E* G+ S) l8 `+ D( E$ i
#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SysTick_CLKSource_HCLK) || \
; z; k7 S; K/ K! \/ `
                                       ((SOURCE) == SysTick_CLKSource_HCLK_Div8))

; S3 c9 Z+ f& F& u$ B5 A6 E, m7 |0 p( K! ]6 z- p2 z& h
Systick定时时间的设定:
$ d3 x7 b- S3 l

6 |1 I; i* D1 Q  g4 G8 I# o
重装载值=systick 时钟频率(Hz)X想要的定时时间(S)

, i$ |0 c% D0 o' g
如:时钟频率为:AHB的8分频;AHB=72MHz那么systick的时钟频率为72/8MHz=9MHz;要定时1秒,则
. K& W' o3 J8 ?
重装载值=9000000X1=9000000;
' S: R  O( c/ C
定时10毫秒
) I! G$ ]" d- W7 U" D" B9 O$ s
重状态值=9000000X0.01=90000
4 |, |+ f! ?6 G
Systick的中断处理函数,
2 v  Y) Q) X1 d# I4 S+ d9 @  O. t- P
在startup_stm32f10x_hd.s启动文件中有定义。
. I% e( @2 ]# s1 t5 H- ~9 {
DCD     SysTick_Handler            ; SysTick Handler
2 ?( L1 z. M* b( }- _$ r/ R" z
根据需要直接编写中断处理函数即可:

* a2 Q1 [& ^' P; F1 g# t
Void SysTick_Handler (void)

2 m6 L# f, j; {7 t7 t
{ ;}
+ P6 Y) U' ^3 \, r' r
注意:

4 ]$ L9 s6 U! i. D2 d9 S+ l$ B
如果在工程中,加入了stm32f10x_it.c,而又在主函数中编写中断函数,则会报错。# M9 A: Z: O1 a: x
$ G8 J& M! L3 _' J! ?* L! _' c

9 l5 E2 L  j- [2 j0 N6 u  ?
因为在stm32f10x_it.c文件中,也有这个中断函数的声明,只是内容是空的。
# n; m0 B) G5 t$ ^3 c
/**
8 [9 l2 j' Q+ M" N1 g( s% c
  * @brief  This function handles SysTick Handler.

7 \6 S- c% D1 m: _0 [
  * @param  None

: `3 X6 F4 s  n" D4 C% h5 c/ X
  * @retval None
. X! f- G6 ?* H2 |3 M: T  ~
  */

: e( l1 V4 t& P; _! L3 G! Z
void SysTick_Handler(void)
5 h$ c$ \) {1 I7 Y
{

1 d( P) l! e  ]8 ^- E7 e3 A. h
}
( z2 i7 o; [; h1 O: p
8 d( F5 a! O! N& ]8 k' y
中断优先级的修改

0 w" V2 P: G/ N" Z8 J" y; P$ J: K; i
在调用SysTick_Config(uint32_t ticks)之后,调用 void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)。这个函数在core_cm3.h头文件中。
; g. {) r( Q$ W
具体内容如下:

! R! ?+ U# y! \& F* g
1 k  I( ^) k8 R7 K' c1 C+ |, v) M3 l* ?
/**
+ Z( n  F3 m+ D" K
* @brief  Set the priority for an interrupt

$ I* [& W' z+ D7 E1 p
*

. F6 Y$ A8 m5 q6 {5 W3 a
* @param  IRQn      The number of the interrupt for set priority
6 F* _3 Z, ~+ y2 |- m1 ^
* @param  priority  The priority to set
$ K% p9 m9 f# s0 w4 }& F4 b
*

6 x7 g- O* O6 C: S$ l3 U
* Set the priority for the specified interrupt. The interrupt
4 q  J# U, Q. ]+ g- u
* number can be positive to specify an external (device specific)

6 F5 _. g, _3 s. f
* interrupt, or negative to specify an internal (core) interrupt.

" i8 P, K& f/ j$ v. d9 f9 ^% @7 C
*

0 i. t$ A3 ^1 ?5 R
* Note: The priority cannot be set for every core interrupt.
# y/ n! C  M0 W4 t: z# {3 B/ g( L4 \
*/

6 ~: F( ^( s. U+ k6 [! M4 B; l2 J0 B8 R  a# Y$ g

/ G+ Q- F, ~" t1 w$ I' j
5 @7 m% w3 g! y9 l" }. a% X6 y " j& o) i. R. F% U" `1 k; M4 ?. k
static __INLINE void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)

$ L6 \. \: h, w( g
{
- j9 Q" S. e5 [6 i
  if(IRQn < 0) {
3 `+ n5 h9 r: j/ f/ k) y% _
    SCB->SHP[((uint32_t)(IRQn) & 0xF)-4] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff); } /* set Priority for Cortex-M3 System Interrupts */

* H) m& u" V5 v) v" q' S
  else {
. X' H; j2 b6 H' ^( h: [6 `8 E
    NVIC->IP[(uint32_t)(IRQn)] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff);    }        /* set Priority for device specific Interrupts  */

. ?  V0 Z4 b: a0 W* p
}

. W$ \: Y! T' K( B. ]7 H# W' j" r9 f" I4 k, k" L: T4 v
" L) f- Y; O8 y: Q" n
下面以一个实例来说明:
# h/ R" \! T9 w' N  A" p
利用systick来实现以1秒的时间间隔,闪亮一个LED指示灯,指示灯接在GPIOA.8,低电平点亮。

5 D$ S( u6 N  N* Y* q: y+ ]; y, @8 f: f' m" Z7 G( K' z" p
#include "stm32f10x.h"
4 Y( b9 J4 x2 N7 ?  M/ j+ U
//函数声明

9 E" M" r; Q" M
void GPIO_Configuration(void);//设置GPIOA.8端口

+ D/ u! x8 b5 c6 e
u32 t;//定义一个全局变量
( R+ W2 \- r: l% Y2 m8 l% o# a1 k
int main(void)
{
// SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
       SysTick_Config(9000000);
       SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
       GPIO_Configuration();
       while(1);      
}
3 E: @4 j  n- [0 Y3 M& H
//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;
       }
}
4 V0 i( ]9 X+ G- t8 H: R& v
模拟后的结果
1、8分频后结果1 }- V- L" p- T' u8 d+ j
/ Z2 Y* V$ J/ I6 |$ J  {- i/ A! H" S; M7 g0 r
2、直接调用SysTick_Config(9000000);即不分频的结果,间隔为1/8=0.125s
3 @# L) ?+ p( q. Z, h

3 j8 H% q' j9 D2 z) q4 j  Y
总结:
1、要使用systick定时器,只需调用SysTick_Config(uint32_t ticks)函数即可,
   自动完成了,重装载值的装载,时钟源选择,计数寄存器复位,中断优先级的设置(最低),开中断,开始计数的工作。
2、要修改时钟源调用SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)。
3、要修改中断优先级调用
     void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)
' P5 t$ y, Y- w7 o* D0 L" ?: }. \
应用说明:
1、因systick是一个24位的定时器,故重装值最大值为2的24次方=16 777 215,
   要注意不要超出这个值。
2、systick是cortex_m3的标配,不是外设。故不需要在RCC寄存器组打开他的时钟。
3、每次systick溢出后会置位计数标志位和中断标志位,计数标志位在计数器重装载后被清除,而中断标志位也会随着中断服务程序的响应被清除,所以这两个标志位都不需要手动清除。
4、采用使用库函数的方法,只能采用中断的方法响应定时器计时时间到,如要采用查询的方法,那只能采用设置systick的寄存器的方法,具体操作以后再做分析。

* K! c" Q, f' k' m$ Z

该用户从未签到

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-10-8 14:17 , Processed in 0.156250 second(s), 24 queries , Gzip On.

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

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

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