bloomberg/bde

replace the `createNull` method with a public `static const s_null` data member

Closed this issue · 17 comments

This change has been discussed with the bde team, and I had intended to make it before handing Datum off to them, but never got around to it. It would simplify use of Datum, provide a potential performance benefit (as null objects are created frequently), and as a side-benefit, allow the logic in the body of createNull to move into bdld_datum.cpp.

Note that createNull could be left intact (perhaps returning a const Datum&) for compatibility.

I'll be happy to write the patch myself if this change is approved.

Hi Brock!

Would this open up any order-of-initialization issues for statics?

We might want the s_null field to be a static local in a static function, initialized in a BCEMT_ONCE_DO just to avoid that possibility.

This sounds like a good idea, but I think we should wait for @mversche or Atilla to comment since they know more about Datum than I do.

Thanks,

Mike

Hi Mike!

That's a quick response! I don't think it would cause any ordering problems -- createNull doesn't call any methods:

    Datum result;
#ifdef BSLS_PLATFORM_CPU_32_BIT
    // Setting exponent using half-word is faster, maybe the compiler folds the
    // two statements into one?

    result.d_as.d_exponent = k_DOUBLE_MASK | Datum::e_INTERNAL_EXTENDED;
    result.d_as.d_ushort   = e_EXTENDED_INTERNAL_NIL;
#else   // BSLS_PLATFORM_CPU_32_BIT
    result.d_as.d_type     = e_INTERNAL_NIL;
#endif  // BSLS_PLATFORM_CPU_32_BIT

    return result;

But yes, definitely happy to hear the opinions of @mversche and Attila.

Why would a static data member be more efficient? I would imagine you should just make createNull constexpr.

@dharesign, not all of our compilers support constexpr yet.

If they did, things would be much simpler, I agree.

(also, nice to see another member of the large godbolt.org fan club :) )

Yeah, godbolt.org is awesome. The constexpr solution works for me, though the static seems clear and does work for your older compilers.

Re. constexpr: what other methods could it be applied to? Might it be a reasonable optimization for the other create methods? I think it's pretty common to call them with, e.g., numeric constants.

I'm not sure just making createNull constexpr is valid. Some compilers were complaining about a variety of things. It probably makes more sense to provide a constexpr constructor which initializes Datum to the null state?

See below for the various options. In each case, compiling with --std=c++14 -O3:

Based on this, I'm not sure creating a static s_null is really beneficial.

A liberal sprinkling of constexpr where it makes sense looks like it would provide real benefits though.

"A liberal sprinkling of constexpr where it makes sense looks like it would provide real benefits though."

Yeah, I think that is more generally useful than making a static s_null.

Also note that I forgot to define createNull inline. If you do that, it's actually better than my constexpr constructor as the current implementation doesn't initialize the memory to 0.

constexpr is great when the compiler supports it, but as I said, we have to support compilers that don't... If you do want to use it, use BSLS_CPP11_CONSTEXPR from the bsls_cpp11 component.

I still like the s_null idea for the non-cpp11 case.

Let me summarize my ramblings: there is no benefit from s_null. As createNull is inline, you're not going to get much better than not initializing a block of memory and then copying a 2 into a short. Having a static s_null actually means you have to copy 16 bytes, rather than just 2 bytes.

Adding constexpr doesn't really help in the general case. If you wanted to be able to create Datum objects in constexpr contexts, then createNull etc. could be made constexpr, but to be able to do that, Datum's constructor would need to be explicitly defined and constexpr, which would lead to Datum no longer being a PODType. It would still be TriviallyCopyable though, and arguably the default constructor giving you a Datum in the null state is better than what you currently get.

There are plenty of situations where a const Datum& is expected and no copy would happen at all. I can't see how it would be more efficient to create and return a value than to use a value that is already initialized.

I guess I misunderstood your use case, given:

as null objects are created frequently

If you have a need for lots of const Datum & to null Datums, then perhaps you should have your own? Sure, BDE could provide it. But maybe someone wants an s_zero, or s_emptyString. Where do you draw the line, and is this something that should be provided by Datum itself?

That's a good point, and part of the reason I was interested in constexpr in general. I think the main compelling reason I could have for s_null in particular is that it would remove the need for createNull and is unambiguously simpler and more efficient. Whereas, s_zero and s_emptyString would be special cases, extending the interface of Datum.

If you actually want to create a Datum having the null state, then as I showed above using the current form of createNull is better than copying an s_null. So I don't think it could replace createNull.

And given that there would be no difference, from a performance standpoint, between an s_null Datum provided by BDE, and one provided by yourself in your application, I'm not so convinced.

Frequently, you don't need to create a new object at all. In that case, the static version is far more efficient (this is true if you use constexpr too): https://godbolt.org/g/il7qmQ

Closing stale issues