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

【安富莱】【RTX操作系统教程】第9章 任务运行在特权级或...

[复制链接]
baiyongbin2009 发布时间:2016-1-23 17:18
第9章      任务运行在特权级或非特权级模式
    本章教程为大家讲解RTX运行模式的一个重要知识点,特权级模式和非特权级模式,有些资料或者书籍将非特权级模式称为用户模式。
    本章教程配套的例子含Cortex-M3内核的STM32F103和Cortex-M4内核的STM32F407。
9.1 RTX任务特权级知识点说明
9.2 RTX任务特权等级深入认识
9.3 实验例程说明
9.4         总结
9.1  RTX任务特权级知识点说明
    对于初学者只需记住本小节的知识点即可,如果要深入的了解还是需要花些时间去研究下Cortex-M3或者M4权威指南。
    对于使用Cortex-M3或者M4内核的芯片来说,RTX操作系统可以让任务运行在特权级或者非特权级模式,这两种模式是M3或者M4内核本身所具有的特性。
    在特权级模式下,用户可以访问和配置系统控制寄存器,比如NVIC中断控制器。然而,如果是在非特权级模式下,系统控制寄存器是不允许访问的,一旦访问将导致硬件异常。
Unprivileged:
    非特权级,起到保护用户任务的作用,防止用户可以在任意任务中访问和修改系统寄存器,操作不当会造成系统崩溃。
Privileged:
    特权级,这种模式下用户可以在任意任务中对系统控制寄存器的访问和修改。
    有了这点基础的认识之后,还有以下四个疑问有待解决。
6 D. T5 @; d0 D
5 n5 z1 Y& V+ \
9.1.1      非特权级模式下那些寄存器不可访问
    对于Cortex-M3或者M4内核来说,所有的核心外设寄存器都是只能在特权级下才可以访问,那些又是核心外设寄存器呢,对于STM32来说需要大家看编程手册,如下这些核心外设所有的寄存器都需要在特权级下才可以访问:
9.1.png
                              
    核心外设主要是MPU,NVIC,SCB和STK四个单元。一些初学者要问了,你又是如何知道这些内核外设的寄存器要在特权级下才可以访问?答案就在这里,我们可以任意打开一个寄存器:
                           
9.2.png
0 Q; F+ s" B1 }9 u3 a
9.3.png
关于MPU,NVIC,SCB和STK四个单元的其它寄存器是否需要在特权级下才可以访问,大家可以按照我上面说的方法进行查看。
    除了核心外设寄存器以外,M3/M4内核的特殊功能寄存器也是不能在非特权级下访问的,特殊功能寄存器主要包括以下寄存器:
1.  程序状态寄存器组(PSRs或曰xPSR)
2.  中断屏蔽寄存器组(PRIMASK, FAULTMASK,以及BASEPRI)
3.  控制寄存器(CONTROL)
    对于参考手册上面所说的SPI,USART,USB等所有外设寄存器都是可以在非特权级下进行访问的。

# @" w' n: s  y) V/ d+ v: d0 g
! d1 j/ @4 a8 E
9.1.2      非特权级模式下核心外设寄存器如何初始化
如果用户将RTX操作系统的任务设置在非特权级模式下运行,那么核心外设寄存器应该放在哪里进行初始化呢,主要有以下两种方法:
1.  使用SVC(Supervisor Call)软中断,这个在第19章有详细讲解。
2.  在初始化和开启RTX多任务前做核心外设的初始化。
8 T+ C, B; z4 m1 c

) s' [, W+ p0 a4 S
9.1.3     Cortex-M3或者M4内核如何切换两种模式
    Cortex-M3/M4中的特殊功能寄存器包括:
1.  程序状态寄存器组(PSRs或曰xPSR)
2.  中断屏蔽寄存器组(PRIMASK, FAULTMASK,以及BASEPRI)
3.  控制寄存器(CONTROL)
    其中控制寄存器CONTROL是用来设置特权级和非特权级切换的,CONTROL寄存器定义如下:
9.4.png
CONTROL[0]
    仅当在特权级下操作时才允许写该位。一旦进入了用户级,唯一返回特权级的途径,就是触发一个软中断,再由服务例程改写该位。
CONTROL寄存器也是通过MRS和MSR指令来操作的:
    MRS   R0,    CONTROL
    MSR   CONTROL, R0

8 q. }9 b& Q" [3 N0 V

/ |; F+ U" @8 q3 T3 x7 ]+ K
9.1.4     RTX任务特权等级的设置方法
    RTX任务特权等级的设置方法比较简单,查看RTX系统的配置向导,如下图9.1所示:
9.5.png
图9.1 RTX配置向导
Run in privileged mode
    此参数就是用来设置特权级和非特权级的,选上单选框表示使能任务工作在特权级模式,取消单选框表示任务工作在非特权级模式。

