Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependents: blinky_max32630fthr
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 /** @}*/
Generated on Tue Jul 12 2022 14:21:02 by
 1.7.2
 1.7.2 
    