trusted purity?
bearophile
bearophileHUGS at lycos.com
Mon Aug 19 05:55:52 PDT 2013
monarch_dodra:
> FYI, the problem I'm trying to fix is this one:
> * "uninitializedArray" returns an array with un-initialized
> elements. This, by definition, is not pure, since the value
> returned is garbage. I'm fixing the function so that it becomes
> *impure*.
> * "array" is implemented in terms of "uninitializedArray":
> Allocate an array, and then fill it. "array" is pure, since its
> return is defined. array also works with ctfe.
Here are some examples of what I think you are talking about.
foo1 is pure, foo2 is pure (but currently the type system doesn't
complain), foo3 is pure again because all array items are
deterministically initialized using only pure/immutable data:
// Pure:
int[3] foo1(in int x) pure {
int[3] arr;
return arr;
}
// Not pure:
int[3] foo2(in int x) pure {
int[3] arr = void;
return arr;
}
// Pure:
int[3] foo3(in int x) pure {
int[3] arr = void;
arr[] = x;
return arr;
}
void main() {}
I presume foo2 should be refused as not pure. The array()
function is like foo3, it creates data that is not pure, not
deterministic, but then overwrites it all with referentially
transparent information. So on the whole foo3 is pure and array()
is often pure.
The problem is that while writing down the proof of the purity of
foo3 is probably not too much hard, later the D compiler is not
able to verify such proof. So some alternative solution is
needed. The trusted pure you talk about is a solution, it means
saying to the compiler, "trust me I have a proof of purity of
this function". But programmers should be trusted as little as
possible if you want a reliable language and reliable programs.
So perhaps some mid-way solution is preferable.
Andrei used the idea of cooked and uncooked variables, it's
probably used here:
class Bar1 {
immutable int[2] x;
this() {
}
}
Bar1 gives the error:
Error: constructor test.Bar1.this missing initializer for
immutable field x
While this gives no errors:
class Bar2 {
immutable int[2] x;
this() {
x[0] = 1;
}
}
Perhaps using a similar strategy you can accept a function like
this:
int[3] foo3(in int x) pure {
int[3] arr = void;
arr[] = x;
return arr;
}
Because now arr is not uncooked, it was overwritten by
referentially transparent data...
For the function array() this is not enough, because instead of
"= void" you have a function that returns some void-initialized
data. To help the D type system a bit perhaps an annotations like
@void_init is useful, to be attached to functions like
uninitializedArray and minimallyInitializedArray.
Bye,
bearophile
More information about the Digitalmars-d
mailing list