https://gcc.gnu.org/bugzilla/show_bug.cgi?id=82021
Bug ID: 82021 Summary: Unnecessary null pointer check in global placement new (and also in any class-specific placement new operator declared as noexcept) Product: gcc Version: 7.2.1 Status: UNCONFIRMED Severity: normal Priority: P3 Component: c++ Assignee: unassigned at gcc dot gnu.org Reporter: carl.cook at gmail dot com Target Milestone: --- Hi, This is just a minor nitpick, but after the clarification made in DR 1748 (http://open-std.org/jtc1/sc22/wg21/docs/cwg_defects.html#1748), there is a redundant null pointer check being performed within placement new. The null pointer check is done (in general) to make sure that any memory passed in is valid, and if not, no object initialization or deinitialization should take place. However, after a clarification to 5.3.4.15, it now states for placement new, passing null is undefined behavior (i.e. the caller must do any pointer checks). When using any version of gcc (from 7.2.1 and below), for any optimization level, and for the following two forms of placement new: * Global placement new (i.e. no user-defined operator placement new) * User-defined placement new (with noexcept specified) Then not only does the null pointer check still exist, but this seems to cause the optimizer to really miss a few things. Below is an obviously silly example, but you can see that the opportunitity to inline gets discarded, as does the opportunity to optimize-out the call to malloc. -- #include <memory> int GetVal (int x) { if (x == 0) return 1; else return 2; } struct Object { Object(int x) : i(GetVal(x)) {} int i; }; void* GetBuffer() { return malloc(sizeof(Object)); } int MyFn(int i) { void* mem = GetBuffer(); Object* object = new(mem)Object(i); return object->i; } int main(int argc, char** argv) { int rv = MyFn(*argv[0]); return rv; } -- If the following is added, at level O2, then this entire program gets optimized away (into returning the address of the first character of argv[0] as an integer) void* Object::operator new(size_t, void* where) { return where; } e.g. mov rax, QWORD PTR [rsi] cmp BYTE PTR [rax], 0 setne al movzx eax, al add eax, 1 ret I'm assuming that gcc now relies on the object-specific placement new to signal a null pointer by way of throwing an exception, hence the optimizer can be more aggressive. But I think the optimizer can now be just as agressive with all forms of placement new.