Want to take part in these discussions? Sign in if you have an account, or apply for one below
Vanilla 1.1.10 is a product of Lussumo. More Information: Documentation, Community Support.
You’d think type refinement might crop up here, and I see a gesture in that direction at least at slide 45 of one of Noam’s talks.
I see also bidirectional typing is mentioned in the abstract of Noam’s A theory of linear typings as flows on 3-valent graphs. I guess there’s interesting combinatorics here.
Maybe there’s something in
I feel like they’re a bit orthogonal. Refinement typing is, as I understand it, about assigning new, more informative, types to terms that already had types. Assigning such types, as well as assigning the less informative types in the first place, could both be done bidirectionally or not. (E.g. it looks like in the Lovas-Pfenning paper, both are bidirectional.)
Hi Mike, thanks for starting this article!
I feel like they’re a bit orthogonal. Refinement typing is, as I understand it, about assigning new, more informative, types to terms that already had types. Assigning such types, as well as assigning the less informative types in the first place, could both be done bidirectionally or not. (E.g. it looks like in the Lovas-Pfenning paper, both are bidirectional.)
You’re right that they can be considered independently, but there are also several connections. On the one hand, bidirectional typing may be naturally seen as a refinement of ordinary typing. Indeed, in the current version of the article you wrote that “these observations correspond to bidirectional refinements of the usual typing rules for abstractions and applications”! I know you didn’t mean that in a technical sense, but in fact it does constitute an example of type refinement, which was considered (in the closely related form of “normal natural deduction”) already in Pfenning’s original paper on Refinement Types for Logical Frameworks.
The other connection is that type checking itself (whether bidirectional or not) is an extrinsic notion, taking a term (perhaps with some annotations) and a type and deciding whether the term has that type. In bidirectional systems, the synthesis judgment is often designed to compute the principal type of a term in a given context, which is another extrinsic notion (if Γ⊢t⇒A, then A is the most precise type that can be assigned to t in the context Γ).
Great to have you back, Noam. Your final paragraph of #6 expresses well the hunch I had in #2.
I thought you might be at SYCO 1, but you must have been away. I’m struggling with lenses at the moment, which seem to have a whiff of type refinement to them. But then functors are everywhere.
Thanks Noam. I don’t quite understand what the last paragraph of #6 has to do with refinement? The word “refinement” doesn’t appear in it.
The other paragraph confuses me a little because I would say that it’s not the types that are being refined but the judgments. But I suppose you could consider this another kind of “typing-judgment refinement”?
I don’t quite understand what the last paragraph of #6 has to do with refinement? The word “refinement” doesn’t appear in it.
As I’ve talked about before, I believe that extrinsic typing in general is best analyzed in terms of type refinement: an extrinsic type should always refine some intrinsic type, whether or not we make the latter explicit. In categorical terms, we speak of a type refinement system as a category 𝒟 of extrinsic types and typing derivations living over a category 𝒯 of intrinsic types and terms. In the limiting case, 𝒯 is a one-object category (or one-object multicategory = operad), but even in that case it is often useful to explicitly consider the “refinement functor” 𝒟→𝒯 (e.g., as mentioned at fibration of multicategories, a representable multicategory can be considered equivalently as a fibration over the terminal multicategory).
The other paragraph confuses me a little because I would say that it’s not the types that are being refined but the judgments. But I suppose you could consider this another kind of “typing-judgment refinement”?
In Pfenning’s example it is actually the types (or to be precise, families) which are being refined: one type “pf” is refined into two types “nf” and “elim”. Of course he is taking a judgments-as-types approach to representing deductive systems in a logical framework.
Type refinement is about lifting arrows from the coarse system to the more refined system, so perhaps could be renamed as you suggest.
We might have λx.x as an arrow is some coarse system and wonder how it can be lifted. The most general presentation of the lift is the ’principal type’, so in this case X→X (for any X), rather than any more specific candidate such as Bool→Bool. (See slide 14 of Noam’s talk to us at Kent.)
So I guess from the principal type, you can do type checking against a given type.
David wrote:
I thought you might be at SYCO 1, but you must have been away.
Yes, I was very disappointed to miss it! But I had already planned a trip to the US before the dates were announced.
I’m struggling with lenses at the moment, which seem to have a whiff of type refinement to them.
Unfortunately I don’t know much about lenses…but I should read up about them sometime, thanks for the pointer.
David #12:
Beware that there are two different notions of “principal type” in the type-theoretic literature: both a “local” (contextual) notion of principal type of a term relative to a given context (what I was referring to in #6), and a “global” notion of principal type of a term that includes types for its free variables (sometimes called a “principal type scheme”, and what I was referring to on slide 14 of the Kent talk). I discussed a bit more about this distinction in Section 2.2.4 of these notes on type refinement. In categorical terms, I view the local notion of principal type as something akin to the pushforward of a typing context Γ along a term t, and the global notion as something like an initial object in an appropriate category of typings for t. (This is something I would like to make more precise, though.)
So I guess from the principal type, you can do type checking against a given type.
Yes, for both notions of “principal type”. For the contextual notion of principal type used in bidirectional typing, to check that Γ⊢t:B, we synthesize a principal type Γ⊢t⇒A and check that A≤B. For the global notion of principal type scheme, we synthesize both a type A and a typing context Δ for the free variables of t, both of which can involve free type variables. Then to check that Γ⊢t:B, we have to decide whether there exists a substitution σ for the type variables such that Γ≤σΔ and σA≤B (or Γ=σΔ and σA=B in systems without subtyping).
…and to make things even trickier, the addition of polymorphism to a type system can sometimes turn the global notion into the local notion: that’s the way I interpret “let-polymorphism” in the style of Milner’s original paper on A Theory of Type Polymorphism in Programming. In that system, the closed term λx.x may be given polymorphic type ∀X.X→X, which is a subtype of any other type that may be assigned to λx.x in the empty context.
Thanks, Noam.
As for lenses, one way to get an idea is to read this brief account of how they’ve been invented 9 times, starting off with Gödel’s Dialectica interpretation.
I certainly agree that it’s sometimes useful to consider categories as fibrations over 1. But I’m a little confused about how this relates to a view of extrinsic typing as type refinement. I thought type refinement was about assigning more refined types to the same terms. But then if we are to view extrinsic typing as a refinement of a “trivial” intrinsic typing, it would seem that the latter would have to have one object but all the (possibly ill-typed) terms as morphisms. Which may be fine from a syntactic point of view, but I think in most semantic models of the refined system there is no base category in which all the possibly-ill-typed terms can be interpreted as anything at all.
I think it would help to have a more concrete type system/semantics to talk about. But I would suggest that you first have a look at the example from Reynolds’ The Meaning of Types: from Intrinsic to Extrinsic Semantics that Paul-André and I formalized in section 6 of our POPL paper. In this example, the base category 𝒯 is indeed freely generated over a single type P of untyped syntactic “phrases”. Then one can (as you suggest) define a semantic functor 𝒟→Dom that completely ignores 𝒯 – what Reynolds called an “intrinsic semantics”. But in order to even state the coherence theorem for this semantics one needs to reference the forgetful functor 𝒟→𝒯, and moreover, Reynolds observed that there is an elegant proof of the coherence theorem that works by first defining another “untyped semantics” 𝒯→Dom (which relies on Dom’s ability to model recursive types), and constructing a logical relation between the two interpretations.
Sure, for semantics in Dom I expect you can model untyped syntax already, but that’s a pretty special case; most categorical semantics don’t have general recursion.
Perhaps Mike’s own type theories would help understanding. Aren’t those modal type theories of the LSR program (as explained in the HoTTEST talk) great examples, where the mode theory of the base is refined to the full theory up top?
Mike #19:
Sure, for semantics in Dom I expect you can model untyped syntax already, but that’s a pretty special case; most categorical semantics don’t have general recursion.
I’m not sure what conclusion you are drawing. In Reynolds’ example there are two different semantics, corresponding to two functors [[−]]D:𝒟→Dom and [[−]]T:𝒯→Dom. The interpretation [[−]]D does not rely on any special properties of Dom other than cartesian closure. The interpretation [[−]]T does rely on special properties of Dom (namely, existence of recursive types), but this is what leads to his elegant proof of the coherence of [[−]]D relative to the original type system. So there is no barrier to taking the target of [[−]]D to be another ccc (say, Set), but you would have to devise another proof of coherence. (I suspect that Reynolds’ proof could be easily adapted taking the logical relation to go between sets and domains, although I haven’t worked that out.)
David #20:
Aren’t those modal type theories of the LSR program (as explained in the HoTTEST talk) great examples, where the mode theory of the base is refined to the full theory up top?
That is indeed the way Paul-André and I like to analyze the example of reasoning in substructural logics, with proofs corresponding to morphisms in 𝒟 that live over “context effects” in 𝒯.
My point is that thinking about 𝒟 as fibered over 𝒯 isn’t useful for the majority of semantics, which seems to be exactly what you’re saying: for semantics of 𝒟 in Dom, you can use 𝒯 to prove coherence, but for general semantics you have to do something else.
My point is that thinking about 𝒟 as fibered over 𝒯 isn’t useful for the majority of semantics, which seems to be exactly what you’re saying
No, that’s not what I’m saying, and I certainly think that it is useful. I explained how Reynolds’ proof uses the semantics [[−]]T:𝒯→Dom in order to prove the coherence of his semantics [[−]]D:𝒟→Dom, and suggested that it could probably be adapted to proving coherence of an alternative “intrinsic” semantics [[−]]D of typing derivations valued in some other ccc such as Set. Also, I made the point that knowledge of the forgetful functor 𝒟→𝒯 is implicitly required even to make the statement of the coherence theorem.
Sorry if I am not being clear, but again I suggest you have a look at Reynolds’ paper and/or our analysis in the POPL paper.
I looked at the papers, but I’m still confused. I think I don’t even understand what you mean by “coherent”. I see the statement is “the interpretation of a judgment doesn’t depend on its derivation”, but in that case I don’t see how you can have a functor 𝒟→Dom that isn’t coherent; doesn’t the construction of a syntactic category already mod out by the existence of different derivations of the same term?
doesn’t the construction of a syntactic category already mod out by the existence of different derivations of the same term?
Do you mean different derivations of the same typing judgment, rather than of the same term? In any case, the construction of 𝒟 doesn’t mod out by either. In general, we do not ask that a type refinement system 𝒟→𝒯 is a faithful functor, since we do not want to rule out the possibility of having essentially different typing derivations of the same judgment.
You could certainly construct the quotient category 𝒟/∼ with the same objects as 𝒟 and where two morphisms of 𝒟 are identified just in case they have the same domain and codomain and map to the same morphism in 𝒯 (i.e., they are derivations of the same typing judgment). But then proving that a functor 𝒟→𝒳 lifts to a functor 𝒟/∼→𝒳 would amount to the same work as proving coherence.
To quote Reynolds: “In fact, as we will see in Section 5, our intrinsic semantics is coherent. However, this fact depends critically on the details of the language design, especially of the semantics of implicit conversions and “overloaded” operators with more than one signature.”
In other words, the coherence theorem is accomplishing something non-trivial, and I would suggest that this thing is best expressed in categorical language (here: for any two morphisms α1,α2 in 𝒟 with the same domain and codomain, if U(α1)=U(α2) then [[α1]]=[[α2]]), rather than as a precondition for the use of that language.
It’s not even clear to me that you get a category if you don’t quotient by anything, let alone a category with all the structure you expect to have. All the usual constructions of the syntactic category that I’ve seen take the morphisms to be typing judgments.
Mike, I think you must be reading more into the construction than what we explain in the paper. The whole point is to have two categories 𝒟 and 𝒯 related by a functor U:𝒟→𝒯, and thereby distinguish between “terms” (= morphisms of 𝒯), “typing judgments” (= morphisms of 𝒯 paired with objects of 𝒟 refining their domain and codomain), and “typing derivations” (= morphisms of 𝒟), rather than taking the usual syntactic category of a type theory which collapses the three notions. This proof-theoretic reading of a functor as a type refinement system can be applied whenever you have a functor U:𝒟→𝒯 (even if it is not always so evocative), because it is just a collection of elementary definitions.
In Section 6 of the POPL paper, we explain how to construct a particular “syntactic type refinement system” U:𝒟→𝒯 corresponding to Reynolds’ type system by first constructing 𝒯 as a freely generated category, then constructing 𝒟 as another freely generated category (with a priori no relation to 𝒯), and finally defining U to express the desired typing laws. Alternatively, this could be seen more directly as the construction of a freely generated displayed category 𝒯→Prof over the freely generated category 𝒯. I don’t think there is anything suspect here, it is just perhaps a different way of looking at type systems categorically than most people are used to.
Well, you can always get a category by postulating that you have a category, but I prefer honest toil. I don’t think it’s reasonable to use words like “term” and “typing judgment” if what you’re really talking about is just a category, rather than actual syntax. Even your section 6, if I’m reading it correctly, doesn’t actually start with Reynolds’ syntax, but generates some category (thereby postulating that you have a category) subject to some rules that look kind of like the rules of Reynolds’ syntax.
As explained at the beginning of the section, we give an encoding of his language + type system in the style of LF, using higher-order abstract syntax. An alternative that is closer to Reynolds’ original syntax (and which you might find more palatable) is to construct 𝒟 and 𝒯 as multicategories.
In any case, part of the “honest toil” in this work was the conceptual one of realizing that many extrinsic typing concepts (such as subtyping and subsumption) are explained by passing from categories to functors, and that moreover this is in harmony with a fibrational view of substructural proof theories such as separation logic (which, as David said above, is very much in the spirit of the LSR program). Perhaps it would focus the discussion if you said what kind of (alternative?) analysis you would give for extrinsic typing?
As explained at the beginning of the section, we give an encoding of his language + type system in the style of LF, using higher-order abstract syntax.
That’s indeed what you say at the beginning of the section, but the actual definition later on says that T is a cartesian category freely generated by an object and some morphisms, and that U:D→T is the “cartesian logical refinement system” — which you’ve defined to be a certain kind of category and functor — freely generated by some other stuff. I don’t think it’s old-fashioned to insist that syntax (even higher-order abstract syntax) is not the same as semantics.
I don’t mean to belittle the conceptual insight of using functors to represent aspects of extrinsic typing. But I think there’s more to be done in using this insight to actually prove theorems about syntax.
I don’t think it’s old-fashioned to insist that syntax (even higher-order abstract syntax) is not the same as semantics.
It does seem to me somewhat old-fashioned to insist that freely generated categories cannot count as syntax. Would you say that a Lawvere theory is semantics rather than syntax? In this case, I think that the difference between Reynolds’ type system and our encoding is sufficiently small for the latter to count as “syntax”, and that our categorical formalization really captures the essence of his proof.
That said, I think I do understand your broader point, that you would like to have better tools/theorems for proving results about concrete type systems without having to think too hard about questions of adequacy. The lack of such tools is a criticism I’ve heard before about this work with Paul-André, and something that I agree should be addressed. Personally, though, I don’t think it is helpful to phrase this criticism in terms of ambiguous and loaded expressions like “syntax vs semantics”. Instead it is better to appeal to well-defined mathematical distinctions, like “category vs multicategory”, “tree vs graph”, etc. In computer science, we are used to the idea that a program can have many different representations, e.g.:
and so on. I wouldn’t say that any of these particular representations has the unique privilege of being counted as “syntax” (or as “semantics”), but we can nonetheless make meaningful distinctions between them and apply them to different purposes.
I agree that there is a continuum between syntax and semantics, and there’s a sense in which a free category is syntax. But it’s not the syntax that one usually writes down as a type theory, because it has a primitive cut/substitution rule and is quotiented by an explicit associativity axiom. Relating this kind of “syntax” to the more usual type-theoretic syntax with implicit substitutions involves something like cut-elimination / normalization / initiality theorems. You can call your free category syntax, but then the onus is on you to relate it to what other people call syntax. Maybe it’s obvious to you, but it’s not obvious to me that the primitive cut/substitution rule and associativity quotient in your free category give you something even essentially analogous to the derivation trees in a deductive system.
Relating this kind of “syntax” to the more usual type-theoretic syntax with implicit substitutions involves something like cut-elimination / normalization / initiality theorems.
Agreed 100%. Though I would add (and perhaps you agree) that it should be possible to express such theorems directly in categorical language, rather than seeing them as a “wormhole” between strangely separated universes of type theory and category theory.
Maybe it’s obvious to you, but it’s not obvious to me that the primitive cut/substitution rule and associativity quotient in your free category give you something even essentially analogous to the derivation trees in a deductive system.
Yes I think it is clear in this case, but agree that it would be nice to have a theorem in the literature to refer to. For example, in this case the appropriate theorem is that equivalence classes of morphisms A→B in a ccc generated freely over some collection of operations Σ are in one-to-one correspondence with βη-normal simply-typed lambda terms x:A⊢Σt:B generated freely over the signature Σ (hence, “higher-order abstract syntax”).
Joshua Dunfield and Neel Krishnaswami wrote a survey article about bidirectional type checking, and Neel linked to a draft/preprint from a blog post. I’m reading it, and it seems very good and thorough. I figure this page should link to it, right?
Yes, of course.
Thanks for opening this conversation, Jon. (Although I am a bit puzzled by the choice of thread – what does it have to do with bidirectional typechecking?)
Maybe we can start out by being a little more clear about what’s being contrasted. At some points it seems to me that you’re conflating what I see as three different axes:
You didn’t say anything about the third one explicitly, but when you wrote “implicit substitutions (don’t forget capture avoiding!)” the most obvious interpretation to me is as a reference to it, since I usually think of “variable capture” as a concept pertaining only to the use of named variables. One does have to do reindexing when substituting with de Bruijn indices, but I don’t think I’ve heard that described as “capture avoidance” before – although that could just be my ignorance.
My understanding is that in extriniscally typed syntax, one can pair either kind of substitution with either kind of variable, so the second and third distinctions are orthogonal. Whereas intrinsically typed syntax seems to require explicit substitutions and de Bruijn indices. So when you say “working at a level of abstraction where it is possible to see the difference between implicit and explicit substitutions” I would phrase it as instead “working with definitions where it is possible to use implicit substitutions and named variables.” (I wouldn’t say that intrinsically typed and extrinsically typed syntax have different amounts of abstraction.)
Overall, it seems to me that you are arguing for what I would call intrinsically typed syntax, not for explicit substitutions. One can have a syntax with raw terms and typing judgments and explicit substitutions in the raw terms, but that’s not what you want. Right?
One other small remark to begin with is that it seems a bit contradictory to me to say that
it would not be taken seriously to argue that the expressions we write or speak are passing through a translation involving the concrete syntax of the ZFC formalism
but then acknowledge that the argument you’re advancing is
in contrast to the point of view which pervades the field of formal logic.
Perhaps I am misunderstanding what you mean by one or both of these two phrases?
Re #38:
I agree that type theory is not really about syntax, but I don’t agree that type theory is about directly working with mathematical objects. It cannot be, because then it would not be different from math itself.
I believe type theory is about the presentation of mathematical objects. Math doesn’t need to be presented directly by syntax. Computational type theory presents math using computations (roughly: behavioral equivalence classes of programs), and material set theory presents math using material sets. I consider these approaches to be legitimate styles of type theory, and I lament that they’re often not treated as such. (If the presentation is not syntax, there’s the separate technical issue of accessing the presentation via syntax. That is not (necessarily) type theory.)
Syntax is the least common denominator style of presentation. Everyone understands the gist of syntax, whether they realize it or not. Type theory is not limited to syntax, but it sure works with it. Syntactic type systems are a way to develop type theoretic ideas by example.
So when it comes to concrete syntax and deciding what aspects of it are important, I think mostly about what is needed in computerized implementations. Here the answers are very clear: implementations of type theory do _not_ need untyped substitutions to exist
That’s true. But it’s mostly irrelevant to implementation, because the notion of typed substitution that’s needed is the one you’re supposed to be implementing, not one that you already have available. The practical question is whether to implement the substitution of the new type system in terms of untyped operations or operations taken from some other type system. It’s quite important that untyped implementations exist, because otherwise there’d be no reason to believe that type theory can be implemented using almost any programming language.
Possible advantages of typed implementations are worth considering, but the more exotic features of the (metalanguage!) type system they require, the less broadly applicable they are, without including an implementation of parts of the metalanguage. Eventually, there needs to be a reduction to something ubiquitous.
If one requires other substitutions to exist which are not captured by the substitution rule, then there is of course a very real difference which we can discuss.
OK, I’m curious. Tell us about it.
I am being a bit provocative and polemical because I hope that I can spark a debate, and that people who believe that implicit substitution vs explicit substitution is a sticking point in the study of type theories can defend their positions.
I admit to preferring implicit substitutions, although they’re the worse-performing implementation, and are more remote from category theory. They’re just so convenient. But I consider it an implementation detail. So no debate from me about that.
Re #39:
Whereas intrinsically typed syntax seems to require explicit substitutions and de Bruijn indices.
I don’t think that’s true. I think that’s just the combination that’s been worked out because the Agda camp likes it.
I guess with a “quotient inductive-recursive definition” you could write down something that could arguably be called intrinsically typed syntax with implicit substitutions? I’m generally rather leery of induction-recursion (and especially quotient/higher induction-recursion) because I’ve never seen a convincing explanation of its categorical semantics in general. In particular, one selling point of intrinsic syntax seems to be its essentially-algebraic character, and I’m unclear on whether that could be applied to the inductive-recursive case. But yes, I can see that it probably at least exists.
1 to 46 of 46