Partage via


Annotation du comportement de verrouillage

Pour éviter les bogues d'accès concurrentiel dans votre programme multithread, suivez toujours une règle de verrouillage appropriée et utilisez les annotations SAL.

Il est notoirement difficile de reproduire, de diagnostiquer et de déboguer des bogues d'accès concurrentiel en raison de leur nature non déterministe. Comprendre les entrelacements de thread est au mieux difficile, et devient impossible lorsque vous concevez un corps de code qui compte plus que quelques threads. Par conséquent, il est recommandé de suivre une discipline de verrouillage dans vos programmes multithread. Par exemple, obéir à un ordre de verrou tout en obtenant plusieurs verrous permet d'éviter des interblocages, et acquérir le verrou de garde approprié avant d'accéder à une ressource partagée permet d'éviter des conditions de concurrence.

Malheureusement, ces règles de verrouillage apparemment simples peuvent être étonnamment difficiles à suivre en pratique. Une limite fondamentale des langages de programmation et compilateurs actuels est qu’ils ne prennent pas directement en charge la spécification et l’analyse des exigences de concurrence. Les programmeurs doivent s'appuyer sur les commentaires de code informels pour exprimer leurs intentions sur la façon dont ils utilisent des verrous.

Les annotations SAL d'accès concurrentiel sont conçues pour vous aider à spécifier les effets secondaires du verrouillage, le responsable du verrouillage, la tutelle de données, la hiérarchie d'ordre de verrou et d'autres comportements de verrouillage prévus. En transformant des règles implicites en règles explicites, les annotations d'accès concurrence SAL vous permettent de documenter la façon dont votre code utilise des règles de verrouillage. Les annotations d'accès concurrentiel améliorent également la capacité des outils d'analyse du code à rechercher des conditions de concurrence, des interblocages, des opérations de synchronisation incompatibles et d'autres erreurs subtiles d'accès concurrentiel.

Instructions générales

Les annotations permettent de déclarer les contrats sous-entendus par les définitions de fonction entre les implémentations (appelés) et les clients (appelants). Elles permettent également d'exprimer les invariants et d'autres propriétés du programme qui peuvent améliorer davantage l'analyse.

SAL prend en charge de nombreux types de primitives de verrouillage, tels que les sections critiques, les mutex, les verrous de rotation et d'autres objets de ressource. De nombreuses annotations d’accès concurrentiel prennent une expression de verrou en tant que paramètre. Par convention, un verrou est indiqué par l’expression de chemin d’accès de l’objet de verrou sous-jacent.

Quelques règles de propriété concernant les threads à garder en tête :

  • Les verrous de rotation sont des verrous non numérotés qui ont une propriété de thread précise.

  • Les mutex et les sections critiques sont des verrous numérotés qui ont une propriété de thread précise.

  • Les sémaphores et les événements sont des verrous numérotés qui n'ont pas de propriété de thread précise.

Annotations de verrouillage

Le tableau suivant répertorie les annotations de verrouillage.

Annotation Description
_Acquires_exclusive_lock_(expr) Annote une fonction et indique qu’à l’état postérieur la fonction incrémente d’une unité le nombre de verrous exclusifs de l’objet verrou nommé par expr.
_Acquires_lock_(expr) Annote une fonction et indique qu’à l’état postérieur la fonction incrémente d’une unité le nombre de verrous de l’objet verrou nommé par expr.
_Acquires_nonreentrant_lock_(expr) Le verrou nommé par expr est acquis. Une erreur est signalée si le verrou est déjà détenu.
_Acquires_shared_lock_(expr) Annote une fonction et indique qu'à l'état postérieur la fonction incrémente d'une unité le nombre de verrous partagés de l'objet verrou nommé par expr.
_Create_lock_level_(name) Instruction qui déclare le symbole name comme un niveau de verrou afin qu'il puisse être utilisé dans les annotations _Has_Lock_level_ et _Lock_level_order_.
_Has_lock_kind_(kind) Annote n’importe quel objet pour affiner les informations de type d’un objet de ressource. Parfois, un type commun est utilisé pour différents types de ressources et le type surchargé n’est pas suffisant pour distinguer les exigences sémantiques entre différentes ressources. Voici la liste des paramètres kind prédéfinis :

_Lock_kind_mutex_
ID de type de verrou pour les mutex.

_Lock_kind_event_
ID de type de verrou pour les événements.

_Lock_kind_semaphore_
ID de type de verrou pour les sémaphores.

_Lock_kind_spin_lock_
ID de type de verrou pour les spin-locks.

