rspec/rules/S6621/cfamily/rule.adoc
github-actions[bot] 3a17c6f769
Create rule S6621: Constexpr literal operators should be consteval.
Co-authored-by: Loghin Vlad-Andrei <vlad.loghin@sonarsource.com>
Co-authored-by: Loïc Joly <loic.joly@sonarsource.com>
Co-authored-by: Arseniy Zaostrovnykh <arseniy.zaostrovnykh@sonarsource.com>
2023-05-23 10:31:20 +02:00

45 lines
1.6 KiB
Plaintext

== Why is this an issue?
{cpp}20 introduces the `consteval` keyword, which enforces that a function is always evaluated at compile time.
`constexpr` functions can produce a compile-time constant in some contexts and when called with appropriate arguments, but they can also be invoked at run-time.
`consteval` functions must always be evaluated at compile-time. If that cannot happen, a compilation error will occur.
User-defined literal operators are designed to be called with constant operands known at compile time.
Therefore, if these are intended to be evaluated at compile time with `constexpr`, `consteval` should be used instead to enforce compile time evaluation.
This guarantees that no code is evaluated at run-time, and also enables error detection at compile-time.
=== Noncompliant code example
[source,cpp,diff-id=1,diff-type=noncompliant]
----
constexpr unsigned char operator ""_u8(unsigned long long value) {
if (value >= 256u) { throw std::overflow_error("Overflow on literal"); }
return static_cast<unsigned char>(value);
}
void f() {
auto const a = 128_u8; // evaluated at compile time
auto const b = 512_u8; // evaluated at runtime, throws when evaluated
}
----
=== Compliant solution
[source,cpp,diff-id=1,diff-type=compliant]
----
consteval unsigned char operator ""_u8(unsigned long long value) {
if (value >= 256u) { throw std::overflow_error("Overflow on literal"); }
return static_cast<unsigned char>(value);
}
void f() {
auto const a = 128_u8; // evaluated at compile time
auto const b = 512_u8; // compilation error, the bug is detected
}
----