Search code examples
c++c++20constexprfmtstring-view

How can I avoid using #define macros in C++ in a case where I have to concatenate two const char* variables?


I would like to remove the reliance of #define macros in my code, and I'm unable to achieve the same functionality with constexpr.

To be practical, consider the following example:

#define PRODUCT_NAME "CloysterHPC"
constexpr const char* productName = PRODUCT_NAME;

class Newt : public View {
private:
    struct TUIText {

#if __cpp_lib_constexpr_string >= 201907L
        static constexpr const char* title =
            fmt::format("{} Installer", productName).data();
#else
        static constexpr const char* title = PRODUCT_NAME " Installer";
#endif

    };
};

I've learned the hard way that fmt::format() function is not a constexpr function and it's only a runtime function. I was expecting that I could it in the code to be more expressive, but I can't. So I've tried using std::string, but again I got the same exact results after changing the code to something like:

#define PRODUCT_NAME "CloysterHPC"
constexpr const char* productName = PRODUCT_NAME;

class Newt : public View {
private:
    struct TUIText {

#if __cpp_lib_constexpr_string >= 201907L
        static constexpr const char* title = std::string{
            std::string{productName} + std::string{" Installer"}}.data();
#else
        static constexpr const char* title = PRODUCT_NAME " Installer";
#endif

    };
};

So what are my misunderstandings:

  1. That I could use fmt in a constexprcontext. Which is untrue.
  2. std::string with proper support from libstdc++ should be constexpr to evaluate string operations at compile time, but it does not seems to be the case.
  3. I misunderstood the utility of __cpp_lib_constexpr_string macro on the Standard Library.
  4. That C++20 would give more flexibility with text manipulation in a constexpr context.

I already done my homework and came across other questions of Stack Overflow_ about similar issues, or how to use std::string in a constexpr context:

But none of them answered my question with clarity: How can I concatenate two given strings at compile time to properly get rid of #define macros in the code?

This seems to be trivial, since both strings are known at compile time, and they are also constexpr. The final goal would be to have a third constexpr const char* with the content: CloysterHPC Installer without using any #define macros on the code.

How can I achieve this? Is that even possible in the current state of the language? Should I continue using the macro? My current setup is GCC 12.1.1 with default libstdc++ on a RHEL 8.7 system:

gcc-toolset-12-12.0-5.el8.x86_64
gcc-toolset-12-libstdc++-devel-12.1.1-3.4.el8_7.x86_64

PS: Please note that sometimes I mentioned strings in the question knowing that they aren't std::string, but actually const char*. It was just a language convention and not a type definition.


Solution

  • That I could use fmt in a constexprcontext. Which is untrue.

    Yes, I am not actually sure why that is the case. Maybe there are some issues that would make it too hard to implement at the moment (e.g. reliance on some non-constexpr function in the implementation).

    But even if it was constexpr, it wouldn't help here because it returns std::string and the following points:

    std::string with proper support from libstdc++ should be constexpr to evaluate string operations at compile time, but it does not seems to be the case.

    Libstdc++ does support C++20 constexpr-friendly std::string and can manipulate them at compile-time. There is no implementation issue here. However, there is currently no mechanism in the language to let a dynamic allocation at compile-time live until runtime. std::string does however require dynamic allocation, because it can store strings of any length. So it can't be possible to pass a std::string from a compile-time context to a runtime context (i.e. define a std::string with the constexpr keyword). Any std::string used at compile-time must be destroyed before the compile-time context ends.

    I misunderstood the utility of __cpp_lib_constexpr_string macro on the Standard Library.

    It only indicates that std::string can be used at compile-time as I described above.

    That C++20 would give more flexibility with text manipulation in a constexpr context.

    Inside a constexpr context you are free to manipulate a std::string in whatever way you want since C++20. But you are not only trying to manipulate a std::string in a compile-time context, you are also trying to pass a std::string across the compile-time/runtime boundary.


    But none of them answered my question with clarity: How to concatenate two given strings at compile time to properly get rid of #define macros in the code?

    As I described above, the fundamental issue is that std::string requires dynamic allocation to provide strings of arbitrary length. So it can't be used for this purpose. Just a const char* also can't be used, because even if you evaluate a constant at compile-time, that is still an object with a lifetime and a raw pointer can't manage the lifetime of an object. (String literals are an exception because the language specifically gives them a lifetime simply by being written in the code.)

    So to solve your problem, you need a type that can own and manage the lifetime of the string's contents and that doesn't need dynamic memory allocation, meaning that it must store fixed length strings, e.g.:

    template<std::size_t N>
    using fixed_string = std::array<char, N>;
    

    Here I would interpret N to contain space for a null-terminator.

    The string length here is part of the type, so we can't simply write a function taking inputs as function arguments that don't encode the length in their types. Or for more general string manipulation, the output string's length might depend on the contents of the string inputs.

    So, we need to make sure that the inputs are passed in a way that their types encode the string. One way of doing this is by passing as template arguments.

    Now you can write e.g. the following function:

    template<auto str1, auto str2>
    constexpr auto concat_constant_strings() {
        constexpr auto size = (std::ranges::size(str1)-1) + (std::ranges:::size(str2)-1) + 1;
        fixed_string<size> result;
        std::ranges::copy(str1, std::ranges::begin(result));
        std::ranges::copy(str2, std::ranges::begin(result)+std::ranges::size(str1)-1);
        return result;
    }
    

    which can be called with fixed_string's as template arguments.

    Now the remaining problem is passing string literals as template arguments, which isn't allowed directly, i.e. the template parameter may not be of type const char* referring to a string literal. We need a way to convert the string literal to a fixed_string, which however isn't too difficult, in fact that is what the library function std::to_array does (assuming our definition of fixed_string):

    static constexpr auto title = concat_constant_strings<std::to_array(PRODUCT_NAME), std::to_array(" Installer")>();
    

    Now title.data() can be used as you did your original title (and if you want you can store a constexpr const char* to that, but you need to store the fixed_string in an actual constexpr variable to manage the lifetime of the data).

    All of this can be improved by actually defining your own fixed_string class with string and range semantics instead of relying on std::array. For example, fixed_string could be given a constructor from string literals (as references to const char arrays). With a matching deduction guide, the auto template parameters could also be replaced with fixed_string, so that CTAD can allow accepting string literals directly as template arguments instead of passing through std::to_array, etc. The implementation above is a minimal one. A lot could be done cleaner.

    Also, one can use std::integral_constant or a similar template and string literal operators to move from template arguments to function arguments.

    As an additional aside, in the implementation of concat_constant_strings you could form std::strings from str1 and str2 first, then produce a new string and finally construct a fixed_string to return. This way you can lift any normal std::string operation:

    template<auto str1, auto str2>
    constexpr auto some_constant_string_operation() {
        constexpr auto lambda = []{
            std::string string1(std::ranges::begin(str1), std::prev(std::ranges::end(str1)));
            std::string string2(std::ranges::begin(str2), std::ranges::prev(std::ranges::end(str2)));
            std::string result;
            /* any std::string manipulation */;
            return result;
        };
        constexpr auto size = std::ranges::size(lambda())+1;
        fixed_string<size> result;
        std::ranges::copy(lambda(), std::ranges::begin(result));
        return result;
    }