Tickless idle fixes/improvement (#59)

* Fix tickless idle when stopping systick on zero...

...and don't stop SysTick at all in the eAbortSleep case.

Prior to this commit, if vPortSuppressTicksAndSleep() happens to stop
the SysTick on zero, then after tickless idle ends, xTickCount advances
one full tick more than the time that actually elapsed as measured by
the SysTick.  See "bug 1" in this forum post:
https://forums.freertos.org/t/ultasknotifytake-timeout-accuracy/9629/40

SysTick
-------
The SysTick is the hardware timer that provides the OS tick interrupt
in the official ports for Cortex M.  SysTick starts counting down from
the value stored in its reload register.  When SysTick reaches zero, it
requests an interrupt.  On the next SysTick clock cycle, it loads the
counter again from the reload register.  To get periodic interrupts
every N SysTick clock cycles, the reload register must be N - 1.

Bug Example
-----------
- Idle task calls vPortSuppressTicksAndSleep(xExpectedIdleTime = 2).
  [Doesn't have to be "2" -- could be any number.]
- vPortSuppressTicksAndSleep() stops SysTick, and the current-count
  register happens to stop on zero.
- SysTick ISR executes, setting xPendedTicks = 1
- vPortSuppressTicksAndSleep() masks interrupts and calls
  eTaskConfirmSleepModeStatus() which confirms the sleep operation. ***
- vPortSuppressTicksAndSleep() configures SysTick for 1 full tick
  (xExpectedIdleTime - 1) plus the current-count register (which is 0)
- One tick period elapses in sleep.
- SysTick wakes CPU, ISR executes and increments xPendedTicks to 2.
- vPortSuppressTicksAndSleep() calls vTaskStepTick(1), then returns.
- Idle task resumes scheduler, which increments xTickCount twice (for
  xPendedTicks = 2)

In the end, two ticks elapsed as measured by SysTick, but the code
increments xTickCount three times.  The root cause is that the code
assumes the SysTick current-count register always contains the number of
SysTick counts remaining in the current tick period.  However, when the
current-count register is zero, there are ulTimerCountsForOneTick
counts remaining, not zero.  This error is not the kind of time slippage
normally associated with tickless idle.

*** Note that a recent commit https://github.com/FreeRTOS/FreeRTOS-Kernel/commit/e1b98f0
results in eAbortSleep in this case, due to xPendedTicks != 0.  That
commit does mostly resolve this bug without specifically mentioning
it, and without this commit.  But that resolution allows the code in
port.c not to directly address the special case of stopping SysTick on
zero in any code or comments.  That commit also generates additional
instances of eAbortSleep, and a second purpose of this commit is to
optimize how vPortSuppressTicksAndSleep() behaves for eAbortSleep, as
noted below.

This commit also includes an optimization to avoid stopping the SysTick
when eTaskConfirmSleepModeStatus() returns eAbortSleep.  This
optimization belongs with this fix because the method of handling the
SysTick being stopped on zero changes with this optimization.

* Fix imminent tick rescheduled after tickless idle

Prior to this commit, if something other than systick wakes the CPU from
tickless idle, vPortSuppressTicksAndSleep() might cause xTickCount to
increment once too many times.  See "bug 2" in this forum post:
https://forums.freertos.org/t/ultasknotifytake-timeout-accuracy/9629/40

SysTick
-------
The SysTick is the hardware timer that provides the OS tick interrupt
in the official ports for Cortex M.  SysTick starts counting down from
the value stored in its reload register.  When SysTick reaches zero, it
requests an interrupt.  On the next SysTick clock cycle, it loads the
counter again from the reload register.  To get periodic interrupts
every N SysTick clock cycles, the reload register must be N - 1.

Bug Example
-----------
- CPU is sleeping in vPortSuppressTicksAndSleep()
- Something other than the SysTick wakes the CPU.
- vPortSuppressTicksAndSleep() calculates the number of SysTick counts
  until the next tick.  The bug occurs only if this number is small.
- vPortSuppressTicksAndSleep() puts this small number into the SysTick
  reload register, and starts SysTick.
- vPortSuppressTicksAndSleep() calls vTaskStepTick()
- While vTaskStepTick() executes, the SysTick expires.  The ISR pends
  because interrupts are masked, and SysTick starts a 2nd period still
  based on the small number of counts in its reload register.  This 2nd
  period is undesirable and is likely to cause the error noted below.
- vPortSuppressTicksAndSleep() puts the normal tick duration into the
  SysTick's reload register.
- vPortSuppressTicksAndSleep() unmasks interrupts before the SysTick
  starts a new period based on the new value in the reload register.
  [This is a race condition that can go either way, but for the bug
  to occur, the race must play out this way.]
- The pending SysTick ISR executes and increments xPendedTicks.
- The SysTick expires again, finishing the second very small period, and
  starts a new period this time based on the full tick duration.
- The SysTick ISR increments xPendedTicks (or xTickCount) even though
  only a tiny fraction of a tick period has elapsed since the previous
  tick.

The bug occurs when *two* consecutive small periods of the SysTick are
both counted as ticks.  The root cause is a race caused by the small
SysTick period.  If vPortSuppressTicksAndSleep() unmasks interrupts
*after* the small period expires but *before* the SysTick starts a
period based on the full tick period, then two small periods are
counted as ticks when only one should be counted.

The end result is xTickCount advancing nearly one full tick more than
time actually elapsed as measured by the SysTick.  This is not the kind
of time slippage normally associated with tickless idle.

After this commit the code starts the SysTick and then immediately
modifies the reload register to ensure the very short cycle (if any) is
conducted only once.  This strategy requires special consideration for
the build option that configures SysTick to use a divided clock.  To
avoid waiting around for the SysTick to load value from the reload
register, the new code temporarily configures the SysTick to use the
undivided clock.  The resulting timing error is typical for tickless
idle.  The error (commonly known as drift or slippage in kernel time)
caused by this strategy is equivalent to one or two counts in
ulStoppedTimerCompensation.

This commit also updates comments and #define symbols related to the
SysTick clock option.  The SysTick can optionally be clocked by a
divided version of the CPU clock (commonly divide-by-8).  The new code
in this commit adjusts these comments and symbols to make them clearer
and more useful in configurations that use the divided clock.  The fix
made in this commit requires the use of these symbols, as noted in the
code comments.

* Fix tickless idle with alternate systick clocking

Prior to this commit, in configurations using the alternate SysTick
clocking, vPortSuppressTicksAndSleep() might cause xTickCount to jump
ahead as much as the entire expected idle time or fall behind as much
as one full tick compared to time as measured by the SysTick.

SysTick
-------
The SysTick is the hardware timer that provides the OS tick interrupt
in the official ports for Cortex M. SysTick starts counting down from
the value stored in its reload register. When SysTick reaches zero, it
requests an interrupt. On the next SysTick clock cycle, it loads the
counter again from the reload register. The SysTick has a configuration
option to be clocked by an alternate clock besides the core clock.
This alternate clock is MCU dependent.

Scenarios Fixed
---------------
The new code in this commit handles the following scenarios that were
not handled correctly prior to this commit.

1. Before the sleep, vPortSuppressTicksAndSleep() stops the SysTick on
zero, long after SysTick reached zero.  Prior to this commit, this
scenario caused xTickCount to jump ahead one full tick for the same
reason documented here: https://github.com/FreeRTOS/FreeRTOS-Kernel/pull/59/commits/0c7b04bd3a745c52151abebc882eed3f811c4c81

2. After the sleep, vPortSuppressTicksAndSleep() stops the SysTick
before it loads the counter from the reload register.  Prior to this
commit, this scenario caused xTickCount to jump ahead by the entire
expected idle time (xExpectedIdleTime) because the current-count
register is zero before it loads from the reload register.

3. Prior to return, vPortSuppressTicksAndSleep() attempts to start a
short SysTick period when the current SysTick clock cycle has a lot of
time remaining.  Prior to this commit, this scenario could cause
xTickCount to fall behind by as much as nearly one full tick because the
short SysTick cycle never started.

Note that #3 is partially fixed by https://github.com/FreeRTOS/FreeRTOS-Kernel/pull/59/commits/967acc9b200d3d4beeb289d9da9e88798074b431
even though that commit addresses a different issue.  So this commit
completes the partial fix.

* Improve comments and name of preprocessor symbol

Add a note in the code comments that SysTick requests an interrupt when
decrementing from 1 to 0, so that's why stopping SysTick on zero is a
special case.  Readers might unknowingly assume that SysTick requests
an interrupt when wrapping from 0 back to the load-register value.

Reconsider new "_SETTING" suffix since "_CONFIG" suffix seems more
descriptive.  The code relies on *both* of these preprocessor symbols:

portNVIC_SYSTICK_CLK_BIT
portNVIC_SYSTICK_CLK_BIT_CONFIG  **new**

A meaningful suffix is really helpful to distinguish the two symbols.

* Revert introduction of 2nd name for NVIC register

When I added portNVIC_ICSR_REG I didn't realize there was already a
portNVIC_INT_CTRL_REG, which identifies the same register.  Not good
to have both.  Note that portNVIC_INT_CTRL_REG is defined in portmacro.h
and is already used in this file (port.c).

* Replicate to other Cortex M ports

Also set a new fiddle factor based on tests with a CM4F.  I used gcc,
optimizing at -O1.  Users can fine-tune as needed.

Also add configSYSTICK_CLOCK_HZ to the CM0 ports to be just like the
other Cortex M ports.  This change allowed uniformity in the default
tickless implementations across all Cortex M ports.  And CM0 is likely
to benefit from configSYSTICK_CLOCK_HZ, especially considering new CM0
devices with very fast CPU clock speeds.

* Revert changes to IAR-CM0-portmacro.h

portNVIC_INT_CTRL_REG was already defined in port.c.  No need to define
it in portmacro.h.

* Handle edge cases with slow SysTick clock

Co-authored-by: Cobus van Eeden <35851496+cobusve@users.noreply.github.com>
Co-authored-by: abhidixi11 <44424462+abhidixi11@users.noreply.github.com>
Co-authored-by: Joseph Julicher <jjulicher@mac.com>
Co-authored-by: alfred gedeon <28123637+alfred2g@users.noreply.github.com>
diff --git a/portable/ARMv8M/non_secure/port.c b/portable/ARMv8M/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/ARMv8M/non_secure/port.c
+++ b/portable/ARMv8M/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/CCS/ARM_CM3/port.c b/portable/CCS/ARM_CM3/port.c
index 35c832b..257aefa 100644
--- a/portable/CCS/ARM_CM3/port.c
+++ b/portable/CCS/ARM_CM3/port.c
@@ -38,27 +38,18 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -86,12 +77,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -213,66 +216,66 @@
 BaseType_t xPortStartScheduler( void )

 {

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -356,7 +359,7 @@
     #pragma WEAK( vPortSuppressTicksAndSleep )

     void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -365,22 +368,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm( "	cpsid i");

@@ -391,23 +378,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

-             * call above. */

+            /* Re-enable interrupts - see comments above the cpsid instruction

+             * above. */

             __asm( "	cpsie i");

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -436,8 +449,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm( "	cpsie i");

             __asm( "	dsb");

             __asm( "	isb");

@@ -457,27 +470,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -491,11 +500,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -506,13 +534,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm( "	cpsie i");

@@ -531,11 +585,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -544,7 +598,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/CCS/ARM_CM4F/port.c b/portable/CCS/ARM_CM4F/port.c
index c503660..1028d83 100644
--- a/portable/CCS/ARM_CM4F/port.c
+++ b/portable/CCS/ARM_CM4F/port.c
@@ -42,27 +42,18 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -95,12 +86,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -232,66 +235,66 @@
 BaseType_t xPortStartScheduler( void )

 {

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -381,7 +384,7 @@
     #pragma WEAK( vPortSuppressTicksAndSleep )

     void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -390,22 +393,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm( "	cpsid i");

@@ -416,23 +403,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

-             * call above. */

+            /* Re-enable interrupts - see comments above the cpsid instruction

+             * above. */

             __asm( "	cpsie i");

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -461,8 +474,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm( "	cpsie i");

             __asm( "	dsb");

             __asm( "	isb");

@@ -482,27 +495,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -516,11 +525,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -531,13 +559,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm( "	cpsie i");

@@ -556,11 +610,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -569,7 +623,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/GCC/ARM_CM0/port.c b/portable/GCC/ARM_CM0/port.c
index 18d073f..1983073 100644
--- a/portable/GCC/ARM_CM0/port.c
+++ b/portable/GCC/ARM_CM0/port.c
@@ -34,27 +34,19 @@
 #include "FreeRTOS.h"

 #include "task.h"

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the NVIC. */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_INT_CTRL_REG                 ( *( ( volatile uint32_t * ) 0xe000ed04 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVSET_BIT                ( 1UL << 28UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )

 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )

 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )

@@ -69,7 +61,19 @@
  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

 #ifndef portMISSED_COUNTS_FACTOR

-    #define portMISSED_COUNTS_FACTOR    ( 45UL )

+    #define portMISSED_COUNTS_FACTOR    ( 94UL )

+#endif

+

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

 #endif

 

 /* Let the user override the pre-loading of the initial LR with the address of

@@ -391,11 +395,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and reset the SysTick. */

@@ -404,7 +408,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

@@ -412,7 +416,7 @@
 

     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -421,22 +425,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm volatile ( "cpsid i" ::: "memory" );

@@ -447,23 +435,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above the cpsid instruction()

+            /* Re-enable interrupts - see comments above the cpsid instruction

              * above. */

             __asm volatile ( "cpsie i" ::: "memory" );

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -492,8 +506,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm volatile ( "cpsie i" ::: "memory" );

             __asm volatile ( "dsb" );

             __asm volatile ( "isb" );

@@ -513,27 +527,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -547,11 +557,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -562,15 +591,41 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-            vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

 

-            /* Exit with interrpts enabled. */

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

+            vTaskStepTick( ulCompleteTickPeriods );

+

+            /* Exit with interrupts enabled. */

             __asm volatile ( "cpsie i" ::: "memory" );

         }

     }

diff --git a/portable/GCC/ARM_CM23/non_secure/port.c b/portable/GCC/ARM_CM23/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/GCC/ARM_CM23/non_secure/port.c
+++ b/portable/GCC/ARM_CM23/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/GCC/ARM_CM23_NTZ/non_secure/port.c b/portable/GCC/ARM_CM23_NTZ/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/GCC/ARM_CM23_NTZ/non_secure/port.c
+++ b/portable/GCC/ARM_CM23_NTZ/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/GCC/ARM_CM3/port.c b/portable/GCC/ARM_CM3/port.c
index ac882e9..f41f215 100644
--- a/portable/GCC/ARM_CM3/port.c
+++ b/portable/GCC/ARM_CM3/port.c
@@ -41,27 +41,18 @@
     #define configKERNEL_INTERRUPT_PRIORITY    255

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -89,12 +80,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /* Let the user override the pre-loading of the initial LR with the address of

  * prvTaskExitError() in case it messes up unwinding of the stack in the

  * debugger. */

@@ -267,66 +270,66 @@
     configASSERT( configMAX_SYSCALL_INTERRUPT_PRIORITY );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -455,7 +458,7 @@
 

     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -464,22 +467,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm volatile ( "cpsid i" ::: "memory" );

@@ -490,23 +477,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above the cpsid instruction()

+            /* Re-enable interrupts - see comments above the cpsid instruction

              * above. */

             __asm volatile ( "cpsie i" ::: "memory" );

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -535,8 +548,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm volatile ( "cpsie i" ::: "memory" );

             __asm volatile ( "dsb" );

             __asm volatile ( "isb" );

@@ -556,27 +569,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -590,11 +599,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -605,13 +633,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm volatile ( "cpsie i" ::: "memory" );

@@ -629,11 +683,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -642,7 +696,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/GCC/ARM_CM33/non_secure/port.c b/portable/GCC/ARM_CM33/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/GCC/ARM_CM33/non_secure/port.c
+++ b/portable/GCC/ARM_CM33/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/GCC/ARM_CM33_NTZ/non_secure/port.c b/portable/GCC/ARM_CM33_NTZ/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/GCC/ARM_CM33_NTZ/non_secure/port.c
+++ b/portable/GCC/ARM_CM33_NTZ/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/GCC/ARM_CM4F/port.c b/portable/GCC/ARM_CM4F/port.c
index ae05a32..c62d160 100644
--- a/portable/GCC/ARM_CM4F/port.c
+++ b/portable/GCC/ARM_CM4F/port.c
@@ -38,27 +38,18 @@
     #error This port can only be used when the project options are configured to enable hardware floating point support.

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 /* Constants used to detect a Cortex-M7 r0p1 core, which should use the ARM_CM7

@@ -101,7 +92,19 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

+

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

 

 /* Let the user override the pre-loading of the initial LR with the address of

  * prvTaskExitError() in case it messes up unwinding of the stack in the

@@ -303,66 +306,66 @@
     configASSERT( portCPUID != portCORTEX_M7_r0p0_ID );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -517,7 +520,7 @@
 

     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -526,22 +529,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm volatile ( "cpsid i" ::: "memory" );

@@ -552,23 +539,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above the cpsid instruction()

+            /* Re-enable interrupts - see comments above the cpsid instruction

              * above. */

             __asm volatile ( "cpsie i" ::: "memory" );

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -597,8 +610,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm volatile ( "cpsie i" ::: "memory" );

             __asm volatile ( "dsb" );

             __asm volatile ( "isb" );

@@ -618,27 +631,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -652,11 +661,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -667,13 +695,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm volatile ( "cpsie i" ::: "memory" );

@@ -691,11 +745,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -704,7 +758,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/GCC/ARM_CM7/r0p1/port.c b/portable/GCC/ARM_CM7/r0p1/port.c
index 130e187..7bfaa1c 100644
--- a/portable/GCC/ARM_CM7/r0p1/port.c
+++ b/portable/GCC/ARM_CM7/r0p1/port.c
@@ -38,27 +38,18 @@
     #error This port can only be used when the project options are configured to enable hardware floating point support.

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -95,7 +86,19 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

+

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

 

 /* Let the user override the pre-loading of the initial LR with the address of

  * prvTaskExitError() in case it messes up unwinding of the stack in the

@@ -291,66 +294,66 @@
     configASSERT( configMAX_SYSCALL_INTERRUPT_PRIORITY );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -507,7 +510,7 @@
 

     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -516,22 +519,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm volatile ( "cpsid i" ::: "memory" );

@@ -542,23 +529,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above the cpsid instruction()

+            /* Re-enable interrupts - see comments above the cpsid instruction

              * above. */

             __asm volatile ( "cpsie i" ::: "memory" );

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -587,8 +600,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm volatile ( "cpsie i" ::: "memory" );

             __asm volatile ( "dsb" );

             __asm volatile ( "isb" );

@@ -608,27 +621,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -642,11 +651,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -657,13 +685,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm volatile ( "cpsie i" ::: "memory" );

@@ -681,11 +735,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -694,7 +748,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/IAR/ARM_CM0/port.c b/portable/IAR/ARM_CM0/port.c
index 6fbc852..4e40442 100644
--- a/portable/IAR/ARM_CM0/port.c
+++ b/portable/IAR/ARM_CM0/port.c
@@ -37,26 +37,18 @@
 #include "FreeRTOS.h"

 #include "task.h"

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the NVIC. */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_INT_CTRL_REG                 ( *( ( volatile uint32_t * ) 0xe000ed04 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )

 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )

 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )

@@ -82,7 +74,7 @@
  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

 #ifndef portMISSED_COUNTS_FACTOR

-    #define portMISSED_COUNTS_FACTOR    ( 45UL )

+    #define portMISSED_COUNTS_FACTOR    ( 94UL )

 #endif

 

 /* The number of SysTick increments that make up one tick period. */

@@ -102,6 +94,18 @@
     static uint32_t ulStoppedTimerCompensation = 0;

 #endif /* configUSE_TICKLESS_IDLE */

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -256,11 +260,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and reset the SysTick. */

@@ -269,7 +273,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

@@ -277,7 +281,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -286,22 +290,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_interrupt();

@@ -312,23 +300,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

+            /* Re-enable interrupts - see comments above the __disable_interrupt()

              * call above. */

             __enable_interrupt();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -357,8 +371,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_interrupt() call above. */

             __enable_interrupt();

             __DSB();

             __ISB();

@@ -378,27 +392,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -412,11 +422,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -427,15 +456,41 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-            vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

 

-            /* Exit with interrpts enabled. */

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

+            vTaskStepTick( ulCompleteTickPeriods );

+

+            /* Exit with interrupts enabled. */

             __enable_interrupt();

         }

     }

