Partager via


Meilleures pratiques et exemples (SAL)

Voici quelques façons d'obtenir le meilleur parti du langage d'annotation du code source (SAL) et d'éviter des problèmes courants.

_In_

Si la fonction est censée écrire dans l'élément, utilisez _Inout_ au lieu de _In_.Ceci est particulièrement adapté en cas de conversion automatisée des anciennes macros vers des SAL.Avant le SAL, de nombreux programmeurs utilisaient des macros comme macros-commentaires qui ont été appelées IN, OUT, IN_OUT, ou des variantes de ces noms.Bien que nous recommandions que vous convertissiez ces macros vers SAL, nous vous invitons également à être prudent lorsque vous les convertissez parce que le code peut avoir changé depuis que le prototype d'origine a été écrit et l'ancienne macro ne peut plus refléter ce que le code fait.Soyez particulièrement prudent sur la macro de commentaire OPTIONAL car elle est souvent définie incorrect - pour l'exemple, du mauvais côté d'une virgule.

// Incorrect
void Func1(_In_ int *p1)
{
    if (p1 == NULL) 
        return;

    *p1 = 1;
}

// Correct
void Func2(_Inout_ PCHAR p1)
{
    if (p1 == NULL) 
        return;

    *p1 = 1;
}

_opt_

Si l'appelant n'est pas autorisée à s'exécuter dans un pointeur null, utilisez _In_ ou _Out_ au lieu de _In_opt_ ou de _Out_opt_.Cela s'applique même à une fonction qui vérifie ses paramètres et retourne une erreur si elle a la valeur NULL lorsqu'elle ne doit pas l'être.Bien qu'ayant une fonction de vérification de son paramètre pour les cas inattendus NULL et qu'un retour correct soit une bonne pratique défensive en matière de codage, il ne signifie pas que l'annotation de paramètre peut être d'un type facultatif (_Xxx_opt_).

// Incorrect
void Func1(_Out_opt_ int *p1)
{
    *p = 1;
}

// Correct
void Func2(_Out_ int *p1)
{
    *p = 1;
}

_Pre_defensive_ et _Post_defensive_

Si une fonction apparaît à la limite d'approbation, nous vous recommandons d'utiliser l'annotation d' _Pre_defensive_ .Le modificateur « défensif » modifie certaines annotations pour indiquer que, au point de l'appel, l'interface doit être vérifiée strictement, mais au corps d'implémentation elle doit supposer que des paramètres incorrects puissent être passés.Dans ce cas, _In_ _Pre_defensive_ est préférable à une limite d'approbation pour indiquer que même si un appelant obtenir une erreur si elle tente de passer NULL, le corps de la fonction est analysé comme si le paramètre peut être NULL, et toute tentative de déréférence le pointeur sans l'examen d'abord pour rechercher la valeur NULL sont marqués d'un indicateur.Une annotation de _Post_defensive_ est également disponible, à utiliser dans les rappels où il est supposé que la partie approuvée est l'appelant et code non fiable correspond au code appelé.

_Out_writes_

L'exemple suivant illustre une utilisation courante _Out_writes_.

// Incorrect
void Func1(_Out_writes_(size) CHAR *pb, 
    DWORD size
);

L'annotation _Out_writes_ signifie que vous avez une mémoire tampon.Elle fait allouer des octets cb , avec le premier octet initialisé sur la sortie.Cette annotation n'est pas strictement erronées et il est utile d'exprimer la taille allouée.Toutefois, il n'indique pas le nombre d'éléments qui sont initialisés par la fonction.

Le code de l'exemple montre trois méthodes nécessaires pour entièrement spécifier la taille exacte de la partie initialisée de la mémoire tampon.

// Correct
void Func1(_Out_writes_to_(size, *pCount) CHAR *pb, 
    DWORD size,
    PDWORD pCount
);

void Func2(_Out_writes_all_(size) CHAR *pb, 
    DWORD size
);

void Func3(_Out_writes_(size) PSTR pb, 
    DWORD size
);

_Out_ PSTR

L'utilisation de _Out_ PSTR est presque toujours erronées.Cela est interprété comme ayant un paramètre de sortie qui pointe vers une mémoire tampon de caractères et c'est NULL-terminé.

// Incorrect
void Func1(_Out_ PSTR pFileName, size_t n);

// Correct
void Func2(_Out_writes_(n) PSTR wszFileName, size_t n);

Une annotation comme _In_ PCSTR est courante et utile.Elle indique une chaîne d'entrée qui a une terminaison NULL car le précondition de _In_ permet la reconnaissance d'une chaîne terminée par le caractère NULL.

_In_ WCHAR* p

_In_ WCHAR* p signifie qu'il y a un pointeur en entrée p qui pointe sur un caractère.Toutefois, dans la plupart des cas, ce n'est probablement pas la spécification qui est attendu.À la place, ce qui est probablement attendu est la spécification d'un tableau se terminant par NULL ; pour ce faire, utilisez _In_ PWSTR.

