February 23

On Sunday, 23 February 2025 at 17:07:30 UTC, Ilya wrote:

>

On Friday, 21 February 2025 at 14:04:41 UTC, Steven Schveighoffer wrote:

>

Paul Backus pointed out on discord, there is a very large problem with this mechanism:

importing a file with a function that is not a template nor an auto function means the semantic is skipped for that function. This means it would be impossible to know that this function is ctfe-only. The only other solution there is to semantic every function, which basically negates any wins we get from not optimizing these functions.

Yes, that's a good point, thanks.

>

So reading all the responses here, my impression is:

  1. I think it's a necessary idea to have an attribute if we want this feature. I like using @__ctfe as this requires no new syntax, and actually already compiles (it just doesn't do the thing expected). What remains to be seen is if we can hook this properly. @__ctfe becomes @(true) currently. If not, then a new attribute is trivial to add to core.attributes.

I've already updated our internal version to use @__ctfe, it works just fine.

>
  1. If we are going to have a mechanism to flag that code generation should be avoided, it would be good to also do this for functions that begin with assert(__ctfe). This at least helps with existing uses of that mechanism.

As I said in another reply, I think it can't (and shouldn't) be a plugin replacement for assert(__ctfe). But we can do a small migration tool.

>
  1. I don't agree with the compiler errors for all uses of ctfe-only functions at runtime. As you pointed out, this can be valid for cases. I don't think it's worth the headache, and I have learned in the past not to pre-optimize for rule following especially in generic code. Just let the thing happen as it would.

Oh, so you are saying we will have to add an attribute (because of compiler implementation reasons), but we'll make it work as if it's an assert(__ctfe) in the first line anyway?

I don't agree here. Static check allows us to simply drop these functions, without having to rewrite call sites, it plugs naturally into existing skipCodegen implementation and it provides faster feedback, helping to set the attributes correctly. It also lets dropping more functions, think templates. If you use map!ctonly(xs), there is no point to codegen MapResult!(ctonly, xs).front for example.

The check will reject some code that could be written with asserts, that's correct. But I don't see it as a problem. It's like static vs dynamic types. You can't put a string into an int in a typed language, but in Python you can. Does it make typed languages worse? I'm not sure.

Since we are adding a new annotation, let's make it right. Such that we are absolutely sure we can omit the codegen and no surprises will happen in run time. And if someone wants to have a contrived behavior, like the one in my example, they are welcome to continue using asserts. We are not going to deprecate assert(__ctfe), right?

>
  1. To that end, calling a known ctfe-only function should result in an appropriate assert(false, "cannot call ctfe-only function blahblah at runtime") when doing code generation instead of the call to a non-existent function that has a linker error.

I honestly prefer a linker error over a run-time assert :) Sounds like a time bomb.

There are people out there which can't read linker errors, contrary to that, I believe that an advanced feature anyway. To be fair, nowadays, I understand them enough for that. Maybe we could implement that behavior of no-codegen vs a stub with assert(false) with the same strategy as -check=.

Since assert(__ctfe) is existing code, I would say that, implementations that skip codegen because of assert(__ctfe), we could instead, we could do like -ctfeskip=stub and -ctfeskip=yes or something like that.

While @__ctfe would altogether trigger the same behavior as no codegen at all, like -ctfeskip=yes.

That would bring:

  1. More control to one which understand what they're doing by using either @__ctfe and -ctfeskip=yes.
  2. Benefits existing code
  3. Avoid breaking existing code since the default could be using the stub.
February 23

On Sunday, 23 February 2025 at 17:57:58 UTC, Hipreme wrote:

> >

I honestly prefer a linker error over a run-time assert :) Sounds like a time bomb.

There are people out there which can't read linker errors, contrary to that, I believe that an advanced feature anyway. To

Yes! Absolutely! That's exactly my point. I even believe that if one is using a modern programming language and is not doing something very hacky or low-level, there should not be any linker errors at all. That's why I'm pushing for a compile-time check here. Clear compilation error is much better than obscure link error. I would never argue that.