diff --git a/portable/IAR/ARM_CM0/portmacro.h b/portable/IAR/ARM_CM0/portmacro.h
index fc9eaa4..56127df 100644
--- a/portable/IAR/ARM_CM0/portmacro.h
+++ b/portable/IAR/ARM_CM0/portmacro.h
@@ -26,7 +26,6 @@
  *

  */

 

-

 #ifndef PORTMACRO_H

     #define PORTMACRO_H

 

@@ -80,8 +79,8 @@
 

 /* Scheduler utilities. */

     extern void vPortYield( void );

-    #define portNVIC_INT_CTRL     ( ( volatile uint32_t * ) 0xe000ed04 )

-    #define portNVIC_PENDSVSET    0x10000000

+    #define portNVIC_INT_CTRL                           ( ( volatile uint32_t * ) 0xe000ed04 )

+    #define portNVIC_PENDSVSET                          0x10000000

     #define portYIELD()                                 vPortYield()

     #define portEND_SWITCHING_ISR( xSwitchRequired )    if( xSwitchRequired ) *( portNVIC_INT_CTRL ) = portNVIC_PENDSVSET

     #define portYIELD_FROM_ISR( x )                     portEND_SWITCHING_ISR( x )

diff --git a/portable/IAR/ARM_CM23/non_secure/port.c b/portable/IAR/ARM_CM23/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/IAR/ARM_CM23/non_secure/port.c
+++ b/portable/IAR/ARM_CM23/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/IAR/ARM_CM23_NTZ/non_secure/port.c b/portable/IAR/ARM_CM23_NTZ/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/IAR/ARM_CM23_NTZ/non_secure/port.c
+++ b/portable/IAR/ARM_CM23_NTZ/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/IAR/ARM_CM3/port.c b/portable/IAR/ARM_CM3/port.c
index 92b211c..f8e6a1b 100644
--- a/portable/IAR/ARM_CM3/port.c
+++ b/portable/IAR/ARM_CM3/port.c
@@ -41,27 +41,18 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -89,7 +80,7 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

