where constraintsThere are a variety of contexts that currently use the keyword let:
In all but the implementation case, the semantics are generally similar to the
semantics of passing a value into a function, with some erasing of the specific
value passed and using the type to determine how the name can legally be used.
However,
proposal #950 has
changed the let in an implementation to use the value specified, not its type,
creating an inconsistency with the other uses of let.
Furthermore, we have come to the realization that we still want to specify the
values of associated constants and types for an implementation even in an API
file where we only want to make a forward declaration. This makes that
information available to clients that only look at the API file, who need to
know those values for type checking, but otherwise don't need to see the full
definition of the implementation. This suggests that those assignments should be
declared outside of definition block's curly braces {...}.
Lastly, there is a bit of redundancy in Carbon since where clauses are also a
way of specifying the values of associated constants and types in other Carbon
contexts.
The let syntax for setting an associated type in an interface implementation
was originally decided in issue
#739: Associated type syntax
and implemented in proposal
#731: Generics details 2: adapters, associated types, parameterized interfaces.
Proposal #950: Generics details 6: remove facets made two relevant changes:
let in an impl block is no longer "load bearing": the
only legal types are auto and whatever was in the corresponding interface.
In particular, the let in an impl block does not erase.let statement in a function
body that can erase depending on the type specified.Combined with the let in an interface giving you an erased type, or archetype,
this has made the meaning of let in an impl block inconsistent with other
places using let.
The suggested change is to use a where clause as part of an impl declaration
to specify associated constants and types instead of let declarations inside
of the impl definition. In effect, it removes let declarations from impl
blocks in exchange for allowing an impl declaration to implement a constraint
expression instead of a simple interface or named constraint.
This proposal updates the following design docs on the generics feature to reflect this change:
As a simplification, this proposal advances the goal of having Carbon code that is easy to read, understand, and write. In particular, having a simple specification and be simple to implement.
This is an example of the "prefer providing only one way to do a given thing" principle, by switching to a single way of specifying associated constants and values.
The main alternative considered was the status quo. We did have two concerns with this proposal, however we felt that this behavior would not be surprising to developers in practice.
Concern: Due to interface defaults, it is possible for copy-pasting the
type-of-type expression from an impl block in a class into a constraint in a
function signature to give a constraint that is weaker than what that impl block
actually delivers.
Concern: Because a specialization of an impl can change the values of
associated constants, a type might not actually satisfy a constraint that it
appears to implement when that constraint specifies the values of associated
constants. In this example:
interface Bar {
let X:! Type;
}
class Foo(T:! Type) {
impl as Bar where .X = T { ... }
}
it appears that Foo(T) satisfies the constraint that Bar where .X = T, but
there could be specializations that set .X to different values for some
specific values of T.
with and , instead of where and andInstead of matching the syntax used when specifying constraints, we could have
used a different syntax to highlight that this is assigning instead of
constraining. The suggestion that came up in discussion was using with instead
of where and a comma , instead of and to join multiple clauses.
We decided that it would not be good to have two syntaxes that were very similar but different, and that there was some benefit to be able to copy-paste between the constraint context and the implementation context.
This proposal will allow us to support declaring that a type implements an
interface inside an API file separate from the definition of the impl, even
for internal impls. However, that feature is waiting on resolution of
#472: Open question: Calling functions defined later in the same file
and proposal
#875: Principle: information accumulation.
If and when we do add support declaration of impls without definition, we will
need to answer the question: do you have to repeat where constraints from a
forward declaration of an impl when it is later defined?
class Vector(T:! Type) {
impl as Container where .Element = T and .Iter = VectIter(T);
}
// Probably okay:
fn Vector(T:! Type).(Container.Begin)[me: Self]() ...
// Maybe okay:
class Vector(T:! Type) {
// Not repeating constraints on .Element and .Iter above:
impl as Container {
fn Begin[me: Self]() ...
}
}