Search code examples
c++macrosplacement-new

Are these placement new macros correct?


I made a couple macros to make using placement new a bit easier. I was just wondering if there were any obvious cases where these would not work. Thanks.

#define CONSTRUCT_INPLACE(TYPE,STORAGE,INIT)   ::new((TYPE*)STORAGE) TYPE INIT
#define DESTRUCT_INPLACE(TYPE,STORAGE)         ((TYPE*)STORAGE)->~TYPE()

Solution

  • I'm not an expert in placement new but there are a couple of issues with how you are defining the macro.

    Issue 1

    The most obvious problem is the use of the cast (TYPE*)STORAGE for the storage location. This is incorrect. Placement new is just another C++ function and it participates in operations like overload resolution. Arbitrarily casting the memory to a specific type could cause the placement new to bind to a different operator new that the user expected.

    For example, it's valid to have the following two definitions of placement new. Your macro would potentially cause the wrong one to be called.

    void * _cdecl operator new(size_t cbSize, void* pv);
    void * _cdecl operator new(size_t cbSize, SomeType* pv)-
    

    ...

    // These two call different overloads
    void* p = malloc(sizeof(SomeType));
    SomeType* f1 = CONSTRUCT_INPLACE(SomeType, p,()) 
    SomeType* f2 = new (p) SomeType();
    

    I wrote a blog post awhile back on how you can use this type of overload resolution to implement custom allocators.

    Issue 2

    The expression STORAGE in the macro should be wrapped in parens to prevent evil macro expansion bugs.

    ::new((TYPE*)(STORAGE)) TYPE INIT