% w- K( F# T- X9 ~' Z: }* T! M4 ], b6 |
收藏 3 评论5 发布时间:2016-1-23 17:18

举报

5个回答
baiyongbin2009 回答时间:2016-1-23 17:30:16
9.2  RTX任务特权等级深入认识
    本小节的知识点对于初学者来说比较难理解,需要积累了一定的经验后才能更好的理解,不过还是建议读一读。
    深入了解Cortex-M3/M4内核的特权等级就不得不说说两种操作模式,Cortex-M3/M4支持两种操作模式,两种操作模式分别是:
Handlermode,中断模式,简单的说就是指的异常服务程序是处在中断模式。
Threadmode,线程模式,简单的说就是异常服务程序以外的程序都是处在线程模式。
    Cortex-M3/M4内核实现这两种操作的目的就是区分普通应用程序的代码和异常服务程序。下面是两种操作模式和两种特权等级的关系:
9.6.png
中断和异常的区别
    在 ARM 编程领域中,凡是打断程序顺序执行的事件,都被称为异常(exception)。除了外部中断外,当有指令执行了“非法操作”, 或者访问被禁的内存区间,因各种错误产生的 fault, 以及不可屏蔽中断发生时,都会打断程序的执行,这些情况统称为异常。在不严格的上下文中,异常与中断也可以混用。另外,程序代码也可以主动请求进入异常状态的(常用于系统调用)。
0 m3 D7 X% o6 J- b: _
    当处理器处在线程状态下时,既可以使用特权级,也可以使用用户级;另一方面,handler模式总是特权级的。在系统复位后,处理器进入线程模式+特权级。
    在特权级下的代码可以通过置位CONTROL[0]来进入用户级。而不管是任何原因产生了任何异常,处理器都将以特权级来运行其服务例程,异常返回后,系统将回到产生异常时所处的级别。用户级下的代码不能再试图修改CONTROL[0]来回到特权级。它必须通过一个异常handler,由那个异常handler来修改CONTROL[0],才能在返回到线程模式后拿到特权级。下图是特权级线程模式和用户级线程模式的切换图:
                              
9.7.png
一些简单的应用程序是不需要用户级线程模式的,如下图所示:
9.8.png
    把代码按特权级和用户级分开对待,有利于使CM3/CM4的架构更加安全和健壮。例如,当某个用户程序代码出问题时,不会让它成为害群之马,因为用户级的代码是禁止写特殊功能寄存器和NVIC中断寄存器的。另外,如果还配有MPU,保护力度就更大,甚至可以阻止用户代码访问不属于它的内存区域。
    为了避免系统堆栈因应用程序的错误使用而毁坏,我们可以给应用程序专门配一个堆栈,不让它共享操作系统内核的堆栈。在这个管理制度下,运行在线程模式的用户代码使用PSP,而异常服务例程则使用MSP。这两个堆栈指针的切换是智能全自动的,就在异常服务的始末由硬件处理。
    如前所述,特权等级和堆栈指针的选择均由CONTROL负责。当CONTROL[0]=0时,在异常处理的始末,只发生了处理器模式的转换,如下图所示。
9.9.png    
    但若CONTROL[0]=1(线程模式+用户级),则在中断响应的始末,处理器模式和特权等极都要发生变化,如下图所示。
9.10.png
    CONTROL[0]只有在特权级下才能访问。用户级的程序如想进入特权级,通常都是使用一条“系统服务呼叫指令(SVC)”来触发“SVC异常”,该异常的服务例程可以视具体情况而修改CONTROL[0]。
           关于操作模式和特权级就跟大家说这么多,后面讲解RTX任务切换时再详述(此章节在后期RTX教程升级版本时再配套)
9 Q' y4 ^) w8 t- d6 e  |
baiyongbin2009 回答时间:2016-1-23 17:48:27
9.3  实验例程说明
* L. J; R3 x9 I. Y) [4 W. k6 f

6 Z+ h5 M) g) }$ ?" Z& a% j
9.3.1     STM32F103开发板实验
配套例子:
    V4-404_RTX实验_任务运行在用户模式(非特权级)
实验目的:
    1. 学习RTX的任务运行在用户模式的设置方法
实验内容:
    1. K1按键按下,串口打印。
     2. K2键按下,设置NVIC分组,从而导致系统进入到硬件异常HardFault_Handler。
    3. 各个任务实现的功能如下:
       AppTaskUserIF任务   :按键消息处理。  
       AppTaskLED任务     :LED闪烁。
       AppTaskMsgPro任务 :消息处理,这里用作LED闪烁。
       AppTaskStart任务    :启动任务,也是最高优先级任务,这里实现按键扫描。
RTX配置:
    RTX配置向导详情如下:
9.11.png
                              
Task Configuration
(1)Number of concurrent running tasks
         允许创建4个任务,实际创建了如下四个任务
                      AppTaskUserIF任务   :按键消息处理。
                      AppTaskLED任务     :LED闪烁。
                      AppTaskMsgPro任务 :消息处理,这里是用作LED闪烁。
                      AppTaskStart任务    :启动任务,也是最高优先级任务,这里实现按键扫描。
(2)Number of tasks with user-provided stack
        创建的4个任务都是采用自定义堆栈方式。
(3)Run in privileged mode
        设置RTX任务运行在非特权级,即用户模式。
