Greg Steiert / pegasus_dev

Dependents:   blinky_max32630fthr

Embed: (wiki syntax)

« Back to documentation index

Show/hide line numbers critical.h Source File

critical.h

00001 
00002 /** \addtogroup platform */
00003 /** @{*/
00004 /*
00005  * Copyright (c) 2015-2016, ARM Limited, All Rights Reserved
00006  * SPDX-License-Identifier: Apache-2.0
00007  *
00008  * Licensed under the Apache License, Version 2.0 (the "License"); you may
00009  * not use this file except in compliance with the License.
00010  * You may obtain a copy of the License at
00011  *
00012  * http://www.apache.org/licenses/LICENSE-2.0
00013  *
00014  * Unless required by applicable law or agreed to in writing, software
00015  * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
00016  * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
00017  * See the License for the specific language governing permissions and
00018  * limitations under the License.
00019  */
00020 
00021 #ifndef __MBED_UTIL_CRITICAL_H__
00022 #define __MBED_UTIL_CRITICAL_H__
00023 
00024 #include <stdbool.h>
00025 #include <stdint.h>
00026 #include <stddef.h>
00027 
00028 #ifdef __cplusplus
00029 extern "C" {
00030 #endif
00031 
00032 
00033 /** Determine the current interrupts enabled state
00034   *
00035   * This function can be called to determine whether or not interrupts are currently enabled.
00036   * \note
00037   * NOTE:
00038   * This function works for both cortex-A and cortex-M, although the underlyng implementation
00039   * differs.
00040   * @return true if interrupts are enabled, false otherwise
00041   */
00042 bool core_util_are_interrupts_enabled(void);
00043 
00044 /** Mark the start of a critical section
00045   *
00046   * This function should be called to mark the start of a critical section of code.
00047   * \note
00048   * NOTES:
00049   * 1) The use of this style of critical section is targetted at C based implementations.
00050   * 2) These critical sections can be nested.
00051   * 3) The interrupt enable state on entry to the first critical section (of a nested set, or single
00052   *    section) will be preserved on exit from the section.
00053   * 4) This implementation will currently only work on code running in privileged mode.
00054   */
00055 void core_util_critical_section_enter(void);
00056 
00057 /** Mark the end of a critical section
00058   *
00059   * This function should be called to mark the end of a critical section of code.
00060   * \note
00061   * NOTES:
00062   * 1) The use of this style of critical section is targetted at C based implementations.
00063   * 2) These critical sections can be nested.
00064   * 3) The interrupt enable state on entry to the first critical section (of a nested set, or single
00065   *    section) will be preserved on exit from the section.
00066   * 4) This implementation will currently only work on code running in privileged mode.
00067   */
00068 void core_util_critical_section_exit(void);
00069 
00070 /**
00071  * Atomic compare and set. It compares the contents of a memory location to a
00072  * given value and, only if they are the same, modifies the contents of that
00073  * memory location to a given new value. This is done as a single atomic
00074  * operation. The atomicity guarantees that the new value is calculated based on
00075  * up-to-date information; if the value had been updated by another thread in
00076  * the meantime, the write would fail due to a mismatched expectedCurrentValue.
00077  *
00078  * Refer to https://en.wikipedia.org/wiki/Compare-and-set [which may redirect
00079  * you to the article on compare-and swap].
00080  *
00081  * @param  ptr                  The target memory location.
00082  * @param[in,out] expectedCurrentValue A pointer to some location holding the
00083  *                              expected current value of the data being set atomically.
00084  *                              The computed 'desiredValue' should be a function of this current value.
00085  *                              @Note: This is an in-out parameter. In the
00086  *                              failure case of atomic_cas (where the
00087  *                              destination isn't set), the pointee of expectedCurrentValue is
00088  *                              updated with the current value.
00089  * @param[in] desiredValue      The new value computed based on '*expectedCurrentValue'.
00090  *
00091  * @return                      true if the memory location was atomically
00092  *                              updated with the desired value (after verifying
00093  *                              that it contained the expectedCurrentValue),
00094  *                              false otherwise. In the failure case,
00095  *                              exepctedCurrentValue is updated with the new
00096  *                              value of the target memory location.
00097  *
00098  * pseudocode:
00099  * function cas(p : pointer to int, old : pointer to int, new : int) returns bool {
00100  *     if *p != *old {
00101  *         *old = *p
00102  *         return false
00103  *     }
00104  *     *p = new
00105  *     return true
00106  * }
00107  *
00108  * @Note: In the failure case (where the destination isn't set), the value
00109  * pointed to by expectedCurrentValue is still updated with the current value.
00110  * This property helps writing concise code for the following incr:
00111  *
00112  * function incr(p : pointer to int, a : int) returns int {
00113  *     done = false
00114  *     value = *p // This fetch operation need not be atomic.
00115  *     while not done {
00116  *         done = atomic_cas(p, &value, value + a) // *value gets updated automatically until success
00117  *     }
00118  *     return value + a
00119  * }
00120  */
00121 bool core_util_atomic_cas_u8(uint8_t *ptr, uint8_t *expectedCurrentValue, uint8_t desiredValue);
00122 
00123 /**
00124  * Atomic compare and set. It compares the contents of a memory location to a
00125  * given value and, only if they are the same, modifies the contents of that
00126  * memory location to a given new value. This is done as a single atomic
00127  * operation. The atomicity guarantees that the new value is calculated based on
00128  * up-to-date information; if the value had been updated by another thread in
00129  * the meantime, the write would fail due to a mismatched expectedCurrentValue.
00130  *
00131  * Refer to https://en.wikipedia.org/wiki/Compare-and-set [which may redirect
00132  * you to the article on compare-and swap].
00133  *
00134  * @param  ptr                  The target memory location.
00135  * @param[in,out] expectedCurrentValue A pointer to some location holding the
00136  *                              expected current value of the data being set atomically.
00137  *                              The computed 'desiredValue' should be a function of this current value.
00138  *                              @Note: This is an in-out parameter. In the
00139  *                              failure case of atomic_cas (where the
00140  *                              destination isn't set), the pointee of expectedCurrentValue is
00141  *                              updated with the current value.
00142  * @param[in] desiredValue      The new value computed based on '*expectedCurrentValue'.
00143  *
00144  * @return                      true if the memory location was atomically
00145  *                              updated with the desired value (after verifying
00146  *                              that it contained the expectedCurrentValue),
00147  *                              false otherwise. In the failure case,
00148  *                              exepctedCurrentValue is updated with the new
00149  *                              value of the target memory location.
00150  *
00151  * pseudocode:
00152  * function cas(p : pointer to int, old : pointer to int, new : int) returns bool {
00153  *     if *p != *old {
00154  *         *old = *p
00155  *         return false
00156  *     }
00157  *     *p = new
00158  *     return true
00159  * }
00160  *
00161  * @Note: In the failure case (where the destination isn't set), the value
00162  * pointed to by expectedCurrentValue is still updated with the current value.
00163  * This property helps writing concise code for the following incr:
00164  *
00165  * function incr(p : pointer to int, a : int) returns int {
00166  *     done = false
00167  *     value = *p // This fetch operation need not be atomic.
00168  *     while not done {
00169  *         done = atomic_cas(p, &value, value + a) // *value gets updated automatically until success
00170  *     }
00171  *     return value + a
00172  * }
00173  */
00174 bool core_util_atomic_cas_u16(uint16_t *ptr, uint16_t *expectedCurrentValue, uint16_t desiredValue);
00175 
00176 /**
00177  * Atomic compare and set. It compares the contents of a memory location to a
00178  * given value and, only if they are the same, modifies the contents of that
00179  * memory location to a given new value. This is done as a single atomic
00180  * operation. The atomicity guarantees that the new value is calculated based on
00181  * up-to-date information; if the value had been updated by another thread in
00182  * the meantime, the write would fail due to a mismatched expectedCurrentValue.
00183  *
00184  * Refer to https://en.wikipedia.org/wiki/Compare-and-set [which may redirect
00185  * you to the article on compare-and swap].
00186  *
00187  * @param  ptr                  The target memory location.
00188  * @param[in,out] expectedCurrentValue A pointer to some location holding the
00189  *                              expected current value of the data being set atomically.
00190  *                              The computed 'desiredValue' should be a function of this current value.
00191  *                              @Note: This is an in-out parameter. In the
00192  *                              failure case of atomic_cas (where the
00193  *                              destination isn't set), the pointee of expectedCurrentValue is
00194  *                              updated with the current value.
00195  * @param[in] desiredValue      The new value computed based on '*expectedCurrentValue'.
00196  *
00197  * @return                      true if the memory location was atomically
00198  *                              updated with the desired value (after verifying
00199  *                              that it contained the expectedCurrentValue),
00200  *                              false otherwise. In the failure case,
00201  *                              exepctedCurrentValue is updated with the new
00202  *                              value of the target memory location.
00203  *
00204  * pseudocode:
00205  * function cas(p : pointer to int, old : pointer to int, new : int) returns bool {
00206  *     if *p != *old {
00207  *         *old = *p
00208  *         return false
00209  *     }
00210  *     *p = new
00211  *     return true
00212  * }
00213  *
00214  * @Note: In the failure case (where the destination isn't set), the value
00215  * pointed to by expectedCurrentValue is still updated with the current value.
00216  * This property helps writing concise code for the following incr:
00217  *
00218  * function incr(p : pointer to int, a : int) returns int {
00219  *     done = false
00220  *     value = *p // This fetch operation need not be atomic.
00221  *     while not done {
00222  *         done = atomic_cas(p, &value, value + a) // *value gets updated automatically until success
00223  *     }
00224  *     return value + a
00225  * }
00226  */
00227 bool core_util_atomic_cas_u32(uint32_t *ptr, uint32_t *expectedCurrentValue, uint32_t desiredValue);
00228 
00229 /**
00230  * Atomic compare and set. It compares the contents of a memory location to a
00231  * given value and, only if they are the same, modifies the contents of that
00232  * memory location to a given new value. This is done as a single atomic
00233  * operation. The atomicity guarantees that the new value is calculated based on
00234  * up-to-date information; if the value had been updated by another thread in
00235  * the meantime, the write would fail due to a mismatched expectedCurrentValue.
00236  *
00237  * Refer to https://en.wikipedia.org/wiki/Compare-and-set [which may redirect
00238  * you to the article on compare-and swap].
00239  *
00240  * @param  ptr                  The target memory location.
00241  * @param[in,out] expectedCurrentValue A pointer to some location holding the
00242  *                              expected current value of the data being set atomically.
00243  *                              The computed 'desiredValue' should be a function of this current value.
00244  *                              @Note: This is an in-out parameter. In the
00245  *                              failure case of atomic_cas (where the
00246  *                              destination isn't set), the pointee of expectedCurrentValue is
00247  *                              updated with the current value.
00248  * @param[in] desiredValue      The new value computed based on '*expectedCurrentValue'.
00249  *
00250  * @return                      true if the memory location was atomically
00251  *                              updated with the desired value (after verifying
00252  *                              that it contained the expectedCurrentValue),
00253  *                              false otherwise. In the failure case,
00254  *                              exepctedCurrentValue is updated with the new
00255  *                              value of the target memory location.
00256  *
00257  * pseudocode:
00258  * function cas(p : pointer to int, old : pointer to int, new : int) returns bool {
00259  *     if *p != *old {
00260  *         *old = *p
00261  *         return false
00262  *     }
00263  *     *p = new
00264  *     return true
00265  * }
00266  *
00267  * @Note: In the failure case (where the destination isn't set), the value
00268  * pointed to by expectedCurrentValue is still updated with the current value.
00269  * This property helps writing concise code for the following incr:
00270  *
00271  * function incr(p : pointer to int, a : int) returns int {
00272  *     done = false
00273  *     value = *p // This fetch operation need not be atomic.
00274  *     while not done {
00275  *         done = atomic_cas(p, &value, value + a) // *value gets updated automatically until success
00276  *     }
00277  *     return value + a
00278  * }
00279  */
00280 bool core_util_atomic_cas_ptr(void **ptr, void **expectedCurrentValue, void *desiredValue);
00281 
00282 /**
00283  * Atomic increment.
00284  * @param  valuePtr Target memory location being incremented.
00285  * @param  delta    The amount being incremented.
00286  * @return          The new incremented value.
00287  */
00288 uint8_t core_util_atomic_incr_u8(uint8_t *valuePtr, uint8_t delta);
00289 
00290 /**
00291  * Atomic increment.
00292  * @param  valuePtr Target memory location being incremented.
00293  * @param  delta    The amount being incremented.
00294  * @return          The new incremented value.
00295  */
00296 uint16_t core_util_atomic_incr_u16(uint16_t *valuePtr, uint16_t delta);
00297 
00298 /**
00299  * Atomic increment.
00300  * @param  valuePtr Target memory location being incremented.
00301  * @param  delta    The amount being incremented.
00302  * @return          The new incremented value.
00303  */
00304 uint32_t core_util_atomic_incr_u32(uint32_t *valuePtr, uint32_t delta);
00305 
00306 /**
00307  * Atomic increment.
00308  * @param  valuePtr Target memory location being incremented.
00309  * @param  delta    The amount being incremented in bytes.
00310  * @return          The new incremented value.
00311  *
00312  * @note The type of the pointer argument is not taken into account
00313  *       and the pointer is incremented by bytes.
00314  */
00315 void *core_util_atomic_incr_ptr(void **valuePtr, ptrdiff_t delta);
00316 
00317 /**
00318  * Atomic decrement.
00319  * @param  valuePtr Target memory location being decremented.
00320  * @param  delta    The amount being decremented.
00321  * @return          The new decremented value.
00322  */
00323 uint8_t core_util_atomic_decr_u8(uint8_t *valuePtr, uint8_t delta);
00324 
00325 /**
00326  * Atomic decrement.
00327  * @param  valuePtr Target memory location being decremented.
00328  * @param  delta    The amount being decremented.
00329  * @return          The new decremented value.
00330  */
00331 uint16_t core_util_atomic_decr_u16(uint16_t *valuePtr, uint16_t delta);
00332 
00333 /**
00334  * Atomic decrement.
00335  * @param  valuePtr Target memory location being decremented.
00336  * @param  delta    The amount being decremented.
00337  * @return          The new decremented value.
00338  */
00339 uint32_t core_util_atomic_decr_u32(uint32_t *valuePtr, uint32_t delta);
00340 
00341 /**
00342  * Atomic decrement.
00343  * @param  valuePtr Target memory location being decremented.
00344  * @param  delta    The amount being decremented in bytes.
00345  * @return          The new decremented value.
00346  *
00347  * @note The type of the pointer argument is not taken into account
00348  *       and the pointer is decremented by bytes
00349  */
00350 void *core_util_atomic_decr_ptr(void **valuePtr, ptrdiff_t delta);
00351 
00352 #ifdef __cplusplus
00353 } // extern "C"
00354 #endif
00355 
00356 
00357 #endif // __MBED_UTIL_CRITICAL_H__
00358 
00359 /** @}*/