On Sunday, 16 March 2025 at 07:05:50 UTC, Walter Bright wrote:
> Note that linkers will remove unreferenced functions, so this would only impact compile time.
Yes, this is exactly about improving compile time (allowing to skip codegen for a templated CT-only function with ~20K instantiations does make a difference).
> Steven brought up:
int func() { assert(__ctfe); ... }
as meaning code won't be generated for the function. Paul indicated this has problems because for non-auto external functions, semantic is not run on the function body.
Well, that's a technical issue that I'm sure could be solved one way or another. There is however a more important issue with the way asserts work.
> However, the compiler can just look at the first statement in a function and see if it is an assert(__ctfe);
and then make it an error to call it outside of CTFE.
We can, but it can break existing code. I've posted example in this thread already. Let's say we have a function starting with assert(__ctfe)
.
- It could be called from a branch that is only reached inside CTFE, compiler won't be able to figure this out, and that will be a compile-time error.
- It could be called at runtime, but very rarely, and there are no tests triggering that behavior. So it works perfectly, since
assert(__ctfe)
is only reached either inside CTFE or in release code, with the assert compiled out. So the code that previously worked is a compile-time error now. That's arguably a good one, since clearly it just slipped into release version due to runtime nature of asserts. But I still feel bad about breaking previously working code.
> I.e. instead of:
@ctonly int func() { ... }
write:
int func() { assert(__ctfe); ... }
and so we're not introducing new syntax or semantics, just
Many people suggested to use @__ctfe
instead of @ctonly
, that can be parsed with unmodified compiler, so no new syntax.
> adding the ability to diagnose an error at compile time rather than runtime.
But that part is not possible without breaking existing programs. Asserts happen at run-time, we simply can't precisely predict at compile time if they will trigger or not, so with a compile time check we have to approximate, meaning some existing code that works fine at run-time will result in compile time error.
That's why I'm proposing not to squash the two things together, keep the existing run time behavior of assert(__ctfe)
and add a brand new compile time behavior for @__ctfe
attribute.
And while I strongly believe we generally can't piggyback on assert(__ctfe)
as a "skip codegen" flag, it still provides a very strong signal that a function is likely CT-only. So once the new attribute thing works, we could provide users with a tool that tries adding @__ctfe
attribute to functions with assert(__ctfe)
and if it compiles, the change could be committed safely. That would simplify the migration of existing code heavily relying on assert(__ctfe)
.
What do you think?