But in the quoted sentence I'm talking about linker error vs run-time assert in release build. And I would totally pick a linker error over that even if I couldn't read link errors. Even if cryptic, that's still a build time error, one can bisect the changes even without understanding it, find the culprit and either fix or revert. Compare that to hitting an assert in release mode in production three days after deployment.

>

be fair, nowadays, I understand them enough for that. Maybe we could implement that behavior of no-codegen vs a stub with assert(false) with the same strategy as -check=.

We can certainly gate the new functionality behind a flag... But TBH I don't see much value in it. There is literally no existing code with @__ctfe, and it can't really work with assert(__ctfe) anyway.

>

Since assert(__ctfe) is existing code, I would say that, implementations that skip codegen because of assert(__ctfe), we could instead, we could do like -ctfeskip=stub and -ctfeskip=yes or something like that.

Well, with assert(__ctfe) -ctfeskip=yes could lead to link errors. And -ctfeskip=stub is changing existing behavior.

>

While @__ctfe would altogether trigger the same behavior as no codegen at all, like -ctfeskip=yes.

Ok, that's fine. But we either need a compile time check, or there will be link errors.

>

That would bring:

  1. More control to one which understand what they're doing by using either @__ctfe and -ctfeskip=yes.

I suggest we just pick that. More control for those who opt in.

>
  1. Benefits existing code
  2. Avoid breaking existing code since the default could be using the stub.

But stubs do break existing code. I've already posted a scenario. Now if you don't hit assert(__ctfe) with debug executable, in release it will happily execute the function in run time. With stubs it will break.

February 24
On 2/20/2025 9:46 AM, Ilya wrote:
> On Thursday, 20 February 2025 at 15:51:51 UTC, Paul Backus wrote:
>> In case you weren't aware, this exists in C++, and is called [consteval][1].
> 
> Thanks for the pointer, I haven't used consteval in C++, that seems relevant indeed.
> 

In C++, a function not annotated with consteval cannot be run at compile time.
March 01
On Tuesday, 25 February 2025 at 03:34:37 UTC, Walter Bright wrote:
> On 2/20/2025 9:46 AM, Ilya wrote:
>> On Thursday, 20 February 2025 at 15:51:51 UTC, Paul Backus wrote:
>>> In case you weren't aware, this exists in C++, and is called [consteval][1].
>> 
>> Thanks for the pointer, I haven't used consteval in C++, that seems relevant indeed.
>> 
>
> In C++, a function not annotated with consteval cannot be run at compile time.

Is that still being considered? That optimization was supposed to happens so many years ago, would be a big ball drop not carrying on with that right now.
March 01
On Tuesday, 25 February 2025 at 03:34:37 UTC, Walter Bright wrote:
> On 2/20/2025 9:46 AM, Ilya wrote:
>> On Thursday, 20 February 2025 at 15:51:51 UTC, Paul Backus wrote:
>>> In case you weren't aware, this exists in C++, and is called [consteval][1].
>> 
>> Thanks for the pointer, I haven't used consteval in C++, that seems relevant indeed.
>> 
>
> In C++, a function not annotated with consteval cannot be run at compile time.

As I understand it, functions in C++ fall into one of three categories:

1. consteval - cannot be run at runtime, only at compile time.
2. constexpr - may be run at compile time or at runtime.
3. No attribute - cannot be run at compile time, only at runtime.

In D, we do not draw a hard line between categories (2) and (3), and we have no equivalent of category (1).

The proposal in this thread is about bringing category (1) to D.
March 03

On Saturday, 1 March 2025 at 21:02:25 UTC, Paul Backus wrote:

>

As I understand it, functions in C++ fall into one of three categories:

  1. consteval - cannot be run at runtime, only at compile time.
  2. constexpr - may be run at compile time or at runtime.
  3. No attribute - cannot be run at compile time, only at runtime.

In D, we do not draw a hard line between categories (2) and (3), and we have no equivalent of category (1).

The proposal in this thread is about bringing category (1) to D.

Yes, exactly! Thanks for describing it so clearly! BTW, I can see a value in declaring that some functions should never be executed at compile-time, but this proposal focuses specifically on (1).