@@ -102,6 +93,18 @@
     #define configKERNEL_INTERRUPT_PRIORITY    255

 #endif

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -216,66 +219,66 @@
     configASSERT( configMAX_SYSCALL_INTERRUPT_PRIORITY );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -358,7 +361,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -367,22 +370,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_interrupt();

@@ -393,23 +380,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

+            /* Re-enable interrupts - see comments above the __disable_interrupt()

              * call above. */

             __enable_interrupt();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -438,8 +451,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_interrupt() call above. */

             __enable_interrupt();

             __DSB();

             __ISB();

@@ -459,27 +472,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -493,11 +502,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -508,13 +536,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_interrupt();

@@ -532,11 +586,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -545,7 +599,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/IAR/ARM_CM33/non_secure/port.c b/portable/IAR/ARM_CM33/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/IAR/ARM_CM33/non_secure/port.c
+++ b/portable/IAR/ARM_CM33/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/IAR/ARM_CM33_NTZ/non_secure/port.c b/portable/IAR/ARM_CM33_NTZ/non_secure/port.c
index d746923..9976dae 100644
--- a/portable/IAR/ARM_CM33_NTZ/non_secure/port.c
+++ b/portable/IAR/ARM_CM33_NTZ/non_secure/port.c
@@ -78,20 +78,13 @@
 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )
 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )
+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )
 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )
+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )
+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )
 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )
 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )
 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )
-#ifndef configSYSTICK_CLOCK_HZ
-    #define configSYSTICK_CLOCK_HZ            configCPU_CLOCK_HZ
-    /* Ensure the SysTick is clocked at the same frequency as the core. */
-    #define portNVIC_SYSTICK_CLK_BIT          ( 1UL << 2UL )
-#else
-
-/* The way the SysTick is clocked is not modified in case it is not the
- * same a the core. */
-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )
-#endif
 /*-----------------------------------------------------------*/
 
 /**
@@ -199,7 +192,7 @@
  * have occurred while the SysTick counter is stopped during tickless idle
  * calculations.
  */
-#define portMISSED_COUNTS_FACTOR    ( 45UL )
+#define portMISSED_COUNTS_FACTOR    ( 94UL )
 /*-----------------------------------------------------------*/
 
 /**
@@ -259,6 +252,20 @@
 #define portINITIAL_CONTROL_PRIVILEGED      ( 0x2 )
 
 /**
+ * @brief Let the user override the default SysTick clock rate.  If defined by the
+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the
+ * configuration register.
+ */
+#ifndef configSYSTICK_CLOCK_HZ
+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )
+    /* Ensure the SysTick is clocked at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )
+#else
+    /* Select the option to clock SysTick not at the same frequency as the core. */
+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )
+#endif
+
+/**
  * @brief Let the user override the pre-loading of the initial LR with the
  * address of prvTaskExitError() in case it messes up unwinding of the stack
  * in the debugger.
@@ -386,7 +393,7 @@
 #if ( configUSE_TICKLESS_IDLE == 1 )
     __attribute__( ( weak ) ) void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
     {
-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;
+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;
         TickType_t xModifiableIdleTime;
 
         /* Make sure the SysTick reload value does not overflow the counter. */
@@ -395,22 +402,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;
         }
 
-        /* Stop the SysTick momentarily. The time the SysTick is stopped for is
-         * accounted for as best it can be, but using the tickless mode will
-         * inevitably result in some tiny drift of the time maintained by the
-         * kernel with respect to calendar time. */
-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;
-
-        /* Calculate the reload value required to wait xExpectedIdleTime
-         * tick periods. -1 is used because this code will execute part way
-         * through one of the tick periods. */
-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
-
-        if( ulReloadValue > ulStoppedTimerCompensation )
-        {
-            ulReloadValue -= ulStoppedTimerCompensation;
-        }
-
         /* Enter a critical section but don't use the taskENTER_CRITICAL()
          * method as that will mask interrupts that should exit sleep mode. */
         __asm volatile ( "cpsid i" ::: "memory" );
