March 12

On Tuesday, 11 March 2025 at 15:59:08 UTC, Ilya wrote:

>

So, what are my next steps? Should I write and submit a DIP? But https://github.com/dlang/DIPs says

>

The DIP queue is currently closed. Please do not submit pull requests for new DIPs.

Sorry about that. That's an oversight. It's been open for a while. But if you choose to write a DIP, the new process is to post the first draft to the DIP Development forum for community feedback, followed by as many draft reviews as necessary, before submitting a PR.

I ask people to notify me before they initiating the first draft. Most people so far have ended up skipping that step. The reason I ask is so that I can ensure that Walter and/or Atila have provided initial feedback in the idea forum before you invest the time on the draft.

I see in this thread that Walter replied to a comment, but neither he nor Atila has provided any feedback on the proposal. I'll ping them and ask them to chime in. That may influence your decision to proceed with a draft.

March 16
It's an interesting idea.

Note that linkers will remove unreferenced functions, so this would only impact compile time.

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.

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.

I.e. instead of:
```
@ctonly int func() { ... }
```
write:
```
int func() { assert(__ctfe); ... }
```
and so we're not introducing new syntax or semantics, just adding the ability to diagnose an error at compile time rather than runtime.
March 16
On Sunday, 16 March 2025 at 07:05:50 UTC, Walter Bright wrote:
> It's an interesting idea.
>
>
> 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.
>
> I.e. instead of:
> ```
> @ctonly int func() { ... }
> ```
> write:
> ```
> int func() { assert(__ctfe); ... }
> ```
> and so we're not introducing new syntax or semantics, just adding the ability to diagnose an error at compile time rather than runtime.

I would greatly prefer, controlling this outside of the function if possible.

@__ctfe int func() { ... }
pragma(__ctfe) int func() { ... }
@assert(__ctfe) int func() { ... }

March 16

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).

  1. 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.
  2. 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?

March 16
I think there is a slight misunderstanding.

I meant that the compiler, when compiling the declarations and not the definitions (i.e. the function body) can still check if the first statement is `assert(__ctfe)`, and then mark the function as CTFE-only. It will not need to run semantic on the definition. It will be semantically equivalent to adding a @__ctfe attribute on the declaration.

An `assert(__ctfe)` statement that is not the very first statement, will not be so recognized.

A declaration-only function can never be CTFE anyway.
March 16

On Sunday, 16 March 2025 at 16:57:44 UTC, Walter Bright wrote:

>

I meant that the compiler, when compiling the declarations and not the definitions (i.e. the function body) can still check if the first statement is assert(__ctfe), and then mark the function as CTFE-only. It will not need to run semantic on the definition. It will be semantically equivalent to adding a @__ctfe attribute on the declaration.

Yes, and that's exactly the problem. With the existing assert semantics, assert(__ctfe) as the first statement of a function does not provide a guarantee that this function could be omitted from codegen. So we can't use assert(__ctfe) as a CTFE-only marker.

March 16
On 3/16/2025 10:15 AM, Ilya wrote:
> On Sunday, 16 March 2025 at 16:57:44 UTC, Walter Bright wrote:
>> I meant that the compiler, when compiling the declarations and not the definitions (i.e. the function body) can still check if the first statement is `assert(__ctfe)`, and then mark the function as CTFE-only. It will not need to run semantic on the definition. It will be semantically equivalent to adding a @__ctfe attribute on the declaration.
> 
> Yes, and that's exactly the problem. With the existing assert semantics, `assert(__ctfe)` as the first statement of a function does **not** provide a guarantee that this function could be omitted from codegen.

I don't understand why it doesn't.

March 16

On Sunday, 16 March 2025 at 19:19:03 UTC, Walter Bright wrote:

> >

semantics, assert(__ctfe) as the first statement of a function does not provide a guarantee that this function could be omitted from codegen.

I don't understand why it doesn't.

Ok, let me repeat it once again. I can think of two scenarios:

  1. A function with assert(__ctfe) is called behind a guarding condition. For example:

    void f() {
      assert(__ctfe);
    }
    int g(int x) {
      if (x < 5) {
         f();
      }
      return x;
    }
    void main() {
      enum x = g(3); // OK, we are in CTFE
      writeln(g(6)); // OK, f is not called
    }
    

    Skipping codegen of f will result in a linker error, since g still references f, even though it's never called outside of CTFE. Steven suggested replacing calls to f with assert(__ctfe) (or a stricter version of it, that would also fire in release build), but it doesn't cover the second case...

  2. A function with assert(__ctfe) is occasionally called outside of CTFE, but it's a rare and untested execution path:

    int f(int x) {
      assert(__ctfe);
      return x + x;
    }
    void is_called_once_a_month_and_not_covered_by_tests(int x) {
      int x = f(x); // this happens at runtime, but rarely and is not covered by tests, so it only ever executed with release build and assert doesn't fire
    }
    void covered_by_test() {
      enum v = f(10); // other callers of `f` call it under CTFE.
    }
    

    In this case just skipping codegen of f is arguably better: it will break the build with a linker error, but at least people would look at it, find out the cause and fix it. Replacing calls to f with assert(__ctfe) will make the code break at run-time after the rare codepath is triggered, likely after running for a long time.

March 20
On Sunday, 16 March 2025 at 19:19:03 UTC, Walter Bright wrote:
> On 3/16/2025 10:15 AM, Ilya wrote:
>> On Sunday, 16 March 2025 at 16:57:44 UTC, Walter Bright wrote:
>>> I meant that the compiler, when compiling the declarations and not the definitions (i.e. the function body) can still check if the first statement is `assert(__ctfe)`, and then mark the function as CTFE-only. It will not need to run semantic on the definition. It will be semantically equivalent to adding a @__ctfe attribute on the declaration.
>> 
>> Yes, and that's exactly the problem. With the existing assert semantics, `assert(__ctfe)` as the first statement of a function does **not** provide a guarantee that this function could be omitted from codegen.
>
> I don't understand why it doesn't.

I think Ilya's mostly thinking about not breaking existing compilation. Either by linker errors or by compilation errors (which in the ideal world, would be the best).

This is why the `@__ctfe` is being recommended here. Mostly because it does not introduces new syntax to the language and does not break any build. That attribute would just be ignored on older compilers, which is a good strategy. Even though I don't think there would be TOO much breakage, the mainly affected projects would be the ones which does massive reflection work.
March 24
On Sunday, 16 March 2025 at 07:05:50 UTC, Walter Bright wrote:
> It's an interesting idea.
>
> Note that linkers will remove unreferenced functions, so this would only impact compile time.
>
> 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.
>
> 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.

Even better, the compiler can also look at contracts, and I for one would rather use `in(__ctfe)` instead of `assert(__ctfe);` as the first statement. It feels less hacky and uses a feature D already has.