/swift-to-js

Swift to JavaScript compiler using swiftc's AST printing facility

Primary LanguageTypeScript

swift-to-js

This is an experimental JavaScript backend for the standard Swift compiler. It's not ready for even beta use yet as much of the language is missing.

Installation

Install Swift

Choose and install a compatible Swift 5.0 snapshot from swift.org/download/. 2018-12-16's snapshot build is known to be compatible. For macOS, a recent version of Xcode is required.

Install Node & NPM

You know what to do...

Clone and run the project

$ git clone git@github.com:rpetrich/swift-to-js.git
$ cd swift-to-js
$ npm install
$ npm run build
$ node . test.swift

Run the tests

$ npm test

Development Notes

Approach used in swift-to-js is to translate the AST produced by swiftc -dump-ast -- file.swift incrementally using the type information provided by the compiler at each step to determine how to translate. Walking each AST node produces a partially-translated value on which peephole optimization can be performed before baking into the node to insert into an ever-growing babylon AST. Generated code uses ES6 features including module syntax. Basic support for source maps is supported, but not all mappings are applied correctly.

It relies heavily on swiftc's proper pretty-printing of the AST. As the AST isn't documented or supported and changes from version-to-version—staying up to date with the latest swift releases may prove a chore! PEG grammars are provided for swift's pretty-printed AST, declaration reference syntax, and type declaration syntax. These were built piecemeal by examining output from the compiler and likely aren't exhaustive. Similarly, many details of the standard library's internal structure must be replicated directly and can drift from upstream.

Types are categorized into simple value types, complex value types and reference types. Simple value types are mapped directly to a single primitive type in JavaScript. Many Swift types are represented natively as the closest JavaScript types, and appropriate conversion/checking/boxing code is integrated when values are converted between types. Complex types are deep copied by value whenever they are returned or assigned into another value, including when inside data structures or wrapper types. Reference types will be implemented as ES6 classes (was in previous SSA-based implementation)

Simple value types are represented as the closest JavaScript equivalent. Bool becomes boolean, Double becomes number and String becomes string. Similarly, Int, Float, and UInt also become number, and optional runtime checks/conversions are inserted automatically to ensure values stay in the proper ranges.

Structs are represented as JavaScript objects containing the appropriate fields and are automatically copied when necessary to preserve Swift's semantics. Much work has been done to remove many superfluous copies. Empty structs are represented as undefined and unary structs are represented as the underlying value so that wrapper and trait-only types can be zero overhead.

Basic support for consuming generic APIs is implemented (via Optional, Array, Dictionary and numerous protocol types) including cases where the full type is only known at runtime. Support for calling generic functions at runtime is via witness tables that are generated at the call site. Mutation through a witness table isn't entirely correct, but will be made correct in the future.

Optionals are implemented by representing .none as null and .some(T) as the underlying value for T. null is chosen instead of undefined somewhat arbitrarily. Nested optionals are implemented by boxing inside an array to avoid the ambiguity between .none and .some(.none). Bool??.none becomes [], Bool??.some(Bool?.none) becomes [null], Bool??.some(Bool?.some(true)) becomes [true]. Optimization for most operations on Optional is implemented.

Tuples are implemented as arrays of the appropriate length, except unary tuples are represented as the underlying value and empty tuples (the unit type) are represented as undefined. Tuples are considered a complex value type and will be deep copied as needed. Binary and larger tuples containing only simple values are optimized into .slice() calls when copied.

Array types are implemented as JavaScript arrays, with deep copying of embedded value types implemented if necessary. All array operations are bounds-checked with panics when the array is accessed outside its range—reads are allowed within the bounds and writes are allowed one passed the end to match Swift's behaviour.

Dictionary types are implemented as a JavaScript object and support only primitive keys known at compiletype. For non-string keys, appropriate conversion functions are implemented when reading keys. swift-to-js may migrate to using ES6 Maps for some types, but will need a custom map to support compound value types as keys.

Simple enums are implemented as a number representing each case. Enums that contain fields are implemented as an array with the first element representing the discriminant index and case fields stored starting at element index 1. If any field of any case requires copying a deep copy helper will be emitted for every assignment, otherwise a simple call to slice. In the future it may be possible that enums containing values with disjoint representations could even be stored unboxed. This requires more research.

Exceptions will be implemented as normal JavaScript exceptions and try/catch blocks (was in previous SSA-based implementation). Panics are, unfortunately, also implemented as exceptions leading to collisions in any code that recovers from an exception. More work will need to be done to discriminate Swift exceptions from panics and JavaScript exceptions.

Constructors, private functions, any functions that call themselves recursively, and any library functions implemented in builtins.ts not marked with noinline will be inlined into calling functions. This avoids some amount of code bloat with Swift's wrapper type functions present in the AST and allows copies of complex value types to be elided. In the future, better inlining decisions can be made.

inout parameters are supported only in cases where the type is known fully. Intention is to use the runtime type information to determine how to create boxes and to read/write through boxes.

Destructors aren't supported and won't ever be due to JavaScript's GC model. Intention is to warn during compilation if any complex/impure destructors are discovered in the generated AST.

Weak references are converted to strong references due to JavaScript's GC model and lack of support for weak references.

Names are mangled based on a few simple rules. Symbols that aren't supported in JavaScript identifiers are converted to $name$ format. For example, the == function is converted to $equals$. Method/function names are mangled to include all named arguments separated by $. Internal helper functions are always prefixed by a $ symbol, because $ obviously represents Swift and not JQuery.

Source maps are produced by populating the bablyon AST's loc properties with data the Swift compiler includes in it's AST's range properties. When peephole optimizations remove operations, the inner-most source location is generally preserved. Mapping information will be missing on purely generated code and any code inlined from standard library functions.