@@ -418,26 +409,52 @@
         __asm volatile ( "isb" );
 
         /* If a context switch is pending or a task is waiting for the scheduler
-         * to be un-suspended then abandon the low power entry. */
+         * to be unsuspended then abandon the low power entry. */
         if( eTaskConfirmSleepModeStatus() == eAbortSleep )
         {
-            /* Restart from whatever is left in the count register to complete
-             * this tick period. */
-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;
-
-            /* Restart SysTick. */
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
-
-            /* Reset the reload register to the value required for normal tick
-             * periods. */
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
-
-            /* Re-enable interrupts - see comments above the cpsid instruction()
+            /* Re-enable interrupts - see comments above the cpsid instruction
              * above. */
             __asm volatile ( "cpsie i" ::: "memory" );
         }
         else
         {
+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for
+             * is accounted for as best it can be, but using the tickless mode will
+             * inevitably result in some tiny drift of the time maintained by the
+             * kernel with respect to calendar time. */
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
+
+            /* Use the SysTick current-value register to determine the number of
+             * SysTick decrements remaining until the next tick interrupt.  If the
+             * current-value register is zero, then there are actually
+             * ulTimerCountsForOneTick decrements remaining, not zero, because the
+             * SysTick requests the interrupt when decrementing from 1 to 0. */
+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+
+            if( ulSysTickDecrementsLeft == 0 )
+            {
+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;
+            }
+
+            /* Calculate the reload value required to wait xExpectedIdleTime
+             * tick periods.  -1 is used because this code normally executes part
+             * way through the first tick period.  But if the SysTick IRQ is now
+             * pending, then clear the IRQ, suppressing the first tick, and correct
+             * the reload value to reflect that the second tick period is already
+             * underway.  The expected idle time is always at least two ticks. */
+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );
+
+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )
+            {
+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;
+                ulReloadValue -= ulTimerCountsForOneTick;
+            }
+
+            if( ulReloadValue > ulStoppedTimerCompensation )
+            {
+                ulReloadValue -= ulStoppedTimerCompensation;
+            }
+
             /* Set the new reload value. */
             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;
 
@@ -448,12 +465,11 @@
             /* Restart SysTick. */
             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
 
-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can
-             * set its parameter to 0 to indicate that its implementation
-             * contains its own wait for interrupt or wait for event
-             * instruction, and so wfi should not be executed again. However,
-             * the original expected idle time variable must remain unmodified,
-             * so a copy is taken. */
+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can
+             * set its parameter to 0 to indicate that its implementation contains
+             * its own wait for interrupt or wait for event instruction, and so wfi
+             * should not be executed again.  However, the original expected idle
+             * time variable must remain unmodified, so a copy is taken. */
             xModifiableIdleTime = xExpectedIdleTime;
             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );
 
@@ -467,48 +483,44 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );
 
             /* Re-enable interrupts to allow the interrupt that brought the MCU
-             * out of sleep mode to execute immediately. See comments above
+             * out of sleep mode to execute immediately.  See comments above
              * the cpsid instruction above. */
             __asm volatile ( "cpsie i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable interrupts again because the clock is about to be stopped
-             * and interrupts that execute while the clock is stopped will
-             * increase any slippage between the time maintained by the RTOS and
-             * calendar time. */
+             * and interrupts that execute while the clock is stopped will increase
+             * any slippage between the time maintained by the RTOS and calendar
+             * time. */
             __asm volatile ( "cpsid i" ::: "memory" );
             __asm volatile ( "dsb" );
             __asm volatile ( "isb" );
 
             /* Disable the SysTick clock without reading the
              * portNVIC_SYSTICK_CTRL_REG register to ensure the
-             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.
-             * Again, the time the SysTick is stopped for is accounted for as
-             * best it can be, but using the tickless mode will inevitably
-             * result in some tiny drift of the time maintained by the kernel
-             * with respect to calendar time*/
-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );
+             * portNVIC_SYSTICK_COUNT_FLAG_BIT is not cleared if it is set.  Again,
+             * the time the SysTick is stopped for is accounted for as best it can
+             * be, but using the tickless mode will inevitably result in some tiny
+             * drift of the time maintained by the kernel with respect to calendar
+             * time*/
+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );
 
-            /* Determine if the SysTick clock has already counted to zero and
-             * been set back to the current reload value (the reload back being
-             * correct for the entire expected idle time) or if the SysTick is
-             * yet to count to zero (in which case an interrupt other than the
-             * SysTick must have brought the system out of sleep mode). */
+            /* Determine whether the SysTick has already counted to zero. */
             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
             {
                 uint32_t ulCalculatedLoadValue;
 
-                /* The tick interrupt is already pending, and the SysTick count
-                 * reloaded with ulReloadValue.  Reset the
-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick
-                 * period. */
+                /* The tick interrupt ended the sleep (or is now pending), and
+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG
+                 * with whatever remains of the new tick period. */
                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );
 
                 /* Don't allow a tiny value, or values that have somehow
                  * underflowed because the post sleep hook did something
-                 * that took too long. */
-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
+                 * that took too long or because the SysTick current-value register
+                 * is zero. */
+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )
                 {
                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );
                 }
@@ -516,17 +528,36 @@
                 portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue;
 
                 /* As the pending tick will be processed as soon as this
-                 * function exits, the tick value maintained by the tick is
-                 * stepped forward by one less than the time spent waiting. */
+                 * function exits, the tick value maintained by the tick is stepped
+                 * forward by one less than the time spent waiting. */
                 ulCompleteTickPeriods = xExpectedIdleTime - 1UL;
             }
             else
             {
-                /* Something other than the tick interrupt ended the sleep.
-                 * Work out how long the sleep lasted rounded to complete tick
+                /* Something other than the tick interrupt ended the sleep. */
+
+                /* Use the SysTick current-value register to determine the
+                 * number of SysTick decrements remaining until the expected idle
+                 * time would have ended. */
+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )
+                {
+                    /* If the SysTick is not using the core clock, the current-
+                     * value register might still be zero here.  In that case, the
+                     * SysTick didn't load from the reload register, and there are
+                     * ulReloadValue decrements remaining in the expected idle
+                     * time, not zero. */
+                    if( ulSysTickDecrementsLeft == 0 )
+                    {
+                        ulSysTickDecrementsLeft = ulReloadValue;
+                    }
+                }
+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+                /* Work out how long the sleep lasted rounded to complete tick
                  * periods (not the ulReload value which accounted for part
                  * ticks). */
-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;
+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;
 
                 /* How many complete tick periods passed while the processor
                  * was waiting? */
@@ -537,13 +568,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;
             }
 
-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG
-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard
-             * value. */
+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,
+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If
+             * the SysTick is not using the core clock, temporarily configure it to
+             * use the core clock.  This configuration forces the SysTick to load
+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next
+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready
+             * to receive the standard value immediately. */
             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;
-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;
+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )
+            {
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+            }
+            #else
+            {
+                /* The temporary usage of the core clock has served its purpose,
+                 * as described above.  Resume usage of the other clock. */
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;
+
+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )
+                {
+                    /* The partial tick period already ended.  Be sure the SysTick
+                     * counts it only once. */
+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;
+                }
+
+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+            }
+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */
+
+            /* Step the tick to account for any tick periods that elapsed. */
             vTaskStepTick( ulCompleteTickPeriods );
-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;
 
             /* Exit with interrupts enabled. */
             __asm volatile ( "cpsie i" ::: "memory" );
@@ -556,11 +613,11 @@
 {
     /* Calculate the constants required to configure the tick interrupt. */
     #if ( configUSE_TICKLESS_IDLE == 1 )
-        {
-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
-        }
+    {
+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );
+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;
+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );
+    }
     #endif /* configUSE_TICKLESS_IDLE */
 
     /* Stop and reset the SysTick. */
@@ -569,7 +626,7 @@
 
     /* Configure SysTick to interrupt at the requested rate. */
     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;
-    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
+    portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;
 }
 /*-----------------------------------------------------------*/
 
@@ -694,10 +751,10 @@
     static void prvSetupFPU( void ) /* PRIVILEGED_FUNCTION */
     {
         #if ( configENABLE_TRUSTZONE == 1 )
-            {
-                /* Enable non-secure access to the FPU. */
-                SecureInit_EnableNSFPUAccess();
-            }
+        {
+            /* Enable non-secure access to the FPU. */
+            SecureInit_EnableNSFPUAccess();
+        }
         #endif /* configENABLE_TRUSTZONE */
 
         /* CP10 = 11 ==> Full access to FPU i.e. both privileged and
@@ -810,22 +867,22 @@
                 ulR0 = pulCallerStackAddress[ 0 ];
 
                 #if ( configENABLE_MPU == 1 )
-                    {
-                        /* Read the CONTROL register value. */
-                        __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
+                {
+                    /* Read the CONTROL register value. */
+                    __asm volatile ( "mrs %0, control"  : "=r" ( ulControl ) );
 
-                        /* The task that raised the SVC is privileged if Bit[0]
-                         * in the CONTROL register is 0. */
-                        ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
+                    /* The task that raised the SVC is privileged if Bit[0]
+                     * in the CONTROL register is 0. */
+                    ulIsTaskPrivileged = ( ( ulControl & portCONTROL_PRIVILEGED_MASK ) == 0 );
 
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
-                    }
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, ulIsTaskPrivileged, pxCurrentTCB );
+                }
                 #else /* if ( configENABLE_MPU == 1 ) */
