你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32 SYSTICK详解

[复制链接]
行色匆匆 发布时间:2017-12-25 22:11
转载什么是SYSTICK:/ l- E6 u) r) m8 x/ r
这是一个24位的系统节拍定时器system tick timer,SysTick,具有自动重载和溢出中断功能,所有基于Cortex_M3处理器的微控制器都可以由这个定时器获得一定的时间间隔。& y; k8 e# d. H0 z8 ]/ A
作用:) w; E; Z4 {" y0 p/ x6 L
在单任务引用程序中,因为其架构就决定了它执行任务的串行性,这就引出一个问题:当某个任务出现问题时,就会牵连到后续的任务,进而导致整个系统崩溃。要解决这个问题,可以使用实时操作系统(RTOS).
6 }9 ?7 X5 C- ]% |! X, _# y7 V因为RTOS以并行的架构处理任务,单一任务的崩溃并不会牵连到整个系统。这样用户出于可靠性的考虑可能就会基于RTOS来设计自己的应用程序。这样SYSTICK存在的意义就是提供必要的时钟节拍,为RTOS的任务调度提供一个有节奏的“心跳”。% i* v5 I# R) H
微控制器的定时器资源一般比较丰富,比如STM32存在8个定时器,为啥还要再提供一个SYSTICK?原因就是所有基于ARM Cortex_M3内核的控制器都带有SysTick定时器,这样就方便了程序在不同的器件之间的移植。而使用RTOS的第一项工作往往就是将其移植到开发人员的硬件平台上,由于SYSTICK的存在无疑降低了移植的难度。
- q7 C9 ]: r8 M7 \, p/ H. X! x0 {7 ]/ C! V2 h+ B
  SysTick定时器除了能服务于操作系统之外,还能用于其它目的:如作为一个闹铃,用于测量时间等。7 R0 s9 M4 m: ^9 |6 F: O" R0 H
