Microsoftâs Anders Hejlsberg explained that Go was chosen for its TypeScript compiler port because of its native code, garbage collection, and excellent control over memory and data layout.
Last week Microsoft announced that TypeScriptâs compiler is being ported to a new programming language â Go. And then a round of discussions kicked off online⊠Everyoneâs fascinated by their high-stakes decision, and a few commenters couldnât resist second-guessing the development teamâs choice of a new programming language.
Why Go? Why not Microsoftâs own C# or the hot language of the day, Rust?
TypeScriptâs developers soon found themselves in kind of ad hoc colloquium taking place on GitHub, Reddit, YouTube, and Hacker News, explaining their decision-making process and all the clear and undeniable advantages of Go. Amid all the back-and-forth discussion, together they offered a surprisingly educational exploration of the various merits of several different programming languages.
And along the way, they even delivered a detailed explanation for why theyâve decided to port TypeScriptâs compiler to GoâŠ
But Why Not C#?
TypeScriptâs current compiler is written in the TypeScript language. But on Reddit there was a funny yet irrefutable comeback when one commenter suggested that C# âhas almost all you needâ for rewriting TypeScript code.
âWell you can tell that to the guy who created Typescript and C#, but he disagrees with you.â
Indeed, itâs a question Anders Hejlsberg addressed during a video announcing the move. âSome of you might ask, âWell why not my favorite language? Why not C#? Why not Rust? Why not C++?â
Hejlsberg answered that Go was âthe lowest-level language we can get to that gives us full, optimized, native-code support on all platforms, great control over data layout, the ability to have cyclic data structures and so forth. It gives you automatic memory management with a garbage collector and great access to concurrency.â
Hejlsberg also addressed the question in a special Zoom interview for the YouTube channel of a monthly TypeScript Meetup in Ann Arbor, Michigan. Hejlsberg described C# as âbytecode-first, if you will.â Plus, when it comes to performance, C#âs ahead-of-time compilation isnât available on all platforms, and âit doesnât have a decade or more of hardening⊠â
âAnd then I think Go has a little more expressiveness when it comes to data structure layout and inline structs and so forth.â
But there was also something unique about Microsoftâs original TypeScript codebase for its compiler. While C# is an object-oriented language, TypeScript uses âvery few classes,â Hejlsberg said in the Zoom interview. âIn fact, the core compiler doesnât use classes at allâŠ
âGo is functions and data structures, where C# is heavily object-oriented programming (OOP)-oriented. And we would sort of have to switch to an OOP paradigm to move to C#⊠Thereâs just more friction in that transition than there is in the transition to Go.â
On Reddit one commenter even collected all the official answers from Microsoft into a handy chart â that Go had a well-tested native-first option supporting all their desired platforms (unlike C#).
Why Not Rust?
In a Reddit comment that received 1,305 upvotes, TypeScript development lead Ryan Cavanaugh acknowledged the groundswell of curiosity. âWe definitely knew when choosing Go that there were going to be people questioning why we didnât choose Rust (or others).
âItâs a good question because Rust is an excellent language, and barring other constraints, is a strong first choice when writing new native code.â
Both Rust and Go are good at representing data, have âexcellentâ code generation tools, and perform well on single-core systems, Cavanaugh wrote. âIn our opinion, Rust succeeds wildly at its design goals, but âis straightforward to port to Rust from this particular JavaScript codebaseâ is very rationally not one of its design goals.
âItâs not one of Goâs either, but in our case given the way weâve written the code so far, it does turn out to be pretty good at it.â
Cavanaugh shared some insider info: that they did try Rust. But theyâd wanted the new codebase to be âalgorithmically similar to the current one,â and Rust just wasnât a fit. âWe tried tons of approaches to get to a representation that would have made that port approach tractable in Rust, but all of them either had unacceptable trade-offs (performance, ergonomics, etc.) or devolved into âwrite your own Garbage Collectionâ-style strategies. Some of them came close, but often required dropping into lots of unsafe codeâŠâ
Cavanaugh explains in the projectâs FAQ on GitHub that theyâd also tried other languages too, and even âdid deep investigations into the approaches used by existing native TypeScript parsers like swc, oxc, and esbuild.â Cavanaugh says they reached two important conclusions.
- âMany languages would be suitable in a ground-up rewrite situation.â
- âGo did the best when considering multiple criteria that are particular to this situationâŠâ
On Reddit someone pointed out Go gives âmore fine-grained control over memoryâ â and Ryan Cavanaugh jumped in to agree that Go âhas excellent controlâ over memory allocation. Bypassing Goâs built-in allocator with a pool allocator âis extremely straightforward (and easy to experiment with without changing downstream usage).â
In comparison, Cavanaugh adds, âRust has better control of âWhen do you free memory,â but in a type checker there is almost nothing you can free until youâre done doing the entire batch, so you donât really gain anything over a GC model in this scenario.â
In the FAQ Cavanaugh also applauds Goâs control of memory layout â and that it does all this âwithout requiring that the entire codebase continually concern itself with memory management.â
So when it comes to Rust, Cavanaugh explained on Reddit that theyâd face two options:
- âDo a complete from-scratch rewrite in Rust, which could take years and yield an incompatible version of TypeScript that no one could actually useâŠâ
- âJust do a port in Go and get something usable in a year or so and have something thatâs extremely compatible in terms of semantics and extremely competitive in terms of performance.â
In a comment on Hacker News, Cavanaugh reminded readers that in 2022 the SWC (Speedy Web Compiler) project also chose Go for a port of the TypeScript type checker tsc. (Project founder DongYoon Kang noted that tsc âuses a lot of shared mutability and many parts depend on garbage collection. Even though Iâm an advocate and believer in Rust, it doesnât feel like the right tool for the job here.â)
âkdy1 definitely hit on the right general approach with his initial Go port,â Cavanaugh said in a comment on Reddit.
Pros and Cons
In the Zoom interview Anders admits that TypeScript has a âmuch richerâ type system than whatâs available in Go. But on the other hand, Go âdoes actually have excellent support for bit-fiddling and packing flags into ints. And in fact it has much, much, much better support than JavaScript for all of the various data types⊠You can have bytes and short ints and ints and 64-bit ints and what have you, both signed and unsigned⊠In JavaScript, everything is a floating point number. Period.â He laughs. âI mean, you want to represent true or false? Yeah, thatâs eight bytes for you right thereâŠâ
Hejlsberg says thereâs more than just making use of all the bits in Go. âWe can also lay them out as structs â you know, inline, in arrays. And it shows! Our memory consumption is roughly half of what the old compiler was⊠If you can condense your data structures, youâre going to go faster.â
And here Hejlsberg paused to reflect on what a long, strange trip itâs been, saying heâs often laughed about what he wouldâve said if if people had told him, âAnders, youâll be writing compilers in JavaScript for a decadeââŠ
âYou are nuts.â
âJavaScript was never really intended to be the language for compute-intensive, system-level workloadsâŠâ he says. âWhereas Go was precisely intended to be that⊠Go is a system-level tool, and we are a systems-level program.â
Hejlsberg also found himself weighing in on the âWhy Goâ discussion on GitHub, emphasizing that the decision to use Go âunderscores our commitment to pragmatic engineering choices.â
âAt Microsoft, we leverage multiple programming languages including C#, Go, Java, Rust, C++, TypeScript, and others, each chosen carefully based on technical suitability and team productivity.â
Celebrate the Strength
As the original designer of C#, Hejlsberg mustâve taken some pride in pointing out that âC# still happens to be the most popular language internally, by far.â Further down, he even emphasized Microsoft remains invested in C# and .NET âdue to their unmatched productivity, robust ecosystem, and strong scalability.â
And he said theyâd even tried a C# prototype for TypeScriptâs compiler, but âGo emerged as the optimal choice, providing excellent ergonomics for tree traversal, ease of memory allocation, and a code structure that closely mirrors the existing compiler, enabling easier maintenance and compatibility.â
Starting from scratch wouldâve been an entirely different question, Hejlsberg adds, but âthis was not a green field â itâs a port of an existing codebase with 100 man-years of investmentâŠâ With a codebase that was âall functions and data structuresâ â with no classes â Go just turned out to be âmore one-to-one in its mapping⊠Idiomatic Go looked just like our existing codebase, so the port was greatly simplified.â
Hejlsberg used the moment to make the case that âat Microsoft, we celebrate the strength that comes from diversity in programming languages.â
And his comment also closed with some historical perspective. âLetâs be real. Microsoft using Go to write a compiler for TypeScript wouldnât have been possible or conceivable in years past.â But âover the last few decades, weâve seen Microsoftâs strong and ongoing commitment to open-source software, prioritizing developer productivity and community collaboration above all.â Today, Hejlsberg argued, Microsoft sought to âempower developers with the best tools available, unencumbered by internal politics or narrow constraints.
âThis freedom to choose the right tool for each specific job ultimately benefits the entire developer community, driving innovation, efficiency, and improved outcomes. And you canât argue with a 10x outcome!â
SHARE THIS STORY