-                    {
-                        /* Allocate and load a context for the secure task. */
-                        xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
-                    }
+                {
+                    /* Allocate and load a context for the secure task. */
+                    xSecureContext = SecureContext_AllocateContext( ulR0, pxCurrentTCB );
+                }
                 #endif /* configENABLE_MPU */
 
                 configASSERT( xSecureContext != securecontextINVALID_CONTEXT_ID );
@@ -833,6 +890,7 @@
                 break;
 
             case portSVC_FREE_SECURE_CONTEXT:
+
                 /* R0 contains TCB being freed and R1 contains the secure
                  * context handle to be freed. */
                 ulR0 = pulCallerStackAddress[ 0 ];
@@ -845,21 +903,21 @@
 
         case portSVC_START_SCHEDULER:
             #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    /* De-prioritize the non-secure exceptions so that the
-                     * non-secure pendSV runs at the lowest priority. */
-                    SecureInit_DePrioritizeNSExceptions();
+            {
+                /* De-prioritize the non-secure exceptions so that the
+                 * non-secure pendSV runs at the lowest priority. */
+                SecureInit_DePrioritizeNSExceptions();
 
-                    /* Initialize the secure context management system. */
-                    SecureContext_Init();
-                }
+                /* Initialize the secure context management system. */
+                SecureContext_Init();
+            }
             #endif /* configENABLE_TRUSTZONE */
 
             #if ( configENABLE_FPU == 1 )
-                {
-                    /* Setup the Floating Point Unit (FPU). */
-                    prvSetupFPU();
-                }
+            {
+                /* Setup the Floating Point Unit (FPU). */
+                prvSetupFPU();
+            }
             #endif /* configENABLE_FPU */
 
             /* Setup the context of the first task so that the first task starts
@@ -904,105 +962,105 @@
     /* Simulate the stack frame as it would be created by a context switch
      * interrupt. */
     #if ( portPRELOAD_REGISTERS == 0 )
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
+        *pxTopOfStack = portINITIAL_EXC_RETURN;
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack -= 5;                                       /* R12, R3, R2 and R1. */
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack -= 9;                                       /* R11..R4, EXC_RETURN. */
-            *pxTopOfStack = portINITIAL_EXC_RETURN;
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #else /* portPRELOAD_REGISTERS */
+    {
+        pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
+        *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
+        pxTopOfStack--;
+        *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
+
+        #if ( configENABLE_MPU == 1 )
         {
-            pxTopOfStack--;                                          /* Offset added to account for the way the MCU uses the stack on entry/exit of interrupts. */
-            *pxTopOfStack = portINITIAL_XPSR;                        /* xPSR */
             pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxCode;                  /* PC */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) portTASK_RETURN_ADDRESS; /* LR */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x12121212UL;            /* R12 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x03030303UL;            /* R3 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x02020202UL;            /* R2 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x01010101UL;            /* R1 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pvParameters;            /* R0 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x11111111UL;            /* R11 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x10101010UL;            /* R10 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x09090909UL;            /* R09 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x08080808UL;            /* R08 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x07070707UL;            /* R07 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x06060606UL;            /* R06 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x05050505UL;            /* R05 */
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) 0x04040404UL;            /* R04 */
-            pxTopOfStack--;
-            *pxTopOfStack = portINITIAL_EXC_RETURN;                  /* EXC_RETURN */
 
-            #if ( configENABLE_MPU == 1 )
-                {
-                    pxTopOfStack--;
-
-                    if( xRunPrivileged == pdTRUE )
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                    else
-                    {
-                        *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
-                    }
-                }
-            #endif /* configENABLE_MPU */
-
-            pxTopOfStack--;
-            *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
-
-            #if ( configENABLE_TRUSTZONE == 1 )
-                {
-                    pxTopOfStack--;
-                    *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
-                }
-            #endif /* configENABLE_TRUSTZONE */
+            if( xRunPrivileged == pdTRUE )
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_PRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
+            else
+            {
+                *pxTopOfStack = portINITIAL_CONTROL_UNPRIVILEGED; /* Slot used to hold this task's CONTROL value. */
+            }
         }
+        #endif /* configENABLE_MPU */
+
+        pxTopOfStack--;
+        *pxTopOfStack = ( StackType_t ) pxEndOfStack; /* Slot used to hold this task's PSPLIM value. */
+
+        #if ( configENABLE_TRUSTZONE == 1 )
+        {
+            pxTopOfStack--;
+            *pxTopOfStack = portNO_SECURE_CONTEXT; /* Slot used to hold this task's xSecureContext value. */
+        }
+        #endif /* configENABLE_TRUSTZONE */
+    }
     #endif /* portPRELOAD_REGISTERS */
 
     return pxTopOfStack;
@@ -1016,10 +1074,10 @@
     portNVIC_SHPR3_REG |= portNVIC_SYSTICK_PRI;
 
     #if ( configENABLE_MPU == 1 )
-        {
-            /* Setup the Memory Protection Unit (MPU). */
-            prvSetupMPU();
-        }
+    {
+        /* Setup the Memory Protection Unit (MPU). */
+        prvSetupMPU();
+    }
     #endif /* configENABLE_MPU */
 
     /* Start the timer that generates the tick ISR. Interrupts are disabled
diff --git a/portable/IAR/ARM_CM4F/port.c b/portable/IAR/ARM_CM4F/port.c
index a870d27..3c9e006 100644
--- a/portable/IAR/ARM_CM4F/port.c
+++ b/portable/IAR/ARM_CM4F/port.c
@@ -45,27 +45,18 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 /* Constants used to detect a Cortex-M7 r0p1 core, which should use the ARM_CM7

@@ -104,12 +95,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -247,66 +250,66 @@
     configASSERT( portCPUID != portCORTEX_M7_r0p0_ID );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -395,7 +398,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -404,22 +407,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_interrupt();

@@ -430,23 +417,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

+            /* Re-enable interrupts - see comments above the __disable_interrupt()

              * call above. */

             __enable_interrupt();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -475,8 +488,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_interrupt() call above. */

             __enable_interrupt();

             __DSB();

             __ISB();

@@ -496,27 +509,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -530,11 +539,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -545,13 +573,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_interrupt();

@@ -569,11 +623,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -582,7 +636,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/IAR/ARM_CM7/r0p1/port.c b/portable/IAR/ARM_CM7/r0p1/port.c
index 93ab626..a7300b2 100644
--- a/portable/IAR/ARM_CM7/r0p1/port.c
+++ b/portable/IAR/ARM_CM7/r0p1/port.c
@@ -45,27 +45,18 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -98,12 +89,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -235,66 +238,66 @@
     configASSERT( configMAX_SYSCALL_INTERRUPT_PRIORITY );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -383,7 +386,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -392,22 +395,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_interrupt();

@@ -418,23 +405,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_interrupt()

+            /* Re-enable interrupts - see comments above the __disable_interrupt()

              * call above. */

             __enable_interrupt();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -463,8 +476,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_interrupt() call above. */

             __enable_interrupt();

             __DSB();

             __ISB();

@@ -484,27 +497,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -518,11 +527,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -533,13 +561,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_interrupt();

@@ -557,11 +611,11 @@
 {

     /* Calculate the constants required to configure the tick interrupt. */

     #if ( configUSE_TICKLESS_IDLE == 1 )

-        {

-            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-        }

+    {

+        ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+        xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+        ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+    }

     #endif /* configUSE_TICKLESS_IDLE */

 

     /* Stop and clear the SysTick. */

@@ -570,7 +624,7 @@
 

     /* Configure SysTick to interrupt at the requested rate. */

     portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+    portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

 }

 /*-----------------------------------------------------------*/

 

diff --git a/portable/MikroC/ARM_CM4F/port.c b/portable/MikroC/ARM_CM4F/port.c
index 0281405..a8e7e7c 100644
--- a/portable/MikroC/ARM_CM4F/port.c
+++ b/portable/MikroC/ARM_CM4F/port.c
@@ -34,28 +34,18 @@
 #include "FreeRTOS.h"

 #include "task.h"

 

-

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Constants required to manipulate the core.  Registers first... */

 #define portNVIC_SYSTICK_CTRL_REG             ( *( ( volatile uint32_t * ) 0xe000e010 ) )

 #define portNVIC_SYSTICK_LOAD_REG             ( *( ( volatile uint32_t * ) 0xe000e014 ) )

 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -88,7 +78,19 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

+

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

 

 /* Let the user override the pre-loading of the initial LR with the address of

  * prvTaskExitError() in case it messes up unwinding of the stack in the

@@ -298,70 +300,70 @@
     configASSERT( configMAX_SYSCALL_INTERRUPT_PRIORITY );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* The kernel interrupt priority should be set to the lowest

+         * priority. */