So, what's the conclusion? I think everyone agreed that it has to be an attribute.

Can we also agree that stubs break existing behavior? And that without stubs it must be a compile-time check, so it can't possibly work exactly like assert(__ctfe) does?

March 03

On Monday, 3 March 2025 at 11:33:16 UTC, Ilya wrote:

>

On Saturday, 1 March 2025 at 21:02:25 UTC, Paul Backus wrote:

>

As I understand it, functions in C++ fall into one of three categories:

  1. consteval - cannot be run at runtime, only at compile time.
  2. constexpr - may be run at compile time or at runtime.
  3. No attribute - cannot be run at compile time, only at runtime.

In D, we do not draw a hard line between categories (2) and (3), and we have no equivalent of category (1).

The proposal in this thread is about bringing category (1) to D.

Yes, exactly! Thanks for describing it so clearly! BTW, I can see a value in declaring that some functions should never be executed at compile-time, but this proposal focuses specifically on (1).

So, what's the conclusion? I think everyone agreed that it has to be an attribute.

Can we also agree that stubs break existing behavior? And that without stubs it must be a compile-time check, so it can't possibly work exactly like assert(__ctfe) does?

Ok, so, IMO, we could do the following:

  1. Add the attribute @__ctfe, which would completely ignore the assert(__ctfe) situation. That won't break any existing code.
  2. Optimization on the existing assert(__ctfe) code must happen under a flag.
    2.1. Under Optimize assert : If we DO implement the skip code-gen for those, we would need to get that error at compilation time. That would mean there's a high risk of having multiple libraries inside dub breaking at compilation time.
    2.2. Keep the existing behavior: If that flag is not specified, assert(__ctfe) will simply keep the existing behavior.
March 04

On Monday, 3 March 2025 at 16:55:30 UTC, Hipreme wrote:

>

Ok, so, IMO, we could do the following:

  1. Add the attribute @__ctfe, which would completely ignore the assert(__ctfe) situation. That won't break any existing code.

How about we focus on this first? And once that works, we can do a lint/D-scanner check/whatever, that would suggest replacing assert(__ctfe) with @__ctfe attributes? Possibly even generating the diff. But people would still apply the suggestions explicitly.

>
  1. Optimization on the existing assert(__ctfe) code must happen under a flag.

But now (1) and (2) are completely independent, can't we do them separately? Also, I think there was some technical problem: AFAIU, looking for assert(__ctfe) requires analyzing the function body, which would most likely out-weight all the benefits of skipped codegen...

So, again, could we just focus on (1) for a moment? Do everybody agree?

March 04

On Tuesday, 4 March 2025 at 16:58:05 UTC, Ilya wrote:

>

On Monday, 3 March 2025 at 16:55:30 UTC, Hipreme wrote:

>

Ok, so, IMO, we could do the following:

  1. Add the attribute @__ctfe, which would completely ignore the assert(__ctfe) situation. That won't break any existing code.

How about we focus on this first? And once that works, we can do a lint/D-scanner check/whatever, that would suggest replacing assert(__ctfe) with @__ctfe attributes? Possibly even generating the diff. But people would still apply the suggestions explicitly.

>
  1. Optimization on the existing assert(__ctfe) code must happen under a flag.

But now (1) and (2) are completely independent, can't we do them separately? Also, I think there was some technical problem: AFAIU, looking for assert(__ctfe) requires analyzing the function body, which would most likely out-weight all the benefits of skipped codegen...

So, again, could we just focus on (1) for a moment? Do everybody agree?

Yup, I don't really care about getting the existing code to improve, specially since it may break things if we do that compilation error. Making them separate is a big win IMO since we could first test and check if it is worth even doing it. Also we could get that feature on upstream faster.

March 11

On Tuesday, 4 March 2025 at 17:18:36 UTC, Hipreme wrote:

> >

So, again, could we just focus on (1) for a moment? Do everybody agree?

Yup, I don't really care about getting the existing code to improve, specially since it may break things if we do that compilation error. Making them separate is a big win IMO since we could first test and check if it is worth even doing it. Also we could get that feature on upstream faster.

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.