On Monday, 24 March 2025 at 17:33:20 UTC, Richard (Rikki) Andrew Cattermole wrote:
> >Since the code path is only triggered in release build, contracts are no-op and the code is executed at runtime.
It is fine to break it, see my prior comment:
Ok, let's put the -release thing aside for the moment.
>Unless it looks like this:
void caller() {
if (__ctfe) {
callee;
} else {
// whatever
}
}
void callee() @__ctfe {
}
That caller
function should not compile and link.
It doesn't have to be called like that.
int caller(int x) {
if (x < 2) {
callee();
} else {
// whatever
}
return 0;
}
void main() {
enum x = caller(1); // OK, we are in CTFE
writeln(caller(2)); // OK, callee is not called
}
The condition could be as complicated as we want. It can even depend on something that is only known at run time.
You probably made a mistake unintentionally but as a result I couldn't agree more with your statement :) Sure, with callee
having a @__ctfe
attribute, it should be an error to call it from caller
like that. And that's exactly what my implementation does. But I thought you were advocating an approach with callee
having an assert(__ctfe)
statement instead of @__ctfe
attribute, like
void calleeWithAssert() {
assert(__ctfe);
}
We can also make it an error to call calleeWithAssert
from caller
in the same way it calls callee
in my example. But it is a perfectly valid and working code currently, so that will be a breaking change. That's what I'm trying to say here.
If it does look like that, then callee
will never be referenced during codegen and all is well.
My current implementation will report an error here as well, but that we can likely fix.