_Lock_kind_critical_section_
ID de type de verrou pour les sections critiques.
_Has_lock_level_(name) Annote un objet verrou, en lui donnant le niveau de verrou de name.
_Lock_level_order_(name1, name2) Instruction qui donne l’ordre des verrous entre name1 et name2. Les verrous qui ont un niveau name1 doivent être acquis avant les verrous qui ont le niveau name2.
_Post_same_lock_(dst, src) Annote une fonction et indique, qu’à l’état postérieur, les deux verrous, dst et src, sont traités comme s’ils étaient le même objet verrou, en appliquant les propriétés de verrou de src à dst.
_Releases_exclusive_lock_(expr) Annote une fonction et indique qu'à l'état postérieur la fonction décrémente d'une unité le nombre de verrous exclusifs de l'objet verrou nommé par expr.
_Releases_lock_(expr) Annote une fonction et indique qu’à l’état postérieur la fonction décrémente d’une unité le nombre de verrous de l’objet verrou nommé par expr.
_Releases_nonreentrant_lock_(expr) Le verrou nommé par expr est libéré. Une erreur est signalée si le verrou n'est pas actuellement détenu.
_Releases_shared_lock_(expr) Annote une fonction et indique qu'à l'état postérieur la fonction décrémente d'une unité le nombre de verrous partagés de l'objet verrou nommé par expr.
_Requires_lock_held_(expr) Annote une fonction et indique qu'à l'état antérieur le nombre de verrous de l'objet nommé par expr est d'au moins un.
_Requires_lock_not_held_(expr) Annote une fonction et indique qu'à l'état antérieur le nombre de verrous de l'objet nommé par expr est zéro.
_Requires_no_locks_held_ Annote une fonction et indique que le nombre de verrous de tous les verrous connus par le vérificateur est égal à zéro.
_Requires_shared_lock_held_(expr) Annote une fonction et indique qu'à l'état antérieur le nombre de verrous partagés de l'objet nommé par expr est d'au moins un.
_Requires_exclusive_lock_held_(expr) Annote une fonction et indique qu'à l'état antérieur le nombre de verrous exclusifs de l'objet nommé par expr est d'au moins un.

Intrinsèques SAL pour les objets de verrouillage non exposés

Certains objets verrous ne sont pas exposés par l’implémentation des fonctions de verrouillage associées. Le tableau suivant répertorie les variables intrinsèques SAL qui autorisent les annotations sur les fonctions qui s'exécutent sur ces objets verrous non exposés.

Annotation Description
_Global_cancel_spin_lock_ Décrit le verrou d'annulation de rotation.
_Global_critical_region_ Décrit la zone critique.
_Global_interlock_ Décrit les opérations verrouillées.
_Global_priority_region_ Décrit la zone de priorité.

Annotations d’accès aux données partagées

Le tableau suivant répertorie les annotations à utiliser avec l'accès aux données partagées.

Annotation Description
_Guarded_by_(expr) Annote une variable et indique qu’à chaque accès à la variable, le nombre de verrous de l’objet verrou nommé par expr est d’au moins un.
_Interlocked_ Annote une variable et équivaut à _Guarded_by_(_Global_interlock_).
_Interlocked_operand_ Le paramètre de fonction annoté est l’opérande cible de l’une des différentes fonctions Interlocked. Ces opérandes doivent avoir d’autres propriétés spécifiques.
_Write_guarded_by_(expr) Annote une variable et indique qu'à chaque modification de la variable, le nombre de verrous de l'objet verrou nommé par expr est d'au moins un.

Annotations de verrou intelligent et RAII

Les verrous intelligents encapsulent généralement les verrous natifs et gèrent leur durée de vie. Le tableau suivant répertorie les annotations qui peuvent être utilisées avec des verrous intelligents et des modèles de codage RAII (Resource Acquisition Is Initialization) avec prise en charge de la sémantique move.

Annotation Description
_Analysis_assume_smart_lock_acquired_(lock) Indique à l’analyseur de supposer qu’un verrou intelligent a été acquis. Cette annotation attend un type de verrou de référence comme paramètre.
_Analysis_assume_smart_lock_released_(lock) Indique à l’analyseur de supposer qu’un verrou intelligent a été débloqué. Cette annotation attend un type de verrou de référence comme paramètre.
_Moves_lock_(target, source) Décrit une opération move constructor, qui transfère l’état de verrouillage de l’objet source à l’objet target. L'objet target est considéré comme un nouvel objet ; son état précédent est donc perdu et remplacé par l’état source . Le source est également réinitialisé sur un état vierge sans nombre de verrous ni cible d’alias ; les alias pointant vers lui restent toutefois inchangés.
_Replaces_lock_(target, source) Décrit la sémantique move assignment operator, où le verrou cible est débloqué avant de transférer l’état de la source. Vous pouvez le considérer comme une combinaison de _Moves_lock_(target, source) précédé d’un _Releases_lock_(target).
_Swaps_locks_(left, right) Décrit le comportement standard swap , qui suppose que les objets left et right échangent leur état. L’état échangé inclut le nombre de verrous et la cible d’alias, le cas échéant. Les alias qui pointent vers les objets left et right restent inchangés.
_Detaches_lock_(detached, lock) Décrit un scénario dans lequel un type de wrapper de verrou autorise la dissociation avec sa ressource contenue. Il est similaire à la manière dont std::unique_ptr fonctionne avec son pointeur interne : il permet aux programmeurs d’extraire le pointeur et de laisser son conteneur de pointeur intelligent dans un état propre. Une logique similaire est prise en charge par std::unique_lock et peut être implémentée dans des wrappers de verrous personnalisés. Le verrou détaché conserve son état (nombre de verrous et cible d’alias, le cas échéant), tandis que le wrapper est réinitialisé pour contenir un nombre de verrous de zéro et aucune cible d’alias, tout en conservant ses propres alias. Il n’existe aucune opération sur le nombre de verrous (déblocage et acquisition). Cette annotation se comporte exactement comme _Moves_lock_ sauf que l’argument détaché doit être return plutôt que this.

Voir aussi