Summary: Rebindable syntax is powerful, but sometimes too flexible. I had some ideas on how to improve it.
In Haskell, when you write 1, GHC turns that into GHC.Num.fromInteger 1, knowing that the binding is GHC.Num.fromInteger :: Num a => Integer -> a. If you want to use a different fromInteger you can turn on the RebindableSyntax extension, which uses whichever fromInteger is in scope. While I was working at Digital Asset on DAML, we built a GHC-based compiler with a different standard library. That standard library eliminates the Char type, has a packed Text instead of String = [Char], doesn't have overloaded numeric literals, renames Monad to Action and other changes. To get that working, we leveraged RebindableSyntax along with a module DA.Internal.RebindableSyntax which is automatically imported into every module unqualified (via a GHC source plugin).
With RebindableSyntax you can get a long way building your own base library, but there were two unpleasant parts:
- When using RebindableSyntax, if the user writeslet fromInteger = undefined in 1then they use thefromIntegerthey defined in thelet, shadowing the global one. For users who turn onRebindableSyntaxdeliberately, that's what they want. However, if you want to replace thebaselibrary and make it feel "just as good", then you'd rather thanfromIntegerwas always some specific library you point at.
- In the process of building fresh base libraries, we had to follow all the (pretty complex!) layering choices that GHC has made about how the modules and packages form a directed acyclic graph. There are some modules where using integer literals would cause a module cycle to appear. The fact that a number of fully qualified names are hardcoded in GHC makes for a fairly tight coupling with the base libraries, that would be better avoided.
I had an idea to solve that, but it's not fully fleshed out, and as of now, it's not a problem I still suffer from. However, I thought it worth dumping my (potentially unimplementable, certainly incomplete) thoughts out for the world, in case someone wants to pick them up.
My idea to solve the problem was to add a flag to GHC such as -fbuiltins=base.Builtins which would specify where to get all builtins. You could expect the base library to gain a module Builtins which reexported everything like fromInteger. With that extension, using RebindableSyntax is then saying "whatever is in scope", and using -fbuiltins is saying "qualify everything with this name" - they start to become fairly similar ideas. I see that has having a few benefits:
- It becomes easier for someone to write a principled standard library which doesn't have String = [Char], or whatever choice wants making, in a way that provides a coherent experience. One example is DAML, but another is thefoundationlibrary, which usesRebindableSyntaxin its example programs.
- The lowest level GHC base libraries can be restructured to use RebindableSyntaxas a way to more easily manage the dependencies between them in the base libraries themselves, rather than a cross-cutting concern with the compiler and base libraries. (This benefit might be possible even today with what we already have. Some people might strongly disagree that it's a benefit.)
- Things like which integer library to use can become a library concern, rather than requiring compiler changes.
- Currently the code path for RebindableSyntaxis always quite different from the normal syntax path. As a result, sometimes it's not quite right and needs patching.
The main obvious disadvantages (beyond potentially the whole thing not being feasible) are that it would cause the compiler to slow down, as currently these types are hard-wired into the compiler.
 