RTX任务调试信息:
9.12.png
程序设计:
任务栈大小分配:
staticuint64_t AppTaskUserIFStk[512/8];   /* 任务栈 */
staticuint64_t AppTaskLEDStk[256/8];      /* 任务栈 */
staticuint64_t AppTaskMsgProStk[512/8];  /* 任务栈 */
staticuint64_t AppTaskStartStk[512/8];     /* 任务栈 */
将任务栈定义成uint64_t类型可以保证任务栈是8字节对齐的,8字节对齐的含义就是数组的首地址对8求余等于0。如果不做8字节对齐的话,部分C语言库函数,浮点运算和uint64_t类型数据运算会出问题。
系统栈大小分配:
9.13.png
RTX初始化:
  1. /*
    / Z# @0 P+ [" J, {
  2. *********************************************************************************************************
    / c  Q: b+ F. [+ L1 c
  3. *    函 数 名: main
    8 H$ K. V1 I1 ?' _2 j  L
  4. *    功能说明: 标准c程序入口。
    1 c+ i' w& g5 ]* M& p
  5. *    形    参: 无
    5 I) F6 z( j0 m4 ]  O) ^# j9 ~7 k2 c
  6. *    返 回 值: 无
    5 w3 l# y. H  D4 r$ K+ h$ ^
  7. *********************************************************************************************************
    # ^1 p0 d/ I9 ^9 Q$ H% H
  8. */
    1 L$ K4 j: l% F0 d! r
  9. int main (void)3 x# }1 L$ [% k2 \9 c' N4 {) `
  10. {   
    % b+ W7 Z5 M5 y5 S  }* F6 A" i' I; h
  11.      /* 初始化外设 */
    # e6 p( F) U; e3 G
  12.      bsp_Init();3 l' P- E- V4 h) h, ~
  13.     , \6 Z  s* V) k+ F' N$ T: L
  14.      /* 创建启动任务 */
      N. X/ q) a2 B1 c' a. B. s- F
  15.      os_sys_init_user (AppTaskStart,             /* 任务函数 */
    1 `8 @, k! }1 H$ H- t$ o
  16.                        4,                        /* 任务优先级 */
    1 P% z' M" x9 A5 z: x1 T
  17.                        &AppTaskStartStk,         /* 任务栈 */
    ; Z! d  r8 L, S4 |, v2 N, {
  18.                        sizeof(AppTaskStartStk)); /* 任务栈大小,单位字节数 */
    1 T* Z  @! ^; Q. s) u' b" _
  19.      while(1);
    3 [; c& t, s! c* R8 t' a
  20. }
复制代码

; X4 u) S! _8 L7 a& J# I( E
RTX任务创建:
  1. /*& R  ^. K* ]& b9 Q* l' L* |! D3 G
  2. *********************************************************************************************************1 [9 Q9 ]2 _% y' P, Q* E+ a4 m
  3. *    函 数 名: AppTaskCreate7 ^; o" U  u! P- o* C
  4. *    功能说明: 创建应用任务
    & m7 Q( _3 q2 A# i1 R: d
  5. *    形    参: 无( v. X' Q9 v# ]! t
  6. *    返 回 值: 无
    2 j2 v, ?8 r, `( M9 L/ r' l: \
  7. *********************************************************************************************************
    ' e7 N% ?5 `; x8 k3 K* l1 A
  8. */; ?$ e1 Q" C% g* a% J
  9. static void AppTaskCreate (void)+ U  h8 g! k: Y
  10. {& v% L2 ?3 v# ]8 p8 ?+ _% Z
  11.      HandleTaskUserIF = os_tsk_create_user(AppTaskUserIF,             /* 任务函数 */
    $ K' |, }$ p5 {0 O# P* L
  12.                                            1,                         /* 任务优先级 */
    1 ~, z2 N0 n- J  l6 P* ?1 P5 c2 r- p
  13.                                            &AppTaskUserIFStk,         /* 任务栈 */
    9 Q- i2 `7 X# E3 D+ R/ i: U3 W- V
  14.                                            sizeof(AppTaskUserIFStk)); /* 任务栈大小,单位字节数 */2 X. C  _8 s/ {
  15.     8 ^# e+ `$ v& ~/ Q  S; h
  16.      HandleTaskLED = os_tsk_create_user(AppTaskLED,              /* 任务函数 */4 _! g, U6 P: k8 A
  17.                                         2,                       /* 任务优先级 */6 B& Z) N* L' W6 z+ {, f* ?
  18.                                         &AppTaskLEDStk,          /* 任务栈 */, k1 i% \9 \' l  N8 `0 V
  19.                                         sizeof(AppTaskLEDStk));  /* 任务栈大小,单位字节数 */
    / A! ^7 T! l- k2 Z7 r- z" `1 g
  20.     ( U6 o4 ~  u: a& R* k: \0 x" `* {
  21.      HandleTaskMsgPro = os_tsk_create_user(AppTaskMsgPro,             /* 任务函数 */
    3 T; ~; T+ r+ j" `) g
  22.                                            3,                         /* 任务优先级 */
    4 R! M3 t/ c4 @: `, m5 h
  23.                                            &AppTaskMsgProStk,         /* 任务栈 */
    . j! ?' k2 v2 [% @6 M, L: |  z
  24.                                            sizeof(AppTaskMsgProStk)); /* 任务栈大小,单位字节数 */# m4 z8 C1 U0 Z4 R
  25. }
复制代码

9 U" i( _% y& j/ r( N- D+ q
四个RTX任务的实现:
  1. /*( j3 s" P8 c$ Z; |; h
  2. *********************************************************************************************************% @: X% D" p- y! R" U) C
  3. *    函 数 名: AppTaskUserIF
    - U) ^# X7 o2 D( F
  4. *    功能说明: 按键消息处理     9 u" ~4 V4 w" [
  5. *    形    参: 无
    7 d; Z/ D+ a% |0 h+ ~; H. R7 `/ X  J
  6. *    返 回 值: 无* R9 V% U+ M8 ]! `0 C$ a0 b
  7. *   优 先 级: 1  (数值越小优先级越低,这个跟uCOS相反). `! u/ U1 Z4 D9 i
  8. *********************************************************************************************************/ C3 D- V: T  r* D9 b0 Q6 ^0 _4 N
  9. */1 A) }; e- X* u* R; E
  10. __task void AppTaskUserIF(void)6 ~5 M" q" n% Y9 A" W+ _+ i
  11. {( j$ T9 s- s' C5 v- ?" c1 I! y
  12.      uint8_t ucKeyCode;. T0 h  y1 B! f! r7 i
  13. ' r& k$ [3 B  H( h) ], U' ]
  14.     while(1)- N2 ^: J: o$ ^1 @; S- E" W7 R
  15.     {+ d* y8 V5 M$ A; H
  16.          ucKeyCode = bsp_GetKey();$ w) o! A/ r0 c1 @
  17.         
    ( I0 w1 n8 x, x- F: Y
  18.          if (ucKeyCode != KEY_NONE)
    ( ]: ], v) B# v
  19.          {
    ( Y3 y6 J" r  L
  20.               switch (ucKeyCode)8 N9 q0 V. R6 O% V- F8 n
  21.               {
    ; V% ]+ Z/ L1 p& C
  22.                    /* K1键按下,打印调试说明 */
    8 H9 N: V( W1 }: H1 U
  23.                    case KEY_DOWN_K1:
    - e5 f6 h' N/ ]$ L
  24.                        printf("K1键按下,使用MDK中自带的RTX调试组件,请务必使用MDK4.74版本进行调试\r\n");
    % {  y/ Q/ y5 g- F# d" p$ w
  25.                        break;  / J4 N& F) K- G. u" l
  26. " Z" j$ Y! \5 N* o; G# w; a
  27.                    /* K2键按下,非特权级模式下设置NVIC分组,从而导致系统进入到硬件异常HardFault_Handler */
    ' M/ l" M! v; j
  28.                    case KEY_DOWN_K2:
    6 b' l; }! V- T
  29.                        printf("K2键按下,非特权级模式下设置NVIC分组,从而导致系统进入到硬件异常
    7 R4 p: N/ e# p, J- Z
  30. HardFault_Handler\r\n");
    2 r: ?, V9 }+ }4 H, I, H: f- A4 A
  31.                        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);' [* h! x: t& h( Q1 e; {
  32.                        break;
    # T6 f) Y% S' v7 Q
  33. - \( u6 ?" l8 k3 F
  34.                    /* 其他的键值不处理 */
    % I- i) A9 f/ q, Q
  35.                    default:                    
    4 z# z% j/ h  w" l% a8 T2 @& ]
  36.                        break;
    : m# S4 q1 f( q$ z5 P# V  U
  37.               }
    + O2 f/ I* J; V  k
  38.          }
    * J9 `& ~  f  E
  39.         
    - v+ a* i9 @: y. l0 [! C
  40.          os_dly_wait(20);
    6 y) l$ a* O" p2 M' s" D1 q- z. s( m
  41.      }
    & B. W/ I9 Q+ E) Y$ T9 k
  42. }$ z& N5 L5 W0 F" L
  43. ' e" |" i+ Y, r7 r  Q+ S+ K8 N" ]
  44. /*
    ! ?3 C* i; o5 ?: O# p( z/ O2 y0 t; X
  45. *********************************************************************************************************% A/ q0 V& k5 |' r
  46. *    函 数 名: AppTaskLED
    7 m3 f) {* c! k0 x8 R" I
  47. *    功能说明: LED闪烁。
    2 E. p0 @' ~3 w$ N3 t7 C9 _, P
  48. *    形    参: 无7 `: q: Y0 p" A/ H  ^4 S
  49. *    返 回 值: 无
    $ k+ i0 h* y) ^- B9 }
  50. *   优 先 级: 2
    5 u8 |6 D; j7 u" \% y
  51. *********************************************************************************************************0 @7 S1 m/ v, O5 ?/ \
  52. */
    % V$ A0 i% P+ d$ P. D) H
  53. __task void AppTaskLED(void)/ j7 k+ _/ P. K$ j/ X
  54. {( K' t4 ^# O4 [
  55.      const uint16_t usFrequency = 200; /* 延迟周期 */
    6 R6 f+ w4 O- ]% Y
  56.     9 D0 U6 c/ b( x- V
  57.      /* 设置延迟周期 */. D, @" D3 f( _9 Y) r& B1 G
  58.      os_itv_set(usFrequency);
    * L# k5 m# A: S2 l# a, Y3 @  K; x
  59.    
    / u( r9 s8 Z( C0 n1 ~
  60.     while(1)
    2 e3 v4 V% X) A- Y8 Q
  61.     {7 z7 s3 Q- l% h
  62.          bsp_LedToggle(2);/ e! f$ V. F1 |. Y& F8 \7 d
  63.          bsp_LedToggle(3);3 |/ K4 ~* R0 ^# J- M& H" u6 ?, X

  64. 0 D# u$ d6 M" s; j
  65.          /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/
    - y9 U! K/ D/ }9 A0 j0 B
  66.          os_itv_wait();: f8 g/ L3 p+ R: f' X+ Y
  67.     }+ i& {3 M; V; m: Z
  68. }0 ?3 C' r6 m1 f, @$ O- g

  69. - z4 ], _1 G$ }; H7 F" Z4 S
  70. /*! C0 n: ]! \4 J& x8 W6 C
  71. *********************************************************************************************************
    : v" e. c2 a/ F1 h
  72. *    函 数 名: AppTaskMsgPro7 S% e8 h* Q' ?3 T5 q
  73. *    功能说明: 消息处理,这里用作LED闪烁
    0 y7 ~! N; @% Q3 `" L
  74. *    形    参: 无% T0 B  t- ^7 h" ]6 U$ n" D
  75. *    返 回 值: 无1 q& Z' d' N- V: d4 w3 D+ v
  76. *   优 先 级: 3
    8 K1 _$ a, q  g# J2 O
  77. *********************************************************************************************************' w4 [. ?0 s) [# @
  78. */
    9 k0 C& h1 t( B. ]0 h. G
  79. __task void AppTaskMsgPro(void)
    6 c6 X2 K; J* T8 J9 L" t
  80. {
      W8 W' d# R/ u6 A3 j# |$ F8 s' f
  81.      const uint16_t usFrequency = 500; /* 延迟周期 */3 I+ T4 y5 e; w0 q0 h  a: B' X: |
  82.    
    + m) P' s- G4 \# r9 ?' P/ e
  83.      /* 设置延迟周期 */. T( X6 u& H7 J
  84.      os_itv_set(usFrequency);, ?- k3 n+ L7 Q9 @
  85.     * y" {$ K9 N3 E7 F& Y) q! s
  86.      while(1)
    $ D* p4 v* Q& C* u" a
  87.      {) k+ ~# f: N1 l7 s; v
  88.          bsp_LedToggle(1);" f9 b: i1 ]1 u  y8 S/ J
  89.          bsp_LedToggle(4);
    ' f+ Y& O. R2 B( u$ n
  90. & J3 s+ x# V) g8 R
  91.          /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/$ j. E/ P( G2 G, E) i
  92.          os_itv_wait();
    8 h/ W* {5 p0 y( j
  93.      }   
    6 [1 z) e# J# ]( k, [3 V1 w
  94. }
    : _- C3 b  F2 M5 n! E
  95. ( _* E: G5 r( r. R( E- m
  96. /*
    # Z# R* d) S- l* B3 B: S4 o
  97. *********************************************************************************************************2 \+ y. Z& ~9 C: c, F# m. i: B3 u
  98. *    函 数 名: AppTaskStart
    # t) b: g- g1 i/ C7 S
  99. *    功能说明: 启动任务,也就是最高优先级任务。这里实现按键扫描。- G; U2 y6 D# m7 l/ F& Z
  100. *    形    参: 无
    7 e1 I5 r7 f2 \9 L; d
  101. *    返 回 值: 无
    ; I8 z( b8 w' B" n2 s7 H" {
  102. *   优 先 级: 4
    6 ~: B: W8 L. ]$ s8 C# Z
  103. *********************************************************************************************************0 w' Z% R5 |' a
  104. */9 `( n0 H% N0 G( G# i
  105. __task void AppTaskStart(void)
    & E+ a7 M# [* x* n) j( k
  106. {
    6 R4 b) N8 P, V
  107.      /* 创建任务 */* j% G4 L/ @9 v3 [) E0 F4 T- j
  108.      AppTaskCreate();4 I. I" M1 }. k8 `# d8 D+ F- V
  109.    
    * _4 d6 G7 R7 M0 ?8 l4 n& F% O2 M5 O
  110.     while(1); ]. k  i8 S& O, t4 F
  111.     {. Y7 p! y; S$ U! `2 s; ?9 @
  112.          /* 按键扫描 */
    3 M3 p2 [) G- M) H, A" d
  113.          bsp_KeyScan();6 \& P5 V: Z$ S; i
  114.         os_dly_wait(10);
    , C- c* K  C" s- R" j
  115.     }
    : P* @7 K2 r. f/ g# i. w
  116. }
复制代码

1 z6 L0 T6 q' z: c& g. D* d( f0 G) |/ U% h+ l% o! A
baiyongbin2009 回答时间:2016-1-23 18:00:25
9.3.2     STM32F407开发板实验
配套例子:
    V5-404_RTX实验_任务运行在用户模式(非特权级)
实验目的:
    1. 学习RTX的任务运行在用户模式的设置方法
实验内容:
    1. K1按键按下,串口打印。
    2. K2键按下,设置NVIC分组,从而导致系统进入到硬件异常HardFault_Handler。
    3. 各个任务实现的功能如下:
      AppTaskUserIF任务  :按键消息处理。
      AppTaskLED任务     :LED闪烁。
      AppTaskMsgPro任务 :消息处理,这里用作LED闪烁。
      AppTaskStart任务    :启动任务,也是最高优先级任务,这里实现按键扫描。
RTX配置:
    RTX配置向导详情如下:
9.14.png
                              
Task Configuration
(1Number of concurrent running tasks
        允许创建4个任务,实际创建了如下四个任务
                     AppTaskUserIF任务   :按键消息处理。
                     AppTaskLED任务     :LED闪烁。
                     AppTaskMsgPro任务 :消息处理,这里是用作LED闪烁。
                     AppTaskStart任务    :启动任务,也是最高优先级任务,这里实现按键扫描。
(2)Number of tasks with user-provided stack
    创建的4个任务都是采用自定义堆栈方式。
(3) Run in privileged mode
    设置RTX任务运行在非特权级,即用户模式。
3 Y0 {3 `- K" p. t/ V+ G) Z0 V7 ~
RTX任务调试信息:
9.15.png
程序设计:
任务栈大小分配:
    staticuint64_t AppTaskUserIFStk[512/8];   /* 任务栈 */
    staticuint64_t AppTaskLEDStk[256/8];      /* 任务栈 */
    staticuint64_t AppTaskMsgProStk[512/8];  /* 任务栈 */
    staticuint64_t AppTaskStartStk[512/8];     /* 任务栈 */
    将任务栈定义成uint64_t类型可以保证任务栈是8字节对齐的,8字节对齐的含义就是数组的首地址对8求余等于0。如果不做8字节对齐的话,部分C语言库函数,浮点运算和uint64_t类型数据运算会出问题。
系统栈大小分配:
9.16.png
RTX初始化:
  1. /*
    , x) c. U* D3 q
  2. *********************************************************************************************************. U4 {2 E) }5 ], a1 Y) W" B
  3. *    函 数 名: main
    3 h0 T* Z+ j& R2 M6 I3 E$ ^5 M
  4. *    功能说明: 标准c程序入口。
    0 X$ @/ V2 P- Y, S0 z8 V' ]
  5. *    形    参: 无
    " J3 F5 G6 u7 X2 H& U4 R5 ?
  6. *    返 回 值: 无2 Y: N8 V" O) V) j# ^
  7. *********************************************************************************************************
    * p5 h4 y% a+ A$ s. a8 y4 u
  8. */( v9 F) i: q. ~8 M9 `# I7 |
  9. int main (void)
    7 M( K/ A" n. K* d0 R0 u
  10. {   ; B+ x9 D5 ]7 J6 I" Q; C
  11.      /* 初始化外设 */
    # L5 F! U3 S; [
  12.      bsp_Init();& y5 o; b3 R1 C( Q  F( ^
  13.     6 b) \% _# D  M0 k) Q5 e
  14.      /* 创建启动任务 */
    2 Z$ d  A! O2 P+ l% V+ r% A
  15.      os_sys_init_user (AppTaskStart,             /* 任务函数 */
    3 p3 q3 H4 ~. D. X  J0 l* ^
  16.                        4,                        /* 任务优先级 */
    + P. q8 w* ]8 W! `) w- G
  17.                        &AppTaskStartStk,         /* 任务栈 */2 ^' O1 h. R2 o5 {* A
  18.                        sizeof(AppTaskStartStk)); /* 任务栈大小,单位字节数 */
    * _# t5 H' O' S2 r3 A
  19.      while(1);
    + H/ N% h. l$ p3 q2 C+ D* q' F' y" f
  20. }
复制代码
( I$ R# a  d  _5 t* W8 _4 O! y
RTX任务创建:
  1. /*  k6 r3 `) g4 Y' ~  ~* y* F
  2. *********************************************************************************************************- [8 `7 Y! u1 |8 I% T  N
  3. *    函 数 名: AppTaskCreate
      z3 }; H9 u! B9 A. w
  4. *    功能说明: 创建应用任务& \$ K. \" e( \+ p. U& G& n4 n
  5. *    形    参: 无/ A" z- z+ L- `+ e( J
  6. *    返 回 值: 无0 k5 W* c5 A1 y9 o0 v- C
  7. *********************************************************************************************************2 L; ~3 [6 }& D, T% D& w
  8. */1 H9 ]4 n- F4 Q2 y. c- k3 k, }# ^3 \
  9. static void AppTaskCreate (void)8 ~8 }  m3 l' }: x8 a- n' L* n
  10. {
    ! h9 o  k/ T& t* Q( @
  11.      HandleTaskUserIF = os_tsk_create_user(AppTaskUserIF,             /* 任务函数 */! k5 b$ T0 c& [# u6 g
  12.                                            1,                         /* 任务优先级 */
    ' i2 s7 C3 T6 j4 Z, d
  13.                                            &AppTaskUserIFStk,         /* 任务栈 */
    , {  J/ s+ H" e  }  q
  14.                                            sizeof(AppTaskUserIFStk)); /* 任务栈大小,单位字节数 */0 a* _' E7 C$ C9 d, H. o8 }
  15.    
    - [# k0 q4 v8 b
  16.      HandleTaskLED = os_tsk_create_user(AppTaskLED,              /* 任务函数 */
    7 E8 G# }4 y0 h. Q
  17.                                         2,                       /* 任务优先级 */! s' }( `5 H+ K+ F- a
  18.                                         &AppTaskLEDStk,          /* 任务栈 */
    ' ~! R' m& l( q- i" U8 S
  19.                                         sizeof(AppTaskLEDStk));  /* 任务栈大小,单位字节数 */# T3 p3 Z  C. e2 U" p+ C( }
  20.    
    / v, W+ p' {1 I# |1 i0 K, V/ C
  21.      HandleTaskMsgPro = os_tsk_create_user(AppTaskMsgPro,             /* 任务函数 */, _+ M7 c4 j/ R/ h
  22.                                            3,                         /* 任务优先级 */
    * s3 Z  E5 Y" X
  23.                                            &AppTaskMsgProStk,         /* 任务栈 */2 ~) I5 Z$ c, k
  24.                                            sizeof(AppTaskMsgProStk)); /* 任务栈大小,单位字节数 */. \; ]( t4 u6 _7 E
  25. }
复制代码
4 w( H6 u4 }  F2 \8 G8 \  j
四个RTX任务的实现:
  1. /*: n' m1 K5 a1 G6 \6 O- W+ a3 ?
  2. *********************************************************************************************************
    * Q  n: w8 k0 w8 U( _# j  G7 f
  3. *    函 数 名: AppTaskUserIF& H6 M9 Z4 h8 D: Q1 O
  4. *    功能说明: 按键消息处理     
    5 T. s) i8 k4 Q; I1 _
  5. *    形    参: 无) S# m( n! z( u( ^! I0 T0 ^6 l$ D
  6. *    返 回 值: 无; O; ?2 p+ E/ \& ?4 O3 D, j- H
  7. *   优 先 级: 1  (数值越小优先级越低,这个跟uCOS相反)8 A' x* [+ i9 I% m9 \
  8. *********************************************************************************************************% a' z' N- D5 L
  9. */4 Z, x/ j& |$ ^' a: M
  10. __task void AppTaskUserIF(void): V) e5 \* ^4 b
  11. {
    # ~- }& A1 k; Z6 q2 J- k
  12.      uint8_t ucKeyCode;
    0 B; g- ]' W0 t. a5 k2 G9 Q

  13. / U6 o5 |6 z8 ]% x- |
  14.     while(1)
    ! B1 `  ?. ?- Y* s
  15.     {; P  t9 |6 U' C1 a0 a
  16.          ucKeyCode = bsp_GetKey();2 S6 Q$ \$ m' _* b+ X
  17.         ( k$ j8 C. ]9 m: v3 G7 @
  18.          if (ucKeyCode != KEY_NONE)
    ; J# ~& S. h% C9 r, _/ I& B
  19.          {
    ; y) ^& R+ C6 n0 g4 U9 {
  20.               switch (ucKeyCode)6 J* R* ]# G8 R$ y+ o
  21.               {
    - g9 j8 `5 {" o2 a/ \3 F
  22.                    /* K1键按下,打印调试说明 */
    * ], G) W. m" ~
  23.                    case KEY_DOWN_K1:2 t0 x. |9 r/ C/ L+ i* @
  24.                        printf("K1键按下,使用MDK中自带的RTX调试组件,请务必使用MDK4.74版本进行调试\r\n");* y0 I' }3 V. Z$ F# S
  25.                        break;  
    9 R) Z4 k& {% T1 G1 b0 ~) X
  26. % x9 |) O0 W* e) M% }! r' R, e
  27.                    /* K2键按下,非特权级模式下设置NVIC分组,从而导致系统进入到硬件异常HardFault_Handler */' A; f* M" q- w+ k' V& n
  28.                    case KEY_DOWN_K2:
    1 J2 o& s  }7 E; {! l. [
  29.                        printf("K2键按下,非特权级模式下设置NVIC分组,从而导致系统进入到硬件异常, P' g7 k. |; R" ^5 C
  30. HardFault_Handler\r\n");
    / E9 F2 }! X+ o! @2 v  P7 f1 V! L2 `
  31.                        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);: L! [- C$ T- O  _7 v: l4 q+ m( a- L
  32.                        break;
    2 N5 h/ }6 \- p' N; _: {' }: f
  33. " P3 u8 }5 `, V/ C$ B! a
  34.                    /* 其他的键值不处理 */! R* A+ Y- Z) T! C0 N& A
  35.                    default:                    2 `( ~2 r" i  l# `6 G) T
  36.                        break;/ z9 W9 j! [# E; }8 E
  37.               }8 n& N/ W1 }$ s* t1 r
  38.          }
    ' @0 `% n$ s- y" p4 F( ^/ @% G
  39.         ) c$ y; n0 Q6 P
  40.          os_dly_wait(20);
    6 C- A4 `' I' m, l
  41.      }
    ) C5 [. [9 a8 |7 v5 S
  42. }
    ( L% Z; P9 S8 ?! u; _9 q
  43. " S( ?& n# G7 Q! o, k, \- H& z
  44. /*( i; X+ E( ?$ U, v& c7 p2 a4 Y
  45. *********************************************************************************************************
    ! u2 R! ^0 P6 G* U2 V2 P: h$ z! v
  46. *    函 数 名: AppTaskLED
    2 _: @1 N7 s9 c9 s8 _% h
  47. *    功能说明: LED闪烁。
    4 x) P! m& I/ B2 d
  48. *    形    参: 无
    / Y/ Y+ _; L/ |4 J8 ^& s$ z
  49. *    返 回 值: 无
    7 W& Z9 E1 W7 V0 |' P% F7 @
  50. *   优 先 级: 2 $ G* H+ q0 S0 C9 m  Z
  51. ********************************************************************************************************** S3 `6 \. W# R" Z, B4 A; _
  52. */
    " N/ g3 f, {: S0 A
  53. __task void AppTaskLED(void)
      i2 {; x3 H# y, P
  54. {
    3 r, F. p# z7 m" Y
  55.      const uint16_t usFrequency = 200; /* 延迟周期 */8 d9 n) ?% w+ K/ O1 U2 T
  56.    
    , i, {& i* {4 Y0 @; b
  57.      /* 设置延迟周期 */
    & w) x1 \* N6 H! e  L  d* U% j
  58.      os_itv_set(usFrequency);# W3 \1 m0 y- n2 }0 A: w
  59.     # V: c; _) b* I; n7 |4 Q& M
  60.     while(1)
    ( P* S! _7 M% M: K3 P  x7 b4 H
  61.     {' W/ q0 Z6 q2 Z$ w3 [' S
  62.          bsp_LedToggle(2);2 o+ }- x' G/ U" c; Y
  63.          bsp_LedToggle(3);& o+ s" x" ]3 d9 N% t
  64. 0 x% \0 C: ]: ]" {3 w. N6 o
  65.          /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/
    , e" U. |6 c& H- U4 `+ A3 ^
  66.           os_itv_wait();
    7 B7 _6 D7 d: \/ J
  67.     }/ i6 D! F& o' Q) e- B
  68. }
    # z# o" h  ~4 p0 b1 v* I6 }

  69. # N1 q4 m* Y8 O$ F5 \4 I$ h) K
  70. /*
    ; B! x) a( y8 X: L
  71. *********************************************************************************************************5 l) t- K1 `2 l# t
  72. *    函 数 名: AppTaskMsgPro
    , U; z6 N7 i+ Y) S
  73. *    功能说明: 消息处理,这里用作LED闪烁4 G. V- R( A/ i( r; I4 }
  74. *    形    参: 无
    ! Q$ {$ g# |1 j% B
  75. *    返 回 值: 无* w* e- Q' H$ k
  76. *   优 先 级: 3 ( G: h  ?, I/ O! y8 L. ^
  77. *********************************************************************************************************. u$ M9 [4 m" e
  78. */
    ) j+ Z6 d* `7 t
  79. __task void AppTaskMsgPro(void)
    8 _4 w' ]- }2 J; S& W0 r# R& _
  80. {
    - W( ^6 \* `6 }9 ~$ P, c3 T
  81.      const uint16_t usFrequency = 500; /* 延迟周期 */
    2 A3 Q, m- S% f9 _
  82.     " N* p" l  j! [3 u
  83.      /* 设置延迟周期 */: c& K' ]$ @; p: Z
  84.      os_itv_set(usFrequency);
    8 D  x) {2 w$ {: D' t
  85.    
    / o; M4 Q* [' Y! w
  86.      while(1)6 |+ U9 `3 P1 a. M% h% e
  87.      {
      X. S+ K5 H  I& h& _( r7 Y
  88.          bsp_LedToggle(1);. [7 y$ S- @3 j& T3 ?
  89.          bsp_LedToggle(4);7 ~9 \$ ^. Z- {
  90. / R) Y7 b8 J, a: X
  91.          /* os_itv_wait是绝对延迟,os_dly_wait是相对延迟。*/
    ' g- l" {. e1 o2 {; I1 j8 y
  92.          os_itv_wait();
    * w& i: R- ?: R  D+ h1 ~- L, Y
  93.      }   
    - \+ H; Z; ]; g( t- o
  94. }& A/ C4 Z- ~0 w
  95. ' A$ _6 E0 W+ W# f5 u# @8 z' H
  96. /*$ y, s; j& v- K  W, ]+ j
  97. *********************************************************************************************************2 l. R( c; u9 a; n0 u0 S: {
  98. *    函 数 名: AppTaskStart7 a8 R' }, {: [
  99. *    功能说明: 启动任务,也就是最高优先级任务。这里实现按键扫描。% T: o4 p) K/ }( N1 U
  100. *    形    参: 无
    ) A+ j9 G& M6 U8 Q7 F# h! ?
  101. *    返 回 值: 无! Z( L) {1 g$ l  w  {
  102. *   优 先 级: 4
    * P, S' M6 Y( X' F3 b
  103. *********************************************************************************************************4 O) v4 B$ u2 E% O
  104. */
    % Z  |8 x" Z4 Y  H8 a( t
  105. __task void AppTaskStart(void)4 N, a0 O5 Y' c# ]4 t
  106. {
    ( D3 b& I/ f) l/ Y- P( [
  107.      /* 创建任务 */1 \. l5 S9 q6 v: p9 h1 V
  108.      AppTaskCreate();6 G0 m. Z: u" z( q/ f
  109.     7 f/ s2 R6 c5 c
  110.     while(1)& s7 o$ }4 @  Q; h; {9 h# e+ t, F
  111.     {( g' k7 b" [/ d7 p) f
  112.          /* 按键扫描 */. d6 C9 a2 y9 [' u" |
  113.          bsp_KeyScan();& G, b8 c/ \% f, V1 n. k, o
  114.         os_dly_wait(10);
    + ?' n5 G8 S* P/ |' z% H$ R6 C
  115.     }
    + O9 n$ Z: K1 n1 [; L" F3 L
  116. }
复制代码
$ p3 D. U; g) i  q- x
9 n; z' _( t" H2 c
baiyongbin2009 回答时间:2016-1-23 18:01:08
9.4  总结
    对于初学者来说,容易在特权级和非特权级的地方出错。如果以前没有这方面基础的话,理解的时候还稍困难些,不急,慢慢来,完全的理解也需要一个循序渐进的过程。

4 G1 ]0 {) M! J! I2 z* `* h8 |: }# G
回到未来~ 回答时间:2016-1-24 20:56:54
感谢分析,详细的帖子!虽然没用过安福来……

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版