uc-sdk
 All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
port.c
Go to the documentation of this file.
1 /*
2  FreeRTOS V7.5.3 - Copyright (C) 2013 Real Time Engineers Ltd.
3  All rights reserved
4 
5  VISIT http://www.FreeRTOS.org TO ENSURE YOU ARE USING THE LATEST VERSION.
6 
7  ***************************************************************************
8  * *
9  * FreeRTOS provides completely free yet professionally developed, *
10  * robust, strictly quality controlled, supported, and cross *
11  * platform software that has become a de facto standard. *
12  * *
13  * Help yourself get started quickly and support the FreeRTOS *
14  * project by purchasing a FreeRTOS tutorial book, reference *
15  * manual, or both from: http://www.FreeRTOS.org/Documentation *
16  * *
17  * Thank you! *
18  * *
19  ***************************************************************************
20 
21  This file is part of the FreeRTOS distribution.
22 
23  FreeRTOS is free software; you can redistribute it and/or modify it under
24  the terms of the GNU General Public License (version 2) as published by the
25  Free Software Foundation >>!AND MODIFIED BY!<< the FreeRTOS exception.
26 
27  >>! NOTE: The modification to the GPL is included to allow you to distribute
28  >>! a combined work that includes FreeRTOS without being obliged to provide
29  >>! the source code for proprietary components outside of the FreeRTOS
30  >>! kernel.
31 
32  FreeRTOS is distributed in the hope that it will be useful, but WITHOUT ANY
33  WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
34  FOR A PARTICULAR PURPOSE. Full license text is available from the following
35  link: http://www.freertos.org/a00114.html
36 
37  1 tab == 4 spaces!
38 
39  ***************************************************************************
40  * *
41  * Having a problem? Start by reading the FAQ "My application does *
42  * not run, what could be wrong?" *
43  * *
44  * http://www.FreeRTOS.org/FAQHelp.html *
45  * *
46  ***************************************************************************
47 
48  http://www.FreeRTOS.org - Documentation, books, training, latest versions,
49  license and Real Time Engineers Ltd. contact details.
50 
51  http://www.FreeRTOS.org/plus - A selection of FreeRTOS ecosystem products,
52  including FreeRTOS+Trace - an indispensable productivity tool, a DOS
53  compatible FAT file system, and our tiny thread aware UDP/IP stack.
54 
55  http://www.OpenRTOS.com - Real Time Engineers ltd license FreeRTOS to High
56  Integrity Systems to sell under the OpenRTOS brand. Low cost OpenRTOS
57  licenses offer ticketed support, indemnification and middleware.
58 
59  http://www.SafeRTOS.com - High Integrity Systems also provide a safety
60  engineered and independently SIL3 certified version for use in safety and
61  mission critical applications that require provable dependability.
62 
63  1 tab == 4 spaces!
64 */
65 
66 /*-----------------------------------------------------------
67  * Implementation of functions defined in portable.h for the ARM CM3 port.
68  *----------------------------------------------------------*/
69 
70 /* Scheduler includes. */
71 #include "FreeRTOS.h"
72 #include "task.h"
73 
74 /* For backward compatibility, ensure configKERNEL_INTERRUPT_PRIORITY is
75 defined. The value should also ensure backward compatibility.
76 FreeRTOS.org versions prior to V4.4.0 did not include this definition. */
77 #ifndef configKERNEL_INTERRUPT_PRIORITY
78  #define configKERNEL_INTERRUPT_PRIORITY 255
79 #endif
80 
81 #ifndef configSYSTICK_CLOCK_HZ
82  #define configSYSTICK_CLOCK_HZ configCPU_CLOCK_HZ
83 #endif
84 
85 /* Constants required to manipulate the core. Registers first... */
86 #define portNVIC_SYSTICK_CTRL_REG ( * ( ( volatile unsigned long * ) 0xe000e010 ) )
87 #define portNVIC_SYSTICK_LOAD_REG ( * ( ( volatile unsigned long * ) 0xe000e014 ) )
88 #define portNVIC_SYSTICK_CURRENT_VALUE_REG ( * ( ( volatile unsigned long * ) 0xe000e018 ) )
89 #define portNVIC_SYSPRI2_REG ( * ( ( volatile unsigned long * ) 0xe000ed20 ) )
90 /* ...then bits in the registers. */
91 #define portNVIC_SYSTICK_CLK_BIT ( 1UL << 2UL )
92 #define portNVIC_SYSTICK_INT_BIT ( 1UL << 1UL )
93 #define portNVIC_SYSTICK_ENABLE_BIT ( 1UL << 0UL )
94 #define portNVIC_SYSTICK_COUNT_FLAG_BIT ( 1UL << 16UL )
95 #define portNVIC_PENDSVCLEAR_BIT ( 1UL << 27UL )
96 #define portNVIC_PEND_SYSTICK_CLEAR_BIT ( 1UL << 25UL )
97 
98 #define portNVIC_PENDSV_PRI ( ( ( unsigned long ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )
99 #define portNVIC_SYSTICK_PRI ( ( ( unsigned long ) configKERNEL_INTERRUPT_PRIORITY ) << 24UL )
100 
101 /* Constants required to check the validity of an interrupt priority. */
102 #define portFIRST_USER_INTERRUPT_NUMBER ( 16 )
103 #define portNVIC_IP_REGISTERS_OFFSET_16 ( 0xE000E3F0 )
104 #define portAIRCR_REG ( * ( ( volatile unsigned long * ) 0xE000ED0C ) )
105 #define portMAX_8_BIT_VALUE ( ( unsigned char ) 0xff )
106 #define portTOP_BIT_OF_BYTE ( ( unsigned char ) 0x80 )
107 #define portMAX_PRIGROUP_BITS ( ( unsigned char ) 7 )
108 #define portPRIORITY_GROUP_MASK ( 0x07UL << 8UL )
109 #define portPRIGROUP_SHIFT ( 8UL )
110 
111 /* Constants required to set up the initial stack. */
112 #define portINITIAL_XPSR ( 0x01000000UL )
113 
114 /* The systick is a 24-bit counter. */
115 #define portMAX_24_BIT_NUMBER ( 0xffffffUL )
116 
117 /* A fiddle factor to estimate the number of SysTick counts that would have
118 occurred while the SysTick counter is stopped during tickless idle
119 calculations. */
120 #define portMISSED_COUNTS_FACTOR ( 45UL )
121 
122 /* Let the user override the pre-loading of the initial LR with the address of
123 prvTaskExitError() in case is messes up unwinding of the stack in the
124 debugger. */
125 #ifdef configTASK_RETURN_ADDRESS
126  #define portTASK_RETURN_ADDRESS configTASK_RETURN_ADDRESS
127 #else
128  #define portTASK_RETURN_ADDRESS prvTaskExitError
129 #endif
130 
131 /* Each task maintains its own interrupt status in the critical nesting
132 variable. */
133 static unsigned portBASE_TYPE uxCriticalNesting = 0xaaaaaaaa;
134 
135 /*
136  * Setup the timer to generate the tick interrupts. The implementation in this
137  * file is weak to allow application writers to change the timer used to
138  * generate the tick interrupt.
139  */
140 void vPortSetupTimerInterrupt( void );
141 
142 /*
143  * Exception handlers.
144  */
145 void xPortPendSVHandler( void ) __attribute__ (( naked ));
146 void xPortSysTickHandler( void );
147 void vPortSVCHandler( void ) __attribute__ (( naked ));
148 
149 /*
150  * Start first task is a separate function so it can be tested in isolation.
151  */
152 static void prvPortStartFirstTask( void ) __attribute__ (( naked ));
153 
154 /*
155  * Used to catch tasks that attempt to return from their implementing function.
156  */
157 static void prvTaskExitError( void );
158 
159 /*-----------------------------------------------------------*/
160 
161 /*
162  * The number of SysTick increments that make up one tick period.
163  */
164 #if configUSE_TICKLESS_IDLE == 1
165  static unsigned long ulTimerCountsForOneTick = 0;
166 #endif /* configUSE_TICKLESS_IDLE */
167 
168 /*
169  * The maximum number of tick periods that can be suppressed is limited by the
170  * 24 bit resolution of the SysTick timer.
171  */
172 #if configUSE_TICKLESS_IDLE == 1
173  static unsigned long xMaximumPossibleSuppressedTicks = 0;
174 #endif /* configUSE_TICKLESS_IDLE */
175 
176 /*
177  * Compensate for the CPU cycles that pass while the SysTick is stopped (low
178  * power functionality only.
179  */
180 #if configUSE_TICKLESS_IDLE == 1
181  static unsigned long ulStoppedTimerCompensation = 0;
182 #endif /* configUSE_TICKLESS_IDLE */
183 
184 /*
185  * Used by the portASSERT_IF_INTERRUPT_PRIORITY_INVALID() macro to ensure
186  * FreeRTOS API functions are not called from interrupts that have been assigned
187  * a priority above configMAX_SYSCALL_INTERRUPT_PRIORITY.
188  */
189 #if ( configASSERT_DEFINED == 1 )
190  static unsigned char ucMaxSysCallPriority = 0;
191  static unsigned long ulMaxPRIGROUPValue = 0;
192  static const volatile unsigned char * const pcInterruptPriorityRegisters = ( const volatile unsigned char * const ) portNVIC_IP_REGISTERS_OFFSET_16;
193 #endif /* configASSERT_DEFINED */
194 
195 /*-----------------------------------------------------------*/
196 
197 /*
198  * See header file for description.
199  */
200 portSTACK_TYPE *pxPortInitialiseStack( portSTACK_TYPE *pxTopOfStack, pdTASK_CODE pxCode, void *pvParameters )
201 {
202  /* Simulate the stack frame as it would be created by a context switch
203  interrupt. */
204  pxTopOfStack--; /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
205  *pxTopOfStack = portINITIAL_XPSR; /* xPSR */
206  pxTopOfStack--;
207  *pxTopOfStack = ( portSTACK_TYPE ) pxCode; /* PC */
208  pxTopOfStack--;
209  *pxTopOfStack = ( portSTACK_TYPE ) portTASK_RETURN_ADDRESS; /* LR */
210  pxTopOfStack -= 5; /* R12, R3, R2 and R1. */
211  *pxTopOfStack = ( portSTACK_TYPE ) pvParameters; /* R0 */
212  pxTopOfStack -= 8; /* R11, R10, R9, R8, R7, R6, R5 and R4. */
213 
214  return pxTopOfStack;
215 }
216 /*-----------------------------------------------------------*/
217 
218 static void prvTaskExitError( void )
219 {
220  /* A function that implements a task must not exit or attempt to return to
221  its caller as there is nothing to return to. If a task wants to exit it
222  should instead call vTaskDelete( NULL ).
223 
224  Artificially force an assert() to be triggered if configASSERT() is
225  defined, then stop here so application writers can catch the error. */
226  configASSERT( uxCriticalNesting == ~0UL );
228  for( ;; );
229 }
230 /*-----------------------------------------------------------*/
231 
232 void vPortSVCHandler( void )
233 {
234  __asm volatile (
235  " ldr r3, pxCurrentTCBConst2 \n" /* Restore the context. */
236  " ldr r1, [r3] \n" /* Use pxCurrentTCBConst to get the pxCurrentTCB address. */
237  " ldr r0, [r1] \n" /* The first item in pxCurrentTCB is the task top of stack. */
238  " ldmia r0!, {r4-r11} \n" /* Pop the registers that are not automatically saved on exception entry and the critical nesting count. */
239  " msr psp, r0 \n" /* Restore the task stack pointer. */
240  " mov r0, #0 \n"
241  " msr basepri, r0 \n"
242  " orr r14, #0xd \n"
243  " bx r14 \n"
244  " \n"
245  " .align 2 \n"
246  "pxCurrentTCBConst2: .word pxCurrentTCB \n"
247  );
248 }
249 /*-----------------------------------------------------------*/
250 
251 static void prvPortStartFirstTask( void )
252 {
253  __asm volatile(
254  " ldr r0, =0xE000ED08 \n" /* Use the NVIC offset register to locate the stack. */
255  " ldr r0, [r0] \n"
256  " ldr r0, [r0] \n"
257  " msr msp, r0 \n" /* Set the msp back to the start of the stack. */
258  " cpsie i \n" /* Globally enable interrupts. */
259  " svc 0 \n" /* System call to start first task. */
260  " nop \n"
261  );
262 }
263 /*-----------------------------------------------------------*/
264 
265 /*
266  * See header file for description.
267  */
269 {
270  /* configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.
271  See http://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */
273 
274  #if( configASSERT_DEFINED == 1 )
275  {
276  volatile unsigned long ulOriginalPriority;
277  volatile char * const pcFirstUserPriorityRegister = ( volatile char * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );
278  volatile unsigned char ucMaxPriorityValue;
279 
280  /* Determine the maximum priority from which ISR safe FreeRTOS API
281  functions can be called. ISR safe functions are those that end in
282  "FromISR". FreeRTOS maintains separate thread and ISR API functions to
283  ensure interrupt entry is as fast and simple as possible.
284 
285  Save the interrupt priority value that is about to be clobbered. */
286  ulOriginalPriority = *pcFirstUserPriorityRegister;
287 
288  /* Determine the number of priority bits available. First write to all
289  possible bits. */
290  *pcFirstUserPriorityRegister = portMAX_8_BIT_VALUE;
291 
292  /* Read the value back to see how many bits stuck. */
293  ucMaxPriorityValue = *pcFirstUserPriorityRegister;
294 
295  /* Use the same mask on the maximum system call priority. */
296  ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;
297 
298  /* Calculate the maximum acceptable priority group value for the number
299  of bits read back. */
300  ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;
301  while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )
302  {
303  ulMaxPRIGROUPValue--;
304  ucMaxPriorityValue <<= ( unsigned char ) 0x01;
305  }
306 
307  /* Shift the priority group value back to its position within the AIRCR
308  register. */
309  ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;
310  ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;
311 
312  /* Restore the clobbered interrupt priority register to its original
313  value. */
314  *pcFirstUserPriorityRegister = ulOriginalPriority;
315  }
316  #endif /* conifgASSERT_DEFINED */
317 
318  /* Make PendSV and SysTick the lowest priority interrupts. */
321 
322  /* Start the timer that generates the tick ISR. Interrupts are disabled
323  here already. */
325 
326  /* Initialise the critical nesting count ready for the first task. */
327  uxCriticalNesting = 0;
328 
329  /* Start the first task. */
330  prvPortStartFirstTask();
331 
332  /* Should not get here! */
333  return 0;
334 }
335 /*-----------------------------------------------------------*/
336 
337 void vPortEndScheduler( void )
338 {
339  /* It is unlikely that the CM3 port will require this function as there
340  is nothing to return to. */
341 }
342 /*-----------------------------------------------------------*/
343 
344 void vPortYield( void )
345 {
346  /* Set a PendSV to request a context switch. */
348 
349  /* Barriers are normally not required but do ensure the code is completely
350  within the specified behaviour for the architecture. */
351  __asm volatile( "dsb" );
352  __asm volatile( "isb" );
353 }
354 /*-----------------------------------------------------------*/
355 
356 void vPortEnterCritical( void )
357 {
359  uxCriticalNesting++;
360  __asm volatile( "dsb" );
361  __asm volatile( "isb" );
362 }
363 /*-----------------------------------------------------------*/
364 
365 void vPortExitCritical( void )
366 {
367  uxCriticalNesting--;
368  if( uxCriticalNesting == 0 )
369  {
371  }
372 }
373 /*-----------------------------------------------------------*/
374 
375 __attribute__(( naked )) unsigned long ulPortSetInterruptMask( void )
376 {
377  __asm volatile \
378  ( \
379  " mrs r0, basepri \n" \
380  " mov r1, %0 \n" \
381  " msr basepri, r1 \n" \
382  " bx lr \n" \
383  :: "i" ( configMAX_SYSCALL_INTERRUPT_PRIORITY ) : "r0", "r1" \
384  );
385 
386  /* This return will not be reached but is necessary to prevent compiler
387  warnings. */
388  return 0;
389 }
390 /*-----------------------------------------------------------*/
391 
392 __attribute__(( naked )) void vPortClearInterruptMask( unsigned long ulNewMaskValue )
393 {
394  __asm volatile \
395  ( \
396  " msr basepri, r0 \n" \
397  " bx lr \n" \
398  :::"r0" \
399  );
400 
401  /* Just to avoid compiler warnings. */
402  ( void ) ulNewMaskValue;
403 }
404 /*-----------------------------------------------------------*/
405 
406 void xPortPendSVHandler( void )
407 {
408  /* This is a naked function. */
409 
410  __asm volatile
411  (
412  " mrs r0, psp \n"
413  " \n"
414  " ldr r3, pxCurrentTCBConst \n" /* Get the location of the current TCB. */
415  " ldr r2, [r3] \n"
416  " \n"
417  " stmdb r0!, {r4-r11} \n" /* Save the remaining registers. */
418  " str r0, [r2] \n" /* Save the new top of stack into the first member of the TCB. */
419  " \n"
420  " stmdb sp!, {r3, r14} \n"
421  " mov r0, %0 \n"
422  " msr basepri, r0 \n"
423  " bl vTaskSwitchContext \n"
424  " mov r0, #0 \n"
425  " msr basepri, r0 \n"
426  " ldmia sp!, {r3, r14} \n"
427  " \n" /* Restore the context, including the critical nesting count. */
428  " ldr r1, [r3] \n"
429  " ldr r0, [r1] \n" /* The first item in pxCurrentTCB is the task top of stack. */
430  " ldmia r0!, {r4-r11} \n" /* Pop the registers. */
431  " msr psp, r0 \n"
432  " bx r14 \n"
433  " \n"
434  " .align 2 \n"
435  "pxCurrentTCBConst: .word pxCurrentTCB \n"
437  );
438 }
439 /*-----------------------------------------------------------*/
440 
442 {
443  /* The SysTick runs at the lowest interrupt priority, so when this interrupt
444  executes all interrupts must be unmasked. There is therefore no need to
445  save and then restore the interrupt mask value as its value is already
446  known. */
448  {
449  /* Increment the RTOS tick. */
450  if( xTaskIncrementTick() != pdFALSE )
451  {
452  /* A context switch is required. Context switching is performed in
453  the PendSV interrupt. Pend the PendSV interrupt. */
455  }
456  }
458 }
459 /*-----------------------------------------------------------*/
460 
461 #if configUSE_TICKLESS_IDLE == 1
462 
463  __attribute__((weak)) void vPortSuppressTicksAndSleep( portTickType xExpectedIdleTime )
464  {
465  unsigned long ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
466  portTickType xModifiableIdleTime;
467 
468  /* Make sure the SysTick reload value does not overflow the counter. */
469  if( xExpectedIdleTime > xMaximumPossibleSuppressedTicks )
470  {
471  xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
472  }
473 
474  /* Stop the SysTick momentarily. The time the SysTick is stopped for
475  is accounted for as best it can be, but using the tickless mode will
476  inevitably result in some tiny drift of the time maintained by the
477  kernel with respect to calendar time. */
479 
480  /* Calculate the reload value required to wait xExpectedIdleTime
481  tick periods. -1 is used because this code will execute part way
482  through one of the tick periods. */
483  ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
484  if( ulReloadValue > ulStoppedTimerCompensation )
485  {
486  ulReloadValue -= ulStoppedTimerCompensation;
487  }
488 
489  /* Enter a critical section but don't use the taskENTER_CRITICAL()
490  method as that will mask interrupts that should exit sleep mode. */
491  __asm volatile( "cpsid i" );
492 
493  /* If a context switch is pending or a task is waiting for the scheduler
494  to be unsuspended then abandon the low power entry. */
496  {
497  /* Restart from whatever is left in the count register to complete
498  this tick period. */
500 
501  /* Restart SysTick. */
503 
504  /* Reset the reload register to the value required for normal tick
505  periods. */
506  portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
507 
508  /* Re-enable interrupts - see comments above the cpsid instruction()
509  above. */
510  __asm volatile( "cpsie i" );
511  }
512  else
513  {
514  /* Set the new reload value. */
515  portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
516 
517  /* Clear the SysTick count flag and set the count value back to
518  zero. */
520 
521  /* Restart SysTick. */
523 
524  /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
525  set its parameter to 0 to indicate that its implementation contains
526  its own wait for interrupt or wait for event instruction, and so wfi
527  should not be executed again. However, the original expected idle
528  time variable must remain unmodified, so a copy is taken. */
529  xModifiableIdleTime = xExpectedIdleTime;
530  configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
531  if( xModifiableIdleTime > 0 )
532  {
533  __asm volatile( "dsb" );
534  __asm volatile( "wfi" );
535  __asm volatile( "isb" );
536  }
537  configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
538 
539  /* Stop SysTick. Again, the time the SysTick is stopped for is
540  accounted for as best it can be, but using the tickless mode will
541  inevitably result in some tiny drift of the time maintained by the
542  kernel with respect to calendar time. */
544 
545  /* Re-enable interrupts - see comments above the cpsid instruction()
546  above. */
547  __asm volatile( "cpsie i" );
548 
550  {
551  unsigned long ulCalculatedLoadValue;
552 
553  /* The tick interrupt has already executed, and the SysTick
554  count reloaded with ulReloadValue. Reset the
555  portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
556  period. */
557  ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
558 
559  /* Don't allow a tiny value, or values that have somehow
560  underflowed because the post sleep hook did something
561  that took too long. */
562  if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
563  {
564  ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
565  }
566 
567  portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
568 
569  /* The tick interrupt handler will already have pended the tick
570  processing in the kernel. As the pending tick will be
571  processed as soon as this function exits, the tick value
572  maintained by the tick is stepped forward by one less than the
573  time spent waiting. */
574  ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
575  }
576  else
577  {
578  /* Something other than the tick interrupt ended the sleep.
579  Work out how long the sleep lasted rounded to complete tick
580  periods (not the ulReload value which accounted for part
581  ticks). */
582  ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
583 
584  /* How many complete tick periods passed while the processor
585  was waiting? */
586  ulCompleteTickPeriods = ulCompletedSysTickDecrements / ulTimerCountsForOneTick;
587 
588  /* The reload value is set to whatever fraction of a single tick
589  period remains. */
590  portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1 ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
591  }
592 
593  /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
594  again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
595  value. The critical section is used to ensure the tick interrupt
596  can only execute once in the case that the reload register is near
597  zero. */
600  {
602  vTaskStepTick( ulCompleteTickPeriods );
603  portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
604  }
606  }
607  }
608 
609 #endif /* #if configUSE_TICKLESS_IDLE */
610 /*-----------------------------------------------------------*/
611 
612 /*
613  * Setup the systick timer to generate the tick interrupts at the required
614  * frequency.
615  */
617 {
618  /* Calculate the constants required to configure the tick interrupt. */
619  #if configUSE_TICKLESS_IDLE == 1
620  {
621  ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
622  xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
623  ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
624  }
625  #endif /* configUSE_TICKLESS_IDLE */
626 
627  /* Configure SysTick to interrupt at the requested rate. */
630 }
631 /*-----------------------------------------------------------*/
632 
633 #if( configASSERT_DEFINED == 1 )
634 
635  void vPortValidateInterruptPriority( void )
636  {
637  unsigned long ulCurrentInterrupt;
638  unsigned char ucCurrentPriority;
639 
640  /* Obtain the number of the currently executing interrupt. */
641  __asm volatile( "mrs %0, ipsr" : "=r"( ulCurrentInterrupt ) );
642 
643  /* Is the interrupt number a user defined interrupt? */
644  if( ulCurrentInterrupt >= portFIRST_USER_INTERRUPT_NUMBER )
645  {
646  /* Look up the interrupt's priority. */
647  ucCurrentPriority = pcInterruptPriorityRegisters[ ulCurrentInterrupt ];
648 
649  /* The following assertion will fail if a service routine (ISR) for
650  an interrupt that has been assigned a priority above
651  configMAX_SYSCALL_INTERRUPT_PRIORITY calls an ISR safe FreeRTOS API
652  function. ISR safe FreeRTOS API functions must *only* be called
653  from interrupts that have been assigned a priority at or below
654  configMAX_SYSCALL_INTERRUPT_PRIORITY.
655 
656  Numerically low interrupt priority numbers represent logically high
657  interrupt priorities, therefore the priority of the interrupt must
658  be set to a value equal to or numerically *higher* than
659  configMAX_SYSCALL_INTERRUPT_PRIORITY.
660 
661  Interrupts that use the FreeRTOS API must not be left at their
662  default priority of zero as that is the highest possible priority,
663  which is guaranteed to be above configMAX_SYSCALL_INTERRUPT_PRIORITY,
664  and therefore also guaranteed to be invalid.
665 
666  FreeRTOS maintains separate thread and ISR API functions to ensure
667  interrupt entry is as fast and simple as possible.
668 
669  The following links provide detailed information:
670  http://www.freertos.org/RTOS-Cortex-M3-M4.html
671  http://www.freertos.org/FAQHelp.html */
672  configASSERT( ucCurrentPriority >= ucMaxSysCallPriority );
673  }
674 
675  /* Priority grouping: The interrupt controller (NVIC) allows the bits
676  that define each interrupt's priority to be split between bits that
677  define the interrupt's pre-emption priority bits and bits that define
678  the interrupt's sub-priority. For simplicity all bits must be defined
679  to be pre-emption priority bits. The following assertion will fail if
680  this is not the case (if some bits represent a sub-priority).
681 
682  If the application only uses CMSIS libraries for interrupt
683  configuration then the correct setting can be achieved on all Cortex-M
684  devices by calling NVIC_SetPriorityGrouping( 0 ); before starting the
685  scheduler. Note however that some vendor specific peripheral libraries
686  assume a non-zero priority group setting, in which cases using a value
687  of zero will result in unpredicable behaviour. */
688  configASSERT( ( portAIRCR_REG & portPRIORITY_GROUP_MASK ) <= ulMaxPRIGROUPValue );
689  }
690 
691 #endif /* configASSERT_DEFINED */
692 
693 
694 
695 
696 
697 
698 
699 
700 
701 
702 
703 
704 
705 
706 
707 
708 
709 
710 
711 
712