Kotlin Help

Interoperability with Swift using Swift export

Kotlin provides experimental support for Swift export. It allows you to export Kotlin sources directly and call Kotlin code from Swift idiomatically, eliminating the need for Objective-C headers.

Swift export makes multiplatform development for Apple targets more streamlined. For example, if you have a Kotlin module with top-level functions, Swift export enables clean, module-specific imports, removing the confusing Objective-C underscores and mangled names.

Current Swift export features are:

  • Multi-module support. Each Kotlin module is exported as a separate Swift module, simplifying function calls.

  • Package support. Kotlin packages are explicitly preserved during export, avoiding naming conflicts in the generated Swift code.

  • Type aliases. Kotlin type aliases are exported and preserved in Swift, improving readability.

  • Enhanced nullability for primitives. Unlike Objective-C interop, which required boxing types like Int? into wrapper classes like KotlinInt to preserve nullability, Swift export converts nullability information directly.

  • Overloads. You can call Kotlin's overloaded functions in Swift without ambiguity.

  • Flattened package structure. You can translate Kotlin packages into Swift enums, removing the package prefix from generated Swift code.

  • Module name customization. You can customize the resulting Swift module names in the Gradle configuration of your Kotlin project.

Enable Swift export

The feature is currently Experimental and not ready for production. To try it out, configure the build file in your Kotlin project and set up Xcode to integrate Swift export.

Configure Kotlin project

You can use the following build file in your project as a starting point for setting up Swift export:

// build.gradle.kts kotlin { iosArm64() iosSimulatorArm64() swiftExport { // Set the root module name moduleName = "Shared" // Set the collapse rule // Removes package prefix from generated Swift code flattenPackage = "com.example.sandbox" // Configure external modules export export(project(":subproject")) { // Set the name for the exported module moduleName = "Subproject" // Set the collapse rule for the exported dependency flattenPackage = "com.subproject.library" } // Provide compiler arguments to link tasks configure { freeCompilerArgs.add("-Xexpect-actual-classes") } } }

The Kotlin compiler automatically generates all the necessary files (including swiftmodule files, static .a library, a header file, and a modulemap file) and copies them into the app's build directory, which you can access from Xcode.

Configure Xcode project

To configure Xcode to integrate Swift export into your project:

  1. In Xcode, open the project settings.

  2. On the Build Phases tab, locate the Run Script phase with the embedAndSignAppleFrameworkForXcode task.

  3. Replace the script with the embedSwiftExportForXcode task in the run script phase:

    ./gradlew :<Shared module name>:embedSwiftExportForXcode
    Add the Swift export script
  4. Build the project. The build generates Swift modules in the output directory.

Current limitations

Swift export currently works only in projects that use direct integration to connect the iOS framework to the Xcode project. This is a standard configuration for Kotlin Multiplatform projects created with the Kotlin Multiplatform plugin in IntelliJ IDEA or through the web wizard.

Other known issues:

  • Swift export breaks when modules have the same name in Gradle coordinates, for example SQLDelight's Runtime module and Compose Runtime module (KT-80185).

  • Types that inherit from List, Set, or Map are ignored during the export (KT-80416).

  • Inheritors of List, Set, or Map cannot be instantiated on the Swift side (KT-80417).

  • When exported to Swift, Kotlin generic type parameters are type-erased to their upper bounds.

  • Swift closures can be passed into Kotlin, but Kotlin cannot export functional types to Swift.

  • Cross-language inheritance is not supported, so Swift classes cannot directly subclass from Kotlin-exported classes or interfaces.

  • No IDE migration tips or automation are available.

  • When using declarations that require opt-in, you must add an explicit optIn compiler option at the module level to your Gradle build file. For example, for the kotlinx.datetime library:

    swiftExport { moduleName = "Shared" export("org.jetbrains.kotlinx:kotlinx-datetime:0.7.1") { moduleName = "KotlinDateTime" flattenPackage = "kotlinx.datetime" } } // Add a separate opt-in block at the module level compilerOptions { optIn.add("kotlin.time.ExperimentalTime") }

Mappings

The table below shows how Kotlin concepts are mapped to Swift.

Kotlin

Swift

Notes

class

class

note

object

class with shared property

note

typealias

typealias

note

Function

Function

note

Property

Property

note

Constructor

Initializer

note

Package

Nested enum

note

Boolean

Bool

Char

Unicode.UTF16.CodeUnit

Byte

Int8

Short

Int16

Int

Int32

Long

Int64

UByte

UInt8

UShort

UInt16

UInt

UInt32

ULong

UInt64

Float

Float

Double

Double

Any

KotlinBase class

Unit

Void

Nothing

Never

note

Declarations

Classes

Swift export supports only final classes that directly inherit from Any, like class Foo(). They are translated to Swift classes that inherit from a special KotlinBase class:

// Kotlin class MyClass { val property: Int = 0 fun method() {} }
// Swift public class MyClass : KotlinRuntime.KotlinBase { public var property: Swift.Int32 { get { // ... } } public override init() { // ... } public func method() -> Swift.Void { // ... } }

Objects

Objects are translated to Swift classes with a private init and static shared accessor:

// Kotlin object O
// Swift public class O : KotlinRuntime.KotlinBase { public static var shared: O { get { // ... } } private override init() { // ... } }

Type aliases

Kotlin type aliases are exported as is:

// Kotlin typealias MyInt = Int
// Swift public typealias MyInt = Swift.Int32

Functions

Swift export supports simple top-level functions and methods:

// Kotlin fun foo(a: Short, b: Bar) {} fun baz(): Long = 0
// Swift public func foo(a: Swift.Int16, b: Bar) -> Swift.Void { // ... } public func baz() -> Swift.Int64 { // ... }

Extension functions are also supported. The receiver parameter of the extension function is moved into ordinary parameters in the first position:

// Kotlin fun Int.foo(): Unit = TODO()
// Swift func foo(_ receiver: Int32) {}

Functions with suspend, inline, and operator keywords are not supported.

Properties

Kotlin properties are translated to Swift properties:

// Kotlin val a: Int = 0 var b: Short = 15 const val c: Int = 0
// Swift public var a: Swift.Int32 { get { // ... } } public var b: Swift.Int16 { get { // ... } set { // ... } } public var c: Swift.Int32 { get { // ... } }

Constructors

Constructors are translated to Swift initializers:

// Kotlin class Foo(val prop: Int)
// Swift public class Foo : KotlinRuntime.KotlinBase { public init( prop: Swift.Int32 ) { // ... } }

Types

kotlin.Nothing

The Kotlin Nothing type is translated to the Never type:

// Kotlin fun foo(): Nothing = TODO() fun baz(input: Nothing) {}
// Swift public func foo() -> Swift.Never { // ... } public func baz(input: Swift.Never) -> Void { // ... }

Classifier types

Swift export currently supports only final classes that directly inherit from Any.

Packages

Kotlin packages are translated to nested Swift enums to avoid name collisions:

// Kotlin // bar.kt file in foo.bar package fun callMeMaybe() {}
// Kotlin // baz.kt file in foo.baz package fun callMeMaybe() {}
// Swift public extension foo.bar { public func callMeMaybe() {} } public extension foo.baz { public func callMeMaybe() {} } public enum foo { public enum bar {} public enum baz {} }

Evolution of Swift export

We're planning to expand and gradually stabilize Swift export in future Kotlin releases, improving interoperability between Kotlin and Swift, particularly around coroutines and flows.

You can leave your feedback:

10 September 2025