Search code examples
windowsvisual-c++new-operatorcdecl

Warning (C28251) when replacing "operator new": Inconsistent annotation for 'new', this instance has no annotations


I'm trying to replace the global new operator in Visual Studio and C++. This is my code (only one new operator shown for simplicity):

void* operator new(size_t _Size)
{
    // Do something
}

It works fine, however Visual Studio is giving me a warning when running Code-Analysis for my project:

warning C28251: Inconsistent annotation for 'new': this instance has no annotations. The first user-provided annotation on this built-in function is at line vcruntime_new.h(48).

Using the annotations from vcruntime_new.h for my operator new replacement, as suggested in the warning from IntelliSense, resolves the warning:

_NODISCARD _Ret_notnull_ _Post_writable_byte_size_(_Size) _VCRT_ALLOCATOR
void* __cdecl operator new(size_t _Size)
{
    // Do something
}
  1. Is it safe to use the annotations inside vcruntime_new.h for my own replacement code like shown above?
  2. What are the consequences of this change?
  3. Are there special use cases were the "new operator" cannot be used anymore like before because of the annotations?
  4. And why is that change necessary?

EDIT:

  1. Am I correct, that the annotations won't change anything in the resulting binary, and are simply for static code analysis? (Except __cdecl, which changes the assembler, but it should be standard anyway I guess?)

Solution

  • This applies to _Ret_notnull_ and _Post_writable_byte_size_(_Size):

    Is it safe to use the annotations inside vcruntime_new.h for my own replacement code like shown above?

    Yes, as long as your operator new actually follow these rules in annotation. It might not follow _Ret_notnull_ (for example non-throwing new returns nullptr), tough normally will. But it should follow _Post_writable_byte_size_(_Size), since operator new is required so.

    What are the consequences of this change?

    Annotations will probably help Visual Studio Code Analysis pinpoint errors, but will make you using non-standard stuff, making your program less portable and less clear.

    Are there special use cases were the "new operator" cannot be used anymore like before because of the annotations?

    No.

    And why is that change necessary?

    Because you want to avoid that warning instead of silencing it.

    Am I correct, that the annotations won't change anything in the resulting binary, and are simply for static code analysis? (Except __cdecl, which changes the assembler, but it should be standard anyway I guess?)

    Correct. Not even optimizer would use that (optimizer would use __assume, __restrict, and other annotations separate from static analysis).