+        configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( volatile uint8_t * const ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* The kernel interrupt priority should be set to the lowest

-             * priority. */

-            configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -510,7 +512,7 @@
 

     void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -519,22 +521,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __asm {

@@ -551,18 +537,7 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above the cpsid instruction()

+            /* Re-enable interrupts - see comments above the cpsid instruction

              * above. */

             __asm {

                 "cpsie i"

@@ -570,6 +545,43 @@
         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -604,8 +616,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the cpsid instruction above. */

             __asm {

                 "cpsie i"

             };

@@ -637,27 +649,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -671,11 +679,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -686,13 +713,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __asm {

@@ -714,11 +767,11 @@
     {

         /* Calculate the constants required to configure the tick interrupt. */

         #if ( configUSE_TICKLESS_IDLE == 1 )

-            {

-                ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-                xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-                ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-            }

+        {

+            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+        }

         #endif /* configUSE_TICKLESS_IDLE */

 

         /* Reset SysTick. */

@@ -727,7 +780,7 @@
 

         /* Configure SysTick to interrupt at the requested rate. */

         portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

     }

 

     #endif /* configOVERRIDE_DEFAULT_TICK_CONFIGURATION */

diff --git a/portable/RVDS/ARM_CM0/port.c b/portable/RVDS/ARM_CM0/port.c
index bb6dfcd..50cd9ba 100644
--- a/portable/RVDS/ARM_CM0/port.c
+++ b/portable/RVDS/ARM_CM0/port.c
@@ -45,6 +45,8 @@
 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVSET_BIT                ( 1UL << 28UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

+#define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 #define portMIN_INTERRUPT_PRIORITY            ( 255UL )

 #define portNVIC_PENDSV_PRI                   ( portMIN_INTERRUPT_PRIORITY << 16UL )

 #define portNVIC_SYSTICK_PRI                  ( portMIN_INTERRUPT_PRIORITY << 24UL )

@@ -59,12 +61,24 @@
  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

 #ifndef portMISSED_COUNTS_FACTOR

-    #define portMISSED_COUNTS_FACTOR    ( 45UL )

+    #define portMISSED_COUNTS_FACTOR    ( 94UL )

 #endif

 

 /* Constants used with memory barrier intrinsics. */

 #define portSY_FULL_READ_WRITE    ( 15 )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /* Legacy macro for backward compatibility only.  This macro used to be used to

  * replace the function that configures the clock used to generate the tick

  * interrupt (prvSetupTimerInterrupt()), but now the function is declared weak so

@@ -360,18 +374,20 @@
     {

         /* Calculate the constants required to configure the tick interrupt. */

         #if ( configUSE_TICKLESS_IDLE == 1 )

-            ulTimerCountsForOneTick = ( configCPU_CLOCK_HZ / configTICK_RATE_HZ );

+        {

+            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

             xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR;

+            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+        }

         #endif /* configUSE_TICKLESS_IDLE */

 

-        /* Stop and reset the SysTick. */

+        /* Stop and clear the SysTick. */

         portNVIC_SYSTICK_CTRL_REG = 0UL;

         portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

 

         /* Configure SysTick to interrupt at the requested rate. */

-        portNVIC_SYSTICK_LOAD_REG = ( configCPU_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-        portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+        portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

+        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

     }

 

 #endif /* configOVERRIDE_DEFAULT_TICK_CONFIGURATION */

@@ -381,7 +397,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -390,22 +406,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily. The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_irq();

@@ -416,23 +416,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_irq() call

-             * above. */

+            /* Re-enable interrupts - see comments above the __disable_irq()

+             * call above. */

             __enable_irq();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -443,10 +469,10 @@
             /* Restart SysTick. */

             portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

 

-            /* Sleep until something happens. configPRE_SLEEP_PROCESSING() can

+            /* Sleep until something happens.  configPRE_SLEEP_PROCESSING() can

              * set its parameter to 0 to indicate that its implementation contains

              * its own wait for interrupt or wait for event instruction, and so wfi

-             * should not be executed again. However, the original expected idle

+             * should not be executed again.  However, the original expected idle

              * time variable must remain unmodified, so a copy is taken. */

             xModifiableIdleTime = xExpectedIdleTime;

             configPRE_SLEEP_PROCESSING( xModifiableIdleTime );

@@ -461,8 +487,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately. see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_irq() call above. */

             __enable_irq();

             __dsb( portSY_FULL_READ_WRITE );

             __isb( portSY_FULL_READ_WRITE );

@@ -482,27 +508,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -516,11 +538,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -531,15 +572,41 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD

-             * again, then set portNVIC_SYSTICK_LOAD back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-            vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

 

-            /* Exit with interrpts enabled. */

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

+            vTaskStepTick( ulCompleteTickPeriods );

+

+            /* Exit with interrupts enabled. */

             __enable_irq();

         }

     }

diff --git a/portable/RVDS/ARM_CM3/port.c b/portable/RVDS/ARM_CM3/port.c
index 6b5efb2..b47d3b7 100644
--- a/portable/RVDS/ARM_CM3/port.c
+++ b/portable/RVDS/ARM_CM3/port.c
@@ -42,17 +42,6 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Legacy macro for backward compatibility only.  This macro used to be used to

  * replace the function that configures the clock used to generate the tick

  * interrupt (prvSetupTimerInterrupt()), but now the function is declared weak so

@@ -68,10 +57,12 @@
 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -99,12 +90,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -262,70 +265,70 @@
 BaseType_t xPortStartScheduler( void )

 {

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* The kernel interrupt priority should be set to the lowest

+         * priority. */

+        configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* The kernel interrupt priority should be set to the lowest

-             * priority. */

-            configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -450,7 +453,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -459,22 +462,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_irq();

@@ -485,23 +472,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_irq() call

-             * above. */

+            /* Re-enable interrupts - see comments above the __disable_irq()

+             * call above. */

             __enable_irq();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -530,8 +543,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_irq() call above. */

             __enable_irq();

             __dsb( portSY_FULL_READ_WRITE );

             __isb( portSY_FULL_READ_WRITE );

@@ -551,27 +564,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -585,11 +594,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -600,13 +628,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_irq();

@@ -627,11 +681,11 @@
     {

         /* Calculate the constants required to configure the tick interrupt. */

         #if ( configUSE_TICKLESS_IDLE == 1 )

-            {

-                ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-                xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-                ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-            }

+        {

+            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+        }

         #endif /* configUSE_TICKLESS_IDLE */

 

         /* Stop and clear the SysTick. */

@@ -640,7 +694,7 @@
 

         /* Configure SysTick to interrupt at the requested rate. */

         portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

     }

 

 #endif /* configOVERRIDE_DEFAULT_TICK_CONFIGURATION */

diff --git a/portable/RVDS/ARM_CM4F/port.c b/portable/RVDS/ARM_CM4F/port.c
index 443ec3e..40e3ac9 100644
--- a/portable/RVDS/ARM_CM4F/port.c
+++ b/portable/RVDS/ARM_CM4F/port.c
@@ -42,17 +42,6 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* Legacy macro for backward compatibility only.  This macro used to be used to

  * replace the function that configures the clock used to generate the tick

  * interrupt (prvSetupTimerInterrupt()), but now the function is declared weak so

@@ -68,10 +57,12 @@
 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 /* Constants used to detect a Cortex-M7 r0p1 core, which should use the ARM_CM7

@@ -110,12 +101,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -324,70 +327,70 @@
     configASSERT( portCPUID != portCORTEX_M7_r0p0_ID );

 

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* The kernel interrupt priority should be set to the lowest

+         * priority. */

+        configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* The kernel interrupt priority should be set to the lowest

-             * priority. */

-            configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -542,7 +545,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -551,22 +554,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_irq();

@@ -577,23 +564,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_irq() call

-             * above. */

+            /* Re-enable interrupts - see comments above the __disable_irq()

+             * call above. */

             __enable_irq();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -622,8 +635,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_irq() call above. */

             __enable_irq();

             __dsb( portSY_FULL_READ_WRITE );

             __isb( portSY_FULL_READ_WRITE );

@@ -643,27 +656,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -677,11 +686,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -692,13 +720,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_irq();

@@ -719,11 +773,11 @@
     {

         /* Calculate the constants required to configure the tick interrupt. */

         #if ( configUSE_TICKLESS_IDLE == 1 )

-            {

-                ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-                xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-                ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-            }

+        {

+            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+        }

         #endif /* configUSE_TICKLESS_IDLE */

 

         /* Stop and clear the SysTick. */

@@ -732,7 +786,7 @@
 

         /* Configure SysTick to interrupt at the requested rate. */

         portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

     }

 

 #endif /* configOVERRIDE_DEFAULT_TICK_CONFIGURATION */

diff --git a/portable/RVDS/ARM_CM7/r0p1/port.c b/portable/RVDS/ARM_CM7/r0p1/port.c
index 910214c..fe869df 100644
--- a/portable/RVDS/ARM_CM7/r0p1/port.c
+++ b/portable/RVDS/ARM_CM7/r0p1/port.c
@@ -42,17 +42,6 @@
     #error configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to 0.  See http: /*www.FreeRTOS.org/RTOS-Cortex-M3-M4.html */

 #endif

 

-#ifndef configSYSTICK_CLOCK_HZ

-    #define configSYSTICK_CLOCK_HZ      configCPU_CLOCK_HZ

-    /* Ensure the SysTick is clocked at the same frequency as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 1UL << 2UL )

-#else

-

-/* The way the SysTick is clocked is not modified in case it is not the same

- * as the core. */

-    #define portNVIC_SYSTICK_CLK_BIT    ( 0 )

-#endif

-

 /* The __weak attribute does not work as you might expect with the Keil tools

  * so the configOVERRIDE_DEFAULT_TICK_CONFIGURATION constant must be set to 1 if

  * the application writer wants to provide their own implementation of

@@ -68,10 +57,12 @@
 #define portNVIC_SYSTICK_CURRENT_VALUE_REG    ( *( ( volatile uint32_t * ) 0xe000e018 ) )

 #define portNVIC_SHPR3_REG                    ( *( ( volatile uint32_t * ) 0xe000ed20 ) )

 /* ...then bits in the registers. */

+#define portNVIC_SYSTICK_CLK_BIT              ( 1UL << 2UL )

 #define portNVIC_SYSTICK_INT_BIT              ( 1UL << 1UL )

 #define portNVIC_SYSTICK_ENABLE_BIT           ( 1UL << 0UL )

 #define portNVIC_SYSTICK_COUNT_FLAG_BIT       ( 1UL << 16UL )

 #define portNVIC_PENDSVCLEAR_BIT              ( 1UL << 27UL )

+#define portNVIC_PEND_SYSTICK_SET_BIT         ( 1UL << 26UL )

 #define portNVIC_PEND_SYSTICK_CLEAR_BIT       ( 1UL << 25UL )

 

 #define portNVIC_PENDSV_PRI                   ( ( ( uint32_t ) configKERNEL_INTERRUPT_PRIORITY ) << 16UL )

@@ -104,12 +95,24 @@
 /* A fiddle factor to estimate the number of SysTick counts that would have

  * occurred while the SysTick counter is stopped during tickless idle

  * calculations. */

-#define portMISSED_COUNTS_FACTOR              ( 45UL )

+#define portMISSED_COUNTS_FACTOR              ( 94UL )

 

 /* For strict compliance with the Cortex-M spec the task start address should

  * have bit-0 clear, as it is loaded into the PC on exit from an ISR. */

 #define portSTART_ADDRESS_MASK                ( ( StackType_t ) 0xfffffffeUL )

 

+/* Let the user override the default SysTick clock rate.  If defined by the

+ * user, this symbol must equal the SysTick clock rate when the CLK bit is 0 in the

+ * configuration register. */

+#ifndef configSYSTICK_CLOCK_HZ

+    #define configSYSTICK_CLOCK_HZ             ( configCPU_CLOCK_HZ )

+    /* Ensure the SysTick is clocked at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( portNVIC_SYSTICK_CLK_BIT )

+#else

+    /* Select the option to clock SysTick not at the same frequency as the core. */

+    #define portNVIC_SYSTICK_CLK_BIT_CONFIG    ( 0 )

+#endif

+

 /*

  * Setup the timer to generate the tick interrupts.  The implementation in this

  * file is weak to allow application writers to change the timer used to

@@ -308,70 +311,70 @@
 BaseType_t xPortStartScheduler( void )

 {

     #if ( configASSERT_DEFINED == 1 )

+    {

+        volatile uint32_t ulOriginalPriority;

+        volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

+        volatile uint8_t ucMaxPriorityValue;

+

+        /* Determine the maximum priority from which ISR safe FreeRTOS API

+         * functions can be called.  ISR safe functions are those that end in

+         * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

+         * ensure interrupt entry is as fast and simple as possible.

+         *

+         * Save the interrupt priority value that is about to be clobbered. */

+        ulOriginalPriority = *pucFirstUserPriorityRegister;

+

+        /* Determine the number of priority bits available.  First write to all

+         * possible bits. */

+        *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

+

+        /* Read the value back to see how many bits stuck. */

+        ucMaxPriorityValue = *pucFirstUserPriorityRegister;

+

+        /* The kernel interrupt priority should be set to the lowest

+         * priority. */

+        configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

+

+        /* Use the same mask on the maximum system call priority. */

+        ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

+

+        /* Calculate the maximum acceptable priority group value for the number

+         * of bits read back. */

+        ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

+

+        while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

         {

-            volatile uint32_t ulOriginalPriority;

-            volatile uint8_t * const pucFirstUserPriorityRegister = ( uint8_t * ) ( portNVIC_IP_REGISTERS_OFFSET_16 + portFIRST_USER_INTERRUPT_NUMBER );

-            volatile uint8_t ucMaxPriorityValue;

-

-            /* Determine the maximum priority from which ISR safe FreeRTOS API

-             * functions can be called.  ISR safe functions are those that end in

-             * "FromISR".  FreeRTOS maintains separate thread and ISR API functions to

-             * ensure interrupt entry is as fast and simple as possible.

-             *

-             * Save the interrupt priority value that is about to be clobbered. */

-            ulOriginalPriority = *pucFirstUserPriorityRegister;

-

-            /* Determine the number of priority bits available.  First write to all

-             * possible bits. */

-            *pucFirstUserPriorityRegister = portMAX_8_BIT_VALUE;

-

-            /* Read the value back to see how many bits stuck. */

-            ucMaxPriorityValue = *pucFirstUserPriorityRegister;

-

-            /* The kernel interrupt priority should be set to the lowest

-             * priority. */

-            configASSERT( ucMaxPriorityValue == ( configKERNEL_INTERRUPT_PRIORITY & ucMaxPriorityValue ) );

-

-            /* Use the same mask on the maximum system call priority. */

-            ucMaxSysCallPriority = configMAX_SYSCALL_INTERRUPT_PRIORITY & ucMaxPriorityValue;

-

-            /* Calculate the maximum acceptable priority group value for the number

-             * of bits read back. */

-            ulMaxPRIGROUPValue = portMAX_PRIGROUP_BITS;

-

-            while( ( ucMaxPriorityValue & portTOP_BIT_OF_BYTE ) == portTOP_BIT_OF_BYTE )

-            {

-                ulMaxPRIGROUPValue--;

-                ucMaxPriorityValue <<= ( uint8_t ) 0x01;

-            }

-

-            #ifdef __NVIC_PRIO_BITS

-                {

-                    /* Check the CMSIS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

-                }

-            #endif

-

-            #ifdef configPRIO_BITS

-                {

-                    /* Check the FreeRTOS configuration that defines the number of

-                     * priority bits matches the number of priority bits actually queried

-                     * from the hardware. */

-                    configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

-                }

-            #endif

-

-            /* Shift the priority group value back to its position within the AIRCR

-             * register. */

-            ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

-            ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

-

-            /* Restore the clobbered interrupt priority register to its original

-             * value. */

-            *pucFirstUserPriorityRegister = ulOriginalPriority;

+            ulMaxPRIGROUPValue--;

+            ucMaxPriorityValue <<= ( uint8_t ) 0x01;

         }

+

+        #ifdef __NVIC_PRIO_BITS

+        {

+            /* Check the CMSIS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == __NVIC_PRIO_BITS );

+        }

+        #endif

+

+        #ifdef configPRIO_BITS

+        {

+            /* Check the FreeRTOS configuration that defines the number of

+             * priority bits matches the number of priority bits actually queried

+             * from the hardware. */

+            configASSERT( ( portMAX_PRIGROUP_BITS - ulMaxPRIGROUPValue ) == configPRIO_BITS );

+        }

+        #endif

+

+        /* Shift the priority group value back to its position within the AIRCR

+         * register. */

+        ulMaxPRIGROUPValue <<= portPRIGROUP_SHIFT;

+        ulMaxPRIGROUPValue &= portPRIORITY_GROUP_MASK;

+

+        /* Restore the clobbered interrupt priority register to its original

+         * value. */

+        *pucFirstUserPriorityRegister = ulOriginalPriority;

+    }

     #endif /* configASSERT_DEFINED */

 

     /* Make PendSV and SysTick the lowest priority interrupts. */

@@ -528,7 +531,7 @@
 

     __weak void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )

     {

-        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements;

+        uint32_t ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements, ulSysTickDecrementsLeft;

         TickType_t xModifiableIdleTime;

 

         /* Make sure the SysTick reload value does not overflow the counter. */

@@ -537,22 +540,6 @@
             xExpectedIdleTime = xMaximumPossibleSuppressedTicks;

         }

 

-        /* Stop the SysTick momentarily.  The time the SysTick is stopped for

-         * is accounted for as best it can be, but using the tickless mode will

-         * inevitably result in some tiny drift of the time maintained by the

-         * kernel with respect to calendar time. */

-        portNVIC_SYSTICK_CTRL_REG &= ~portNVIC_SYSTICK_ENABLE_BIT;

-

-        /* Calculate the reload value required to wait xExpectedIdleTime

-         * tick periods.  -1 is used because this code will execute part way

-         * through one of the tick periods. */

-        ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

-

-        if( ulReloadValue > ulStoppedTimerCompensation )

-        {

-            ulReloadValue -= ulStoppedTimerCompensation;

-        }

-

         /* Enter a critical section but don't use the taskENTER_CRITICAL()

          * method as that will mask interrupts that should exit sleep mode. */

         __disable_irq();

@@ -563,23 +550,49 @@
          * to be unsuspended then abandon the low power entry. */

         if( eTaskConfirmSleepModeStatus() == eAbortSleep )

         {

-            /* Restart from whatever is left in the count register to complete

-             * this tick period. */

-            portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG;

-

-            /* Restart SysTick. */

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

-

-            /* Reset the reload register to the value required for normal tick

-             * periods. */

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

-

-            /* Re-enable interrupts - see comments above __disable_irq() call

-             * above. */

+            /* Re-enable interrupts - see comments above the __disable_irq()

+             * call above. */

             __enable_irq();

         }

         else

         {

+            /* Stop the SysTick momentarily.  The time the SysTick is stopped for

+             * is accounted for as best it can be, but using the tickless mode will

+             * inevitably result in some tiny drift of the time maintained by the

+             * kernel with respect to calendar time. */

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

+

+            /* Use the SysTick current-value register to determine the number of

+             * SysTick decrements remaining until the next tick interrupt.  If the

+             * current-value register is zero, then there are actually

+             * ulTimerCountsForOneTick decrements remaining, not zero, because the

+             * SysTick requests the interrupt when decrementing from 1 to 0. */

+            ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+

+            if( ulSysTickDecrementsLeft == 0 )

+            {

+                ulSysTickDecrementsLeft = ulTimerCountsForOneTick;

+            }

+

+            /* Calculate the reload value required to wait xExpectedIdleTime

+             * tick periods.  -1 is used because this code normally executes part

+             * way through the first tick period.  But if the SysTick IRQ is now

+             * pending, then clear the IRQ, suppressing the first tick, and correct

+             * the reload value to reflect that the second tick period is already

+             * underway.  The expected idle time is always at least two ticks. */

+            ulReloadValue = ulSysTickDecrementsLeft + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) );

+

+            if( ( portNVIC_INT_CTRL_REG & portNVIC_PEND_SYSTICK_SET_BIT ) != 0 )

+            {

+                portNVIC_INT_CTRL_REG = portNVIC_PEND_SYSTICK_CLEAR_BIT;

+                ulReloadValue -= ulTimerCountsForOneTick;

+            }

+

+            if( ulReloadValue > ulStoppedTimerCompensation )

+            {

+                ulReloadValue -= ulStoppedTimerCompensation;

+            }

+

             /* Set the new reload value. */

             portNVIC_SYSTICK_LOAD_REG = ulReloadValue;

 

@@ -608,8 +621,8 @@
             configPOST_SLEEP_PROCESSING( xExpectedIdleTime );

 

             /* Re-enable interrupts to allow the interrupt that brought the MCU

-             * out of sleep mode to execute immediately.  see comments above

-             * __disable_interrupt() call above. */

+             * out of sleep mode to execute immediately.  See comments above

+             * the __disable_irq() call above. */

             __enable_irq();

             __dsb( portSY_FULL_READ_WRITE );

             __isb( portSY_FULL_READ_WRITE );

@@ -629,27 +642,23 @@
              * be, but using the tickless mode will inevitably result in some tiny

              * drift of the time maintained by the kernel with respect to calendar

              * time*/

-            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT );

+            portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT );

 

