Parallelism-UnsafeVarAccess

Do not access global variables or static members outside critical region

Required inputs: IR

Detect shared variables, which are accessed from concurrently running tasks or interrupt service routines without being explicitly sequenced via critical sections.

An issue is reported for a shared variable, that is potentially accessed in a way that is unsafe, meaning that it is concurrently read and written by different tasks/ISRs. Such an access is often called data race and may result in subtle misbehavior of the program. Example:

Non-compliantCompliant
struct S global = ...;

void task1(void)
{
    struct S local1 = ...;

    global = local;

}

void task2(void)
{

    struct S local2 = global;

    ...
}
struct S global = ...;

void task1(void)
{
    struct S local1 = ...;
    DisableAllInterrupts();
    global = local;
    EnableAllInterrupts();
}

void task2(void)
{
    DisableAllInterrupts();
    struct S local2 = global;
    EnableAllInterrupts();
    ...
}
In the non-compliant example, task1 reads global, storing the result in local1. This may occur concurrently with the access in task2 resulting in an undefined result in local2 (assuming the functions task1 and task2 are concurrent tasks). In the compliant example, proper mutual exclusion is achieved by the use of DisableAllInterrupts.

The following model for tasks and interrupt service routines is supported:

  1. A single-core execution model is supported.
  2. Tasks and ISRs are defined by the following properties. Refer to option partitions of this rule to configure them:
    Priority Pre-defined static priority of the task/ISR. Higher values indicate higher priority.
    Entries Set of one or more functions serving as the start functions of the task or ISR. All functions transitively called from these belong to the task/ISR.
    Guarded Property that indicates that this task/ISR will always run exclusively on the processor. It cannot be interrupted or preempted by another task or ISR.
  3. Synchronization and serialization is possible via a pair of disable/enable primitives. In the compliant example, DisableAllInterrupts() and EnableAllInterrupts() was used, but these primitives can be freely configured. After disable was called, no preemption and no interrupt may occur, until enable is called by the current task/ISR. Please refer to the options enter_critical_functions, enter_critical_macros, exit_critical_functions, and exit_critical_macros to configure the primitives.
  4. To omit critical sections in highest-priority tasks/ISRs, please enable option strict_priorities. For example, in the compliant example, the critical section in task2 could be removed if task2 had higher priority than task1.
  5. Access to certain global variables can be excluded from the data race check, if the access to those global variables is considered atomic. Refer to option treat_types_as_atomic to configure data types of global variables that are accessed atomically. Also consider options allow_volatile_sig_atomic_t and allow_c11_atomics.
  6. By default, only pairs of reading and writing access occurring concurrently are considered to cause data race issues, refer to option show_identical_access to also report writing/writing and even reading/reading pairs.

Possible Messages

Key

Text

Severity

Disabled

multiple_lock_add

Lock is acquired while it is already locked.

None

False

non_recursive_lock_add

Non-recursive lock is acquired while it is already locked.

None

False

removed_nonexisting_lock

Lock is released, although it is not currently locked.

None

False

unbalanced_locks_path

Different control flow paths have different sets of locks.

None

False

unbalanced_locks_routine

Routine may return with different lock set than it is entered with ({in_set} vs {out_set}).

None

False

Options

access_kinds

access_kinds : set[bauhaus.ir.LIR_Class_Name] = {'Reading_Operand_Interface', 'Writing_Operand_Interface'}

Access kinds (e.g. Reading_Operand_Interface, Writing_Operand_Interface, Address_Operand_Interface).
 

allow_c11_atomics

allow_c11_atomics : bool = True

If set, do not report races on C11 atomic variables.
 

allow_volatile_sig_atomic_t

allow_volatile_sig_atomic_t : bool = False

If set, do not report races on variables of type volatile sig_atomic_t.
 

debug_output

debug_output : bool = False

Option to provide diagnostic output.
 

enter_critical_functions

enter_critical_functions

Type: set[bauhaus.analysis.config.QualifiedName]

Default: {'EnterCriticalSection', 'mtx_lock', 'pthread_mutex_lock', 'std::_Mutex_base::lock', 'std::mutex::lock'}

Set of function names to enter a critical region.
 

enter_critical_macros

enter_critical_macros : set[bauhaus.analysis.config.MacroName] = set()

Set of macro names to enter a critical region (macros must expand to asm() statement).
 

excluded_routines

excluded_routines : set[bauhaus.analysis.config.QualifiedName] = set()

Set of functions that should be excluded from check.
 

excluded_subgraphs

excluded_subgraphs : set[bauhaus.analysis.config.QualifiedName] = set()

Set of entry functions to subgraphs that should be excluded as subgraph from check.
 

exit_critical_functions

exit_critical_functions

Type: set[bauhaus.analysis.config.QualifiedName]

Default: {'ExitCriticalSection', 'mtx_unlock', 'pthread_mutex_unlock', 'std::_Mutex_base::unlock', 'std::mutex::unlock'}

Set of function names to exit a critical region.
 

exit_critical_macros

exit_critical_macros : set[bauhaus.analysis.config.MacroName] = set()

Set of macro names to exit a critical region (macros must expand to asm() statement).
 

inspect_pointers

inspect_pointers : bool = False

Whether pointer targets should be inspected to detect more global variable uses.
 

nested_critical_regions

nested_critical_regions : bool = True

If set to true, critical regions nest; if set to false, a single exit-critical-region terminates all open critical regions.
 

output_safe_accesses

output_safe_accesses : bool = False

When enabled, outputs not only unsafe variable accesses, but also the safe ones.
 

partitions

partitions : dict[str, dict[str, typing.Any]] = {}

Dict with partition name as key and dict as value. Partitions describe parts of the IR graph that can be run as a task or an interrupt service routine. The partition dict can contain keys as follows:
  1. entries: list of entry functions or this task/isr
  2. functions_passed_to: name of thread creation function. Any function designated by a pointer passed to that function will be considered an entry function.
  3. vectors: list of global variable names with function pointers to entry functions or this task/ISR
  4. guarded: boolean property. Set to True if this task is nonpreemptive and cannot be interrupted by interrupt handlers. Set to False or omit otherwise (default).
The special partition name __interrupts__ will automatically contain all interrupt handlers recorded as Additional_Entries in IR (see compiler toolchain's advanced.main_entries configuration) in addition to any entries specified in its dict.
 

report_cfg_based_critical_region_issues

report_cfg_based_critical_region_issues : bool = False

Report unbalanced lock/unlock pairs within a routine. This has the same intention, but is slightly less strict than the purely syntactic check performed by the rule Parallelism-IncorrectCriticalRegion.
 

show_identical_access

show_identical_access : bool = False

When enabled, outputs variable accesses of same kind (i.e., R/R and W/W).
 

show_object_number

show_object_number : bool = False

Option for debugging (shows internal node numbers). Can be used to generate call graphs for data race visualization.
 

strict_priorities

strict_priorities : bool = False

Set to true if a higher-priority task/ISR can only be preempted by a task/ISR of strictly higher priority. This has the effect that critical regions can be omitted in the highest-priority task/ISR if all accesses are from tasks/ISRs on the same core.
 

treat_types_as_atomic

treat_types_as_atomic : set[typing.Pattern[str] | typing.Tuple[typing.Optional[int], typing.Optional[int], typing.Optional[typing.Pattern[str]]]] = set()

Set of type-patterns. A type-pattern is either a regular expression of a type name, or a triple of (min. alignment, max. size, type name-regex). Each of the triple's components may be None. None is interpreted as general wildcard.