March 24

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:

https://forum.dlang.org/post/vrs4i3$du0$1@digitalmars.com

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.

March 24

On Monday, 24 March 2025 at 16:56:57 UTC, Richard (Rikki) Andrew Cattermole wrote:

>

On 25/03/2025 5:54 AM, Ilya wrote:

>

I'm positively sure we can't skip codegen for functions with | in(__ctfe)| contracts without breaking existing code potentially.

There is no existing code that has this and runs at runtime.

import std.stdio;

string hello() in(__ctfe) => "hello world";
immutable funcTable = [&hello];
enum message = funcTable[0]();

void main()
{
    writeln(message);
}

That's a valid program, and it runs fine regardless of compiler switches. With the proposal to make assert(__ctfe) / in(__ctfe) skip codegen, you get "undefined reference to `hello'".

>

The whole runtime vs compile time is irrelevant here.

It absolutely matters, Ilya is right that runtime and static properties shouldn't be conflated.

March 25
On 25/03/2025 11:22 AM, Dennis wrote:
> import std.stdio; string hello() in(__ctfe) => "hello world"; immutable funcTable = [&hello]; enum message = funcTable[0](); void main() { writeln(message); }

```d
string hello(int v) in(v > 2) => "hello world";
immutable funcTable = [&hello];
enum message = funcTable[0](1);
```

Oh, its eagerly evaluating out the contract.

I did not know that D did that.

Yup that kills assert and in contracts approach.

Which leaves: ``@__ctfe`` ;)

March 25
On 25/03/2025 12:31 PM, Richard (Rikki) Andrew Cattermole wrote:
> On 25/03/2025 11:22 AM, Dennis wrote:
>> import std.stdio; string hello() in(__ctfe) => "hello world"; immutable funcTable = [&hello]; enum message = funcTable[0](); void main() { writeln(message); }
> 
> ```d
> string hello(int v) in(v > 2) => "hello world";
> immutable funcTable = [&hello];
> enum message = funcTable[0](1);
> ```

Nope, I'm tired. My reply there was wrong.

That enum is calling it in CTFE, and that is indeed working correctly from that perspective.

>> That's a valid program, and it runs fine regardless of compiler switches. With the proposal to make assert(__ctfe) / in(__ctfe) skip codegen, you get "undefined reference to `hello'".

``funcTable`` shouldn't be getting codegen'd either.

I don't know what the status is of variables skipping codegen is.

This will need to be solved regardless of how it gets turned on.

March 25

On Monday, 24 March 2025 at 23:57:56 UTC, Richard (Rikki) Andrew Cattermole wrote:

>

funcTable shouldn't be getting codegen'd either.

I don't know what the status is of variables skipping codegen is.

You can't just skip codegen for variables, then you get 'undefined reference to funcTable' when that is used.

6 days ago

On Tuesday, 25 March 2025 at 00:46:15 UTC, Dennis wrote:

>

On Monday, 24 March 2025 at 23:57:56 UTC, Richard (Rikki) Andrew Cattermole wrote:

>

funcTable shouldn't be getting codegen'd either.

I don't know what the status is of variables skipping codegen is.

You can't just skip codegen for variables, then you get 'undefined reference to funcTable' when that is used.

Anyway guys, the proposal was meant to add a property in which makes the compiler skip codegen for functions which has that property. Although it is cool that we could benefit existing code, we have been finding a lot of divergence about how that feature must be implemented without breaking code.

So, I would like to know if breaking code that uses -release is okay or we just add only the @__ctfe property so we can actually move forward.

Either way, I don't think it will benefit a lot of code since I don't know many projects out there using assert(__ctfe);, that might be really niche, even less possible that it happens to have a project which wrongly uses compile time only code which also produces a release build.

That being said, I'm more interested into alive code which keeps updating, and I really don't want to get that proposal not implemented because we can't decide :)

1 day ago

On Monday, 24 March 2025 at 15:25:16 UTC, Ilya wrote:

>

On Monday, 24 March 2025 at 14:59:17 UTC, Atila Neves wrote:

> >

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.

Contracts have the same conceptual issues as asserts: 1. they are run-time; 2. they are only enabled in debug builds. So both examples from my reply to Walter can be easily adapted to contracts.

Contracts:

  • Are semantically different.
  • Can be switched off independently from asserts via -check.
  • Don't need to parse or do semantic analysis on the bodies of functions.
2 hours ago

On Tuesday, 1 April 2025 at 16:23:55 UTC, Atila Neves wrote:

> >

Contracts have the same conceptual issues as asserts: 1. they are run-time; 2. they are only enabled in debug builds. So both examples from my reply to Walter can be easily adapted to contracts.

Contracts:

  • Are semantically different.
  • Can be switched off independently from asserts via -check.
  • Don't need to parse or do semantic analysis on the bodies of functions.

That's all true, but contracts are still evaluated at run time, so in(__ctfe) contract doesn't guarantee we can skip codegen for such function. I've posted an example with such function being called inside a branch. Dennis posted another example, where one stores and address of such function and later uses that to call it. In both cases we can't skip codegen, even though contracts will be satisfied at run time.

1 2 3 4 5 6
Next ›   Last »