-            /* Determine if the SysTick clock has already counted to zero and

-             * been set back to the current reload value (the reload back being

-             * correct for the entire expected idle time) or if the SysTick is yet

-             * to count to zero (in which case an interrupt other than the SysTick

-             * must have brought the system out of sleep mode). */

+            /* Determine whether the SysTick has already counted to zero. */

             if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

             {

                 uint32_t ulCalculatedLoadValue;

 

-                /* The tick interrupt is already pending, and the SysTick count

-                 * reloaded with ulReloadValue.  Reset the

-                 * portNVIC_SYSTICK_LOAD_REG with whatever remains of this tick

-                 * period. */

+                /* The tick interrupt ended the sleep (or is now pending), and

+                 * a new tick period has started.  Reset portNVIC_SYSTICK_LOAD_REG

+                 * with whatever remains of the new tick period. */

                 ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) - ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG );

 

                 /* Don't allow a tiny value, or values that have somehow

                  * underflowed because the post sleep hook did something

-                 * that took too long. */

-                if( ( ulCalculatedLoadValue < ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

+                 * that took too long or because the SysTick current-value register

+                 * is zero. */

+                if( ( ulCalculatedLoadValue <= ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue > ulTimerCountsForOneTick ) )

                 {

                     ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL );

                 }

@@ -663,11 +672,30 @@
             }

             else

             {

-                /* Something other than the tick interrupt ended the sleep.

-                 * Work out how long the sleep lasted rounded to complete tick

+                /* Something other than the tick interrupt ended the sleep. */

+

+                /* Use the SysTick current-value register to determine the

+                 * number of SysTick decrements remaining until the expected idle

+                 * time would have ended. */

+                ulSysTickDecrementsLeft = portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG != portNVIC_SYSTICK_CLK_BIT )