要注意的是,当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作。; f  G/ Y# p% A- Z3 @, D. y  s
! p7 E  P$ C- X7 Q
时钟的选择:# e" ~. U2 M0 }$ E
用户可以在位于Cortex_M3处理器系统控制单元中的系统节拍定时器控制和状态寄存器(SysTick control and status register ,SCSR)选择systick 时钟源。如将SCSR中的CLKSOURCE位置位,SysTick会在CPU频率下运行;而将CLKSOUCE位清除则SysTick会以CPU主频的1/8频率运行。
$ ~8 k' o7 [, p; @* h* [3.5版本的库函数与以往的有所区别
2 U# k( B$ Z: Z- R# C不存在stm32f10x_systick.c文件,故原来的一些函数也不存在,比如SysTick_SetReload(u32 reload);SysTick_ITConfig(FunctionalState NewState);等
1 j4 j# N6 g5 I5 M
$ R. H  k* ^. L$ F在3.5版本的库函数中与systick相关的函数只有两个9 w' k4 i7 o$ c" u% E! F' U0 T
第一个,SysTick_Config(uint32_t ticks),在core_cm3.h头文件中进行定义的。
& M: _1 O5 n, Q# p/ |4 Q第二个,void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),在misc.c文件中定义的。
- \+ e. w6 D% y8 g& r7 m) }: z2 o$ X' B. s: E$ H
SysTick_Config(uint32_t ticks),在core_cm3.h3 d3 a' V& |3 [. K7 Q
主要的作用:
- R5 N- u. d! u6 H8 s8 A1、初始化systick! H% w& B' p& P. f( `6 K
2、打开systick
3 G/ X5 V$ _# _3、打开systick的中断并设置优先级
. H0 }7 \* s2 ?0 b8 T1 i4、返回一个0代表成功或1代表失败4 A6 G: Z$ Q! Y+ ]9 [# `" g, j
注意:
$ @0 `' _4 c  L6 g, h' V" wUint32_t ticks  即为重装值,
" j- k; X$ e' A" ~+ z这个函数默认使用的时钟源是AHB,即不分频。, A5 t( L. e/ c- V9 V8 K
要想分频,调用void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),( [1 d$ X0 u3 y2 u8 T
但是要注意函数调用的次序,先SysTick_Config(uint32_t ticks),
8 Q: U! o- v" Z5 ^, W7 a/ x, v5 j后SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
# ~5 i' W3 x  L1 M7 d5 k
! G1 L! g$ ]4 c* R" U3 H- q, d- S7 Y- I; o) a8 \' t

, S3 w- z: r, F7 }6 d
8 s+ D  y5 g4 Z5 V. [) b( J% q0 g函数说明:
" [6 B4 x3 m. I  K: X& f4 {/ p+ [
/**. a1 `$ z, l5 S( U) r0 a
* @brief  Initialize and start the SysTick counter and its interrupt.6 d9 t3 u' k( b! ]* o
*  |9 i2 W0 J, H( t/ C) c
* @param   ticks   number of ticks between two interrupts. z  Z9 M  K6 p' U% }& H! H0 M: S
* @return  1 = failed, 0 = successful% _' B8 A$ z* J- }
*. ?* L  [2 u' ]4 k& _
* Initialise the system tick timer and its interrupt and start the- u+ r5 Z- p3 E% Q# n/ f) B
* system tick timer / counter in free running mode to generate8 T( w  K) V* {  M. v
* periodical interrupts., N  H' ~: s, k3 g
*/. \0 b" V- @  ~% a' R. t$ V
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
3 F. `1 j. n1 O) [5 Z, _{9 G2 {& Y5 L: j* ]
  if (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            
4 m( [: `: X3 i* {4 ~  /* Reload value impossible */重装载值必须小于0XFF FFFF,为什么,这是一个24位的递减计数器。
+ q2 P( w7 w* c9 F- t
' Q, S' a4 E4 p, k0 }5 F& C  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;% r! T$ C! ^! ?. x9 @* I
     /* set reload register */设置重装载值,SysTick_LOAD_RELOAD_Msk定义见后面
; K7 m1 d! s. \0 [' H  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);1 |3 ?" z, v; }6 Y
/* set Priority for Cortex-M0 System Interrupts */
. n$ q' ?5 ?0 L+ s) ?  SysTick->VAL   = 0;4 C/ t, v( E$ L# p
  /* Load the SysTick Counter Value */
2 [; H$ O: t/ t& C2 c1 j* t  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |
: D: B9 U% }, O6 x6 e. n                   SysTick_CTRL_TICKINT_Msk   |
: K# i: @4 B5 _( h7 q                   SysTick_CTRL_ENABLE_Msk;                  ( k4 i) J5 s7 K0 ^. e+ i$ J
/* Enable SysTick IRQ and SysTick Timer */
3 a! ~1 r) E* P- H$ ~1 c( g  return (0);& J# f. W, w- n
  /* Function successful */9 [. P+ [4 Y2 ?: e* d
}: U; U1 a# |4 j+ ]5 [1 t' {
#endif4 A: `; g5 v0 @
与systick相关的寄存器定义' E* X/ u2 w8 `' [
/** @addtogroup CMSIS_CM3_SysTick CMSIS CM3 SysTick
+ }( I0 p/ G: J' }# h7 |7 y- m) y  memory mapped structure for SysTick
, n; K- i9 \+ `! u2 V  @{
4 c" e0 I' h3 p0 |- Z2 K6 }  e" [*/9 B& r( l, y( k+ y( s
typedef struct! l* R4 U$ f/ Q
{
  v* i9 g# z, A  __IO uint32_t CTRL; /*!< Offset: 0x00  SysTick Control and Status Register */: y/ t! ]2 i$ H
  __IO uint32_t LOAD; /*!< Offset: 0x04  SysTick Reload Value Register       */
& D$ B8 P$ V+ ~8 X1 t  __IO uint32_t VAL; /*!< Offset: 0x08  SysTick Current Value Register      */
8 b; H7 n( @7 K" F6 ~7 V  N# I  __I  uint32_t CALIB; /*!< Offset: 0x0C  SysTick Calibration Register        */
+ N7 b, N, S' l, w5 F: d5 ]" j} SysTick_Type;9 `. ~: O( l4 F! }
# y+ s, z/ T5 M- G. D! h
与systick寄存器相关的寄存器及位的定义# v" E4 H- H4 h; B% Q, s

( W- F6 y, {% l- t/* SysTick Control / Status Register Definitions */控制/状态寄存器! h: d. Y. L2 n, R
#define  SysTick_CTRL_COUNTFLAG_Pos  16      /*!< SysTick CTRL: COUNTFLAG Position */
! Z5 j- b* ?2 k, o! q#define SysTick_CTRL_COUNTFLAG_Msk         (1ul << SysTick_CTRL_COUNTFLAG_Pos)         
- J$ d7 J3 O& R1 w  P/*!< SysTick CTRL: COUNTFLAG Mask */ 溢出标志位
* W' \) x0 D9 b/ q& ?5 }/ ~
( a# y" u% C  |. ^' ~8 f- c8 e#define SysTick_CTRL_CLKSOURCE_Pos   2       /*!< SysTick CTRL: CLKSOURCE Position */
. E* O" s" H+ W& W  H& }#define SysTick_CTRL_CLKSOURCE_Msk         (1ul << SysTick_CTRL_CLKSOURCE_Pos)  : v' U- k* X6 q' N6 e9 i4 ^
/*!< SysTick CTRL: CLKSOURCE Mask */时钟源选择位,0=外部时钟;1=内核时钟
; U! |# y" b# |5 j; U7 k$ p- B  t, E) G3 f! F' A
#define SysTick_CTRL_TICKINT_Pos      1        /*!< SysTick CTRL: TICKINT Position */  s0 G4 d) r% O& ?, q) y
#define SysTick_CTRL_TICKINT_Msk           (1ul << SysTick_CTRL_TICKINT_Pos)         
1 |8 K% Q7 n8 \4 d% `4 o& h0 Y2 v/*!< SysTick CTRL: TICKINT Mask */异常请求位0 y, ]- A5 k1 ?" d. a. T
/ b) F. a" @4 a! Z& B$ Z
#define SysTick_CTRL_ENABLE_Pos             0       /*!< SysTick CTRL: ENABLE Position */# A7 k& n5 j) o) `* D) W
#define SysTick_CTRL_ENABLE_Msk            (1ul << SysTick_CTRL_ENABLE_Pos)               
4 D( L6 I; N8 I1 Z; K8 a/*!< SysTick CTRL: ENABLE Mask */使能位1 s; L7 {" _- }0 p% F3 ~
6 K& P1 x; _, k$ {. X
/* SysTick Reload Register Definitions */" i. i3 c% ~. [3 d
#define SysTick_LOAD_RELOAD_Pos             0    /*!< SysTick LOAD: RELOAD Position */
% I4 Q& x; u- H5 z#define SysTick_LOAD_RELOAD_Msk            (0xFFFFFFul << SysTick_LOAD_RELOAD_Pos)        
3 N! v) W) [$ z5 a/*!< SysTick LOAD: RELOAD Mask */
) U) w2 U% a  c' K& a
" {  N: N: ~$ z+ M% A. \/* SysTick Current Register Definitions */3 E8 w5 h. E' i; H9 a* d. i1 A
#define SysTick_VAL_CURRENT_Pos             0       /*!< SysTick VAL: CURRENT Position */
( @1 E( d) @9 _3 z/ z( ]* }#define SysTick_VAL_CURRENT_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        9 S+ t3 T: d  y- z
/*!< SysTick VAL: CURRENT Mask */
4 G/ I. I; O6 V- m9 t7 Y' s8 L9 u# d2 F# r( o' M$ @
/* SysTick Calibration Register Definitions */
! y8 k1 f2 L/ h& ~. v#define SysTick_CALIB_NOREF_Pos            31      /*!< SysTick CALIB: NOREF Position */
; J0 ]- [+ y5 w7 T#define SysTick_CALIB_NOREF_Msk            (1ul << SysTick_CALIB_NOREF_Pos)              
$ j! x: b. l- S1 @* G0 \/*!< SysTick CALIB: NOREF Mask */
' E7 s' U5 P2 g
8 A" i% I5 L3 r$ l#define SysTick_CALIB_SKEW_Pos             30       /*!< SysTick CALIB: SKEW Position */; `/ o& N+ }5 B  v
#define SysTick_CALIB_SKEW_Msk             (1ul << SysTick_CALIB_SKEW_Pos)               
, N! M* H& ]6 [/*!< SysTick CALIB: SKEW Mask */8 G7 w5 W7 S( j! Z8 @; _

( Y& S# i* ^0 @#define SysTick_CALIB_TENMS_Pos             0       /*!< SysTick CALIB: TENMS Position */
, n6 {' v9 `. d( T' C4 E#define SysTick_CALIB_TENMS_Msk            (0xFFFFFFul << SysTick_VAL_CURRENT_Pos)        /*!< SysTick CALIB: TENMS Mask */2 S, F9 D* b# n: ?# |
/*@}*/ /* end of group CMSIS_CM3_SysTick */
; e' r7 o! u& y2 s0 C' y' c" @9 w1 P4 W# E- J. t1 U
与systick相关的寄存器的说明8 q9 s/ `5 W; V
1 L( Q2 U% U7 y6 f! w8 i
+ t6 G# x  B3 d) C

3 G; C, w( B5 _5 U+ z% L2 a* i$ K9 d
" N; g6 r. f) G+ L& ]" L8 a$ o  d
! s2 S- `1 ], y
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
& [0 p7 `6 F2 z" Y* \& j作用:% J7 _& x+ S& g9 A
选择systick的时钟源,AHB时钟或AHB的8分频: X7 R% p8 K# E, u) b. u- y
默认使用的是AHB时钟,即72MHz
( k& C* }) U& ?2 H  m3 A, t
2 ?$ m. N- E0 s: k) Y  O: |函数说明:* N* g) Y( F5 G  {* t
/**
8 H- A: n* [* }* C4 x/ d  * @brief  Configures the SysTick clock source.
5 [) V( @+ y6 d) k9 j, N, F" W  * @param  SysTick_CLKSource: specifies the SysTick clock source.' a! f6 E& d, z1 W1 M- j
  *   This parameter can be one of the following values:
* b/ n: |5 ]4 x/ A$ s0 I; Q( J  *     @arg SysTick_CLKSource_HCLK_Div8: AHB clock divided by 8 selected as SysTick clock source.4 b5 }* z9 \- k
  *     @arg SysTick_CLKSource_HCLK: AHB clock selected as SysTick clock source.* o1 f8 h4 J- D2 `+ v& {: I# T
  * @retval None
0 P) {5 T% D6 ^) H% Y/ e! y; j  */" }2 u) ~! f0 ~( D5 Y1 @
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
2 k8 J) d- B$ \8 f$ R3 v  a{4 ^* M. z8 {# U1 H; I
  /* Check the parameters */- L: n) ^0 O- J
  assert_param(IS_SYSTICK_CLK_SOURCE(SysTick_CLKSource));
3 T* P  f8 T) _/ A! R  if (SysTick_CLKSource == SysTick_CLKSource_HCLK)
$ A% f2 v5 R3 P; B  {
" f. z1 a. S# A) z    SysTick->CTRL |= SysTick_CLKSource_HCLK;6 ~1 D: V  ^4 _
  }
) X, B- C! f# _9 N, D  else
: X' S8 v! Z  [% r4 A: S, O' ?- h6 G  {
0 E# @5 }8 J) e9 M    SysTick->CTRL &= SysTick_CLKSource_HCLK_Div8;
- v, g' B* e& F9 R: A  }. G  B# `" j& A8 b
}, Y7 x# J; e; w

4 B* @% J6 z' v0 k* R" [6 P$ q( d: c5 Z0 |; v+ N
Systick时钟源的定义:
, R5 I1 p. W* q8 V5 d6 @+ ]/** @defgroup SysTick_clock_source8 S2 ^; `) y( d4 _
  * @{# H5 M6 L4 Z9 M; @* `4 m5 U
  */
1 w& [- p: ~0 J6 @- A# k
8 {# M" n' J, z( R7 ~; P+ {) o#define SysTick_CLKSource_HCLK_Div8    ((uint32_t)0xFFFFFFFB)//将控制状态寄存器的第二位置0,即用外部时钟源
1 l; ^9 ]6 w" T#define SysTick_CLKSource_HCLK         ((uint32_t)0x00000004)//将控制状态寄存器的第二位置1,即用内核时钟
1 M' E# g. t0 \) Q- Q/ x/ r#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SysTick_CLKSource_HCLK) || \
, C; E8 p5 O5 _9 \$ J                                       ((SOURCE) == SysTick_CLKSource_HCLK_Div8))
5 }9 i8 C6 O; r; ?! E2 u/ O% O1 G7 |3 e* H1 k
Systick定时时间的设定:
. P1 v0 B9 {, q7 f& {% W$ Y7 T+ R' @6 V
重装载值=systick 时钟频率(Hz)X想要的定时时间(S)& _! f- P* U- @$ l" u# M7 t
如:时钟频率为:AHB的8分频;AHB=72MHz那么systick的时钟频率为72/8MHz=9MHz;要定时1秒,则# a) t6 B% F- H4 }+ R8 R0 |
重装载值=9000000X1=9000000;2 _5 C# ?' q! J7 Y3 t
定时10毫秒# }. o; T3 j) o
重状态值=9000000X0.01=90000
2 {) [( S% R6 h) nSystick的中断处理函数,# X# Q" ^* h% K" V
在startup_stm32f10x_hd.s启动文件中有定义。$ ^# N: K2 D4 o6 A
DCD     SysTick_Handler            ; SysTick Handler
- c9 G) R$ G0 I. v$ r* w根据需要直接编写中断处理函数即可:
: C1 @8 J0 R  g5 b8 P- uVoid SysTick_Handler (void): e) d8 t2 a& J8 F/ j
{ ;}
( d+ Q; `  D" d5 W  w2 S3 y注意:' X! h. S( r) r2 L/ Q& ^! R+ v
如果在工程中,加入了stm32f10x_it.c,而又在主函数中编写中断函数,则会报错。+ m! W) W& N' r$ \) K! X4 p

; ^- @+ k; ^, `# h" G因为在stm32f10x_it.c文件中,也有这个中断函数的声明,只是内容是空的。4 c  v! p2 j! ?/ I; a2 M0 N
/**
5 B5 H2 N+ G6 U2 T  * @brief  This function handles SysTick Handler.8 r/ W4 g3 j  ~, a: Z% r$ K
  * @param  None
9 o' q# S& r; a% Q  * @retval None5 ~! B. I( q3 K4 E
  */3 _* B7 j- G$ K8 `5 ?0 R
void SysTick_Handler(void)7 t9 I) N1 t, c- c6 P3 h7 K  p2 d
{9 i) O# x% U7 c* A( W( O/ _
}
7 s/ ]' U" ]- W  ?7 \$ r中断优先级的修改  b* K; |8 F/ L8 _: S1 D4 `7 }) M  m
在调用SysTick_Config(uint32_t ticks)之后,调用 void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)。这个函数在core_cm3.h头文件中。
+ H; [; w4 q1 `) g$ e3 ^1 v具体内容如下:
% ]3 U0 k5 o# Z. N
) q* Q5 t& m( w- M' w( D2 P8 Y/**
) u8 R& n3 \6 d1 V- J0 n: E5 z* @brief  Set the priority for an interrupt
5 L! L3 E) A/ W2 ?*4 m" \3 n3 |9 j( ?/ w
* @param  IRQn      The number of the interrupt for set priority0 \. b' Y3 o% U; ]$ n$ \& ^* w
* @param  priority  The priority to set
* f. c" O+ @' V*7 G* I) N0 E8 d8 @" g0 C# Y& u
* Set the priority for the specified interrupt. The interrupt
* m4 a' g& M2 p* number can be positive to specify an external (device specific)6 c; y! |! v1 ]9 ^- |2 {* B
* interrupt, or negative to specify an internal (core) interrupt.
, o2 h0 n6 v' x1 O*5 B1 F& y1 v. X
* Note: The priority cannot be set for every core interrupt.
! ^3 f  s3 D; Y*/
5 }9 r6 e; i, S9 I, b" B/ ]' a4 l6 |
  Z+ }. \; o( X2 e

9 b2 ^: j6 B6 G( |  N- j9 v, Y% k9 M& q7 k* d# x
static __INLINE void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)
0 v# a  d" V0 P8 n, ]8 p{3 w- A, l& Y" w! Y7 v( M
  if(IRQn < 0) {
4 J! E2 w; Q  G! h0 M    SCB->SHP[((uint32_t)(IRQn) & 0xF)-4] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff); } /* set Priority for Cortex-M3 System Interrupts */
  S  x* V/ ~- Z3 J* x! G, @  else {6 J: R9 t4 j3 i: o1 _, U  @! c
    NVIC->IP[(uint32_t)(IRQn)] = ((priority << (8 - __NVIC_PRIO_BITS)) & 0xff);    }        /* set Priority for device specific Interrupts  */
7 v; Q) K# P4 }}
0 U; R. r5 I! d# Z3 s8 r
' V" }' B" v- J. P! Q3 q0 }' c/ R0 i7 [1 j
下面以一个实例来说明:
' N; j# T0 H; x8 ^利用systick来实现以1秒的时间间隔,闪亮一个LED指示灯,指示灯接在GPIOA.8,低电平点亮。
# Z) C4 i/ p& {1 `/ P  b
- @8 Q) a: L9 g  H9 `* G6 h#include "stm32f10x.h"
8 C' T% d4 }+ m//函数声明
& |7 p' o5 U3 g6 v- W# G/ vvoid GPIO_Configuration(void);//设置GPIOA.8端口
$ r6 C3 V7 l' \9 t) R  }! ^u32 t;//定义一个全局变量$ ?4 a; N4 B2 [. i8 O) a
int main(void)
6 t, M3 x# v" Q1 w+ \4 J8 d4 Q* S{( H" m8 V6 B" ]- Z7 {7 h
// SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
6 P" n( x3 y$ T- i       SysTick_Config(9000000);
$ `* ~  \- R, K  Q7 g       SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);, X- a. f, V9 x  X9 q2 I
       GPIO_Configuration();1 T* i. A% c5 ]2 [) K
       while(1);      9 d: ]4 s9 h; w
}
+ t) m1 U. _# E! p3 [9 ^0 B1 A5 c0 A. e2 N8 j1 P
//GPIOA.8设置函数
. R2 Q+ k) U( z8 Y2 kvoid GPIO_Configuration(void)8 A9 T& `5 E( @, o* ]
{) r' R" e( L) v1 \# G
GPIO_InitTypeDef  GPIO_InitStruct;//定义一个端口初始化结构体
; m0 N/ x, s- j. l& m       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//打开GPIOA口时钟
8 b' |4 H: j* y) N1 G7 b       GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;//设置为推挽输出5 x# {' @" ]3 j' ^
       GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;//设置输出频率50M
$ O$ m0 h  G7 T  Y       GPIO_InitStruct.GPIO_Pin=GPIO_Pin_8;//指定第8脚
# u' P) _0 E+ z; F       GPIO_Init(GPIOA,&GPIO_InitStruct);//初始化GPIOA.8      
; n3 ]1 d. w" k  i( O       GPIO_SetBits( GPIOA,  GPIO_Pin_8);//置高GPIOA.8,关闭LED7 }) p0 |8 g! |4 F, f: U
}
  k- Z* {4 ?+ T//systick中断函数
, ?/ }4 a  D6 e0 pvoid SysTick_Handler(void)
* u, d- `+ X1 t  Y: W{) M5 j" M- E* u- p; W" k
t++;! A4 i: ?" ^9 X
       if(t>=1)
" S9 Q8 B( x0 ?# b0 p       {# F, ~5 o2 \) m/ {5 o8 Q! Z
              if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==1)7 J1 v$ O* y" ^4 _7 x$ N3 W
              {GPIO_ResetBits( GPIOA, GPIO_Pin_8);}      6 p1 P$ @1 q9 @: q6 e2 S
       }
3 W8 c( z. K  p4 o! Y$ o7 ?5 d       if(t>=2)( w9 L/ |# B3 e
       {
, b; b$ ?+ D, }8 A* k; c& C              if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==0)' H' z; s, q- _8 H  A
                     {GPIO_SetBits( GPIOA, GPIO_Pin_8);}
6 q/ E2 m$ U: w8 u: P3 v                     t=0;1 s' f* c1 w5 h! Z
       }0 u- Z* S; G; f" y
}
" ^5 T* e# o! O) }7 B4 ~8 b' e/ R4 ~3 W
模拟后的结果
  J; Z) |: J2 w- v1 D1、8分频后结果" k1 v9 v) o9 R9 o2 {7 C
( A- j2 w: Q, C5 N

# k) {8 f' \) L# s5 E. e; o% K% A; d3 `# V$ V; t. N2 \
  I8 O$ r, A8 T. @+ ?
; t! D1 l5 p  F# u
2、直接调用SysTick_Config(9000000);即不分频的结果,间隔为1/8=0.125s$ g3 ^3 [" v! n5 e

5 A; o( n$ R- n0 K' z9 t/ p1 L: ~% ], {3 {, h0 P
总结:1 ^0 f( y$ ?, S/ K
1、要使用systick定时器,只需调用SysTick_Config(uint32_t ticks)函数即可,4 i5 v: J3 w) H0 X- J; P
   自动完成了,重装载值的装载,时钟源选择,计数寄存器复位,中断优先级的设置(最低),开中断,开始计数的工作。0 ?# a! G, C$ l$ P! e% ]
2、要修改时钟源调用SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)。
# B- {2 n7 a# Y- ^3、要修改中断优先级调用
! z# c0 P$ w; X( N" S- d     void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority), T" k# D2 `$ J: x. A
应用说明:
% l* ^4 u  z( w- O1、因systick是一个24位的定时器,故重装值最大值为2的24次方=16 777 215,, K& |8 k: n6 D; s  b! M# d
   要注意不要超出这个值。
* g4 S8 o9 D" T* I+ h" [2、systick是cortex_m3的标配,不是外设。故不需要在RCC寄存器组打开他的时钟。
# q9 w+ `2 c5 r, O3 e& n3、每次systick溢出后会置位计数标志位和中断标志位,计数标志位在计数器重装载后被清除,而中断标志位也会随着中断服务程序的响应被清除,所以这两个标志位都不需要手动清除。
) u: m4 q4 y$ v- z9 Y4、采用使用库函数的方法,只能采用中断的方法响应定时器计时时间到,如要采用查询的方法,那只能采用设置systick的寄存器的方法,具体操作以后再做分析。* t: a- [  @$ J) t: v% a
收藏 1 评论1 发布时间:2017-12-25 22:11

举报

1个回答
zero99 回答时间:2017-12-26 09:21:45
重复了) P1 e& t8 `% i4 F, k# ~8 C, X
/ }* y) w( z# R6 _1 R* ^* a( z
https://www.stmcu.org.cn/module/forum/thread-606435-1-1.html

所属标签

相似分享

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版