// Incorrect
void Func1(_In_ WCHAR* wszFileName);

// Correct
void Func2(_In_ PWSTR wszFileName);

L'oubli de la spécification approprié de la terminaison NULL est courant.Utilisez la version appropriée STR pour remplacer le type, comme le montre l'exemple suivant.

// Incorrect
BOOL StrEquals1(_In_ PCHAR p1, _In_ PCHAR p2)
{
    return strcmp(p1, p2) == 0;
}

// Correct
BOOL StrEquals2(_In_ PSTR p1, _In_ PSTR p2)
{
    return strcmp(p1, p2) == 0;
}

_Out_range_

Si le paramètre est un pointeur et que vous souhaitez exprimer la plage de la valeur de l'élément qui est est pointé par le pointeur, utilisez _Deref_out_range_ au lieu de _Out_range_.Dans l'exemple suivant, la plage de *pcbFilled est exprimé, pas pcbFilled.

// Incorrect
void Func1(
    _Out_writes_bytes_to_(cbSize, *pcbFilled) BYTE *pb, 
    DWORD cbSize, 
    _Out_range_(0, cbSize) DWORD *pcbFilled
);

// Correct
void Func2(
    _Out_writes_bytes_to_(cbSize, *pcbFilled) BYTE *pb, 
    DWORD cbSize, 
    _Deref_out_range_(0, cbSize) _Out_ DWORD *pcbFilled 
);

_Deref_out_range_(0, cbSize) n'est pas strictement requis pour certains outils car il peut être déduit de _Out_writes_to_(cbSize,*pcbFilled), mais il est indiqué ici à des fins de précision.

Contexte incorrect dans _When_

Une autre erreur courante consiste à utiliser l'évaluation post-état des préconditions.Dans l'exemple suivant, _Requires_lock_held_ représente une précondition.

// Incorrect
_When_(return == 0, _Requires_lock_held_(p->cs))
int Func1(_In_ MyData *p, int flag);

// Correct
_When_(flag == 0, _Requires_lock_held_(p->cs))
int Func2(_In_ MyData *p, int flag);

L'expression result fait référence à une valeur post-état qui est pas disponible au stade pré-état.

TRUE dans _Success_

Si la fonction réussit lorsque la valeur de retour est différente de zéro, utilisez return != 0 comme condition de succès au lieu de return == TRUE.Une valeur différente de zéro ne signifie pas nécessairement l'équivalence à la valeur réelle que le compilateur fournit pour TRUE.Le paramètre à _Success_ est une expression, et les expressions suivantes sont évaluées comme équivalents : return != 0, return != false, return != FALSE, et return sans paramètres ou les comparaisons.

// Incorrect
_Success_(return == TRUE, _Acquires_lock_(*lpCriticalSection))
BOOL WINAPI TryEnterCriticalSection(
  _Inout_ LPCRITICAL_SECTION lpCriticalSection
);

// Correct
_Success_(return != 0, _Acquires_lock_(*lpCriticalSection))
BOOL WINAPI TryEnterCriticalSection(
  _Inout_ LPCRITICAL_SECTION lpCriticalSection
);

Variable de référence

Pour une variable de référence, la version antérieure du SAL a utilisé le pointeur implicite comme cible d'annotation et nécessitait l'ajout de __deref aux annotations qui se sont liées à une variable de référence.Cette version utilise l'objet lui-même et ne requiert pas de _Deref_supplémentaire.

// Incorrect
void Func1(
    _Out_writes_bytes_all_(cbSize) BYTE *pb, 
    _Deref_ _Out_range_(0, 2) _Out_ DWORD &cbSize
);

// Correct
void Func2(
    _Out_writes_bytes_all_(cbSize) BYTE *pb, 
    _Out_range_(0, 2) _Out_ DWORD &cbSize
);

Annotations sur les valeurs de retour

L'exemple suivant présente un problème fréquent dans les valeurs de retour des annotations.

// Incorrect
_Out_opt_ void *MightReturnNullPtr1();

// Correct
_Ret_maybenull_ void *MightReturnNullPtr2();

Dans cet exemple, _Out_opt_ indique que le pointeur peut être NULL dans le cadre d'une précondition.Toutefois, les préconditions ne peuvent pas être appliquées à la valeur de retour.Dans ce cas, l'annotation correcte est _Ret_maybenull_.

Voir aussi

Référence

Annotation de paramètres de fonction et valeurs de retour

Annotation du comportement d'une fonction

Structs et classes d'annotation

Annotation du comportement de verrouillage

Spécification du moment où une annotation est applicable et dans quel cas

Fonctions intrinsèques

Concepts

Présentation de SAL

Autres ressources

Utilisation d'annotations SAL pour réduire les défauts du code C/C++