+                {

+                    /* If the SysTick is not using the core clock, the current-

+                     * value register might still be zero here.  In that case, the

+                     * SysTick didn't load from the reload register, and there are

+                     * ulReloadValue decrements remaining in the expected idle

+                     * time, not zero. */

+                    if( ulSysTickDecrementsLeft == 0 )

+                    {

+                        ulSysTickDecrementsLeft = ulReloadValue;

+                    }

+                }

+                #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+                /* Work out how long the sleep lasted rounded to complete tick

                  * periods (not the ulReload value which accounted for part

                  * ticks). */

-                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG;

+                ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - ulSysTickDecrementsLeft;

 

                 /* How many complete tick periods passed while the processor

                  * was waiting? */

@@ -678,13 +706,39 @@
                 portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1UL ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements;

             }

 

-            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG

-             * again, then set portNVIC_SYSTICK_LOAD_REG back to its standard

-             * value. */

+            /* Restart SysTick so it runs from portNVIC_SYSTICK_LOAD_REG again,

+             * then set portNVIC_SYSTICK_LOAD_REG back to its standard value.  If

+             * the SysTick is not using the core clock, temporarily configure it to

+             * use the core clock.  This configuration forces the SysTick to load

+             * from portNVIC_SYSTICK_LOAD_REG immediately instead of at the next

+             * cycle of the other clock.  Then portNVIC_SYSTICK_LOAD_REG is ready

+             * to receive the standard value immediately. */

             portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;

-            portNVIC_SYSTICK_CTRL_REG |= portNVIC_SYSTICK_ENABLE_BIT;

+            portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            #if ( portNVIC_SYSTICK_CLK_BIT_CONFIG == portNVIC_SYSTICK_CLK_BIT )

+            {

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+            }

+            #else

+            {

+                /* The temporary usage of the core clock has served its purpose,

+                 * as described above.  Resume usage of the other clock. */

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT;

+

+                if( ( portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT ) != 0 )

+                {

+                    /* The partial tick period already ended.  Be sure the SysTick

+                     * counts it only once. */

+                    portNVIC_SYSTICK_CURRENT_VALUE_REG = 0;

+                }

+

+                portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

+                portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT;

+            }

+            #endif /* portNVIC_SYSTICK_CLK_BIT_CONFIG */

+

+            /* Step the tick to account for any tick periods that elapsed. */

             vTaskStepTick( ulCompleteTickPeriods );

-            portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL;

 

             /* Exit with interrupts enabled. */

             __enable_irq();

@@ -705,11 +759,11 @@
     {

         /* Calculate the constants required to configure the tick interrupt. */

         #if ( configUSE_TICKLESS_IDLE == 1 )

-            {

-                ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

-                xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

-                ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

-            }

+        {

+            ulTimerCountsForOneTick = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ );

+            xMaximumPossibleSuppressedTicks = portMAX_24_BIT_NUMBER / ulTimerCountsForOneTick;

+            ulStoppedTimerCompensation = portMISSED_COUNTS_FACTOR / ( configCPU_CLOCK_HZ / configSYSTICK_CLOCK_HZ );

+        }

         #endif /* configUSE_TICKLESS_IDLE */

 

         /* Stop and clear the SysTick. */

@@ -718,7 +772,7 @@
 

         /* Configure SysTick to interrupt at the requested rate. */

         portNVIC_SYSTICK_LOAD_REG = ( configSYSTICK_CLOCK_HZ / configTICK_RATE_HZ ) - 1UL;

-        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

+        portNVIC_SYSTICK_CTRL_REG = ( portNVIC_SYSTICK_CLK_BIT_CONFIG | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT );

     }

 

 #endif /* configOVERRIDE_DEFAULT_TICK_CONFIGURATION */