Metadata-Version: 2.2
Name: apache-tvm_ffi_orcjit
Version: 0.1.0
Summary: Load TVM-FFI exported object files using LLVM ORC JIT v2
Keywords: tvm-ffi,llvm,jit,orcjit
Author: TVM-FFI OrcJIT Contributors
License: Apache-2.0
Classifier: Development Status :: 3 - Alpha
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: C++
Project-URL: Homepage, https://github.com/apache/tvm-ffi
Project-URL: Repository, https://github.com/apache/tvm-ffi
Requires-Python: >=3.10
Requires-Dist: apache-tvm-ffi>=0.1.0
Description-Content-Type: text/markdown

<!--- Licensed to the Apache Software Foundation (ASF) under one -->
<!--- or more contributor license agreements.  See the NOTICE file -->
<!--- distributed with this work for additional information -->
<!--- regarding copyright ownership.  The ASF licenses this file -->
<!--- to you under the Apache License, Version 2.0 (the -->
<!--- "License"); you may not use this file except in compliance -->
<!--- with the License.  You may obtain a copy of the License at -->

<!---   http://www.apache.org/licenses/LICENSE-2.0 -->

<!--- Unless required by applicable law or agreed to in writing, -->
<!--- software distributed under the License is distributed on an -->
<!--- "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY -->
<!--- KIND, either express or implied.  See the License for the -->
<!--- specific language governing permissions and limitations -->
<!--- under the License. -->

# TVM-FFI OrcJIT

A Python package that enables dynamic loading of compiled object files (`.o`)
using LLVM ORC JIT v2, providing a flexible JIT execution environment for
TVM-FFI exported functions.

## Features

- **JIT Execution**: Load and execute compiled object files at runtime using LLVM's ORC JIT v2
- **Multiple Libraries**: Create separate dynamic libraries with independent symbol namespaces
- **Incremental Loading**: Add multiple object files to the same library incrementally
- **Symbol Isolation**: Different libraries can define the same symbol without conflicts
- **Init/Fini Support**: Handles static constructors/destructors across ELF (`.init_array`/`.ctors`), Mach-O (`__mod_init_func`), and COFF (`.CRT$XC*`/`.CRT$XT*`)
- **Cross-Platform**: Linux (x86_64, aarch64), macOS (arm64), Windows (AMD64)
- **Multi-Compiler**: Tested with LLVM Clang, GCC, Apple Clang, MSVC, and clang-cl
- **TVM-FFI Integration**: Seamlessly works with TVM-FFI's stable C ABI
- **Python API**: Simple Pythonic interface for JIT compilation and execution

## Supported Platforms and Compilers

Object files compiled with any of the following compiler/platform combinations
can be loaded and executed by the ORC JIT:

| Platform | Compilers | C | C++ |
| -------- | --------- | :-: | :-: |
| Linux (x86_64, aarch64) | LLVM Clang, GCC | yes | yes |
| macOS (arm64) | LLVM Clang, Apple Clang | yes | yes |
| Windows (AMD64) | LLVM Clang, MSVC, clang-cl | yes | no |

Windows is C-only across all compilers. C++ objects compiled with
`TVM_FFI_DLL_EXPORT_TYPED_FUNC` use `try`/`catch` (via `TVM_FFI_SAFE_CALL_BEGIN/END`),
which requires Itanium exception ABI symbols (`__cxa_begin_catch`,
`__gxx_personality_v0`, etc.) that the MSVC-built host process cannot provide.
Pure C objects using the `TVMFFISafeCallType` ABI work on all platforms.

## Installation

### Install from PyPI

```bash
pip install apache-tvm-ffi apache-tvm_ffi_orcjit
```

### Build from Source

#### Prerequisites

- Python 3.10+, CMake 3.20+, C++17 compiler
- LLVM 22+ development libraries (`llvmdev`, `llvm-config`)
- Static `zlib` and `zstd` libraries (in the same prefix as LLVM)

#### Install LLVM via conda-forge

The easiest way to get all dependencies is via conda-forge:

```bash
conda create -p /opt/llvm -c conda-forge \
  llvmdev=22.1.0 clangdev=22.1.0 compiler-rt=22.1.0 zlib zstd-static -y
export LLVM_PREFIX=/opt/llvm
```

On Windows:

```cmd
conda create -p C:\opt\llvm -c conda-forge llvmdev=22.1.0 zlib zstd-static -y
set LLVM_PREFIX=C:\opt\llvm
```

#### Build and install

```bash
git clone --recursive https://github.com/apache/tvm-ffi.git
cd tvm-ffi

# Install tvm-ffi first
pip install -e .

# Build and install the orcjit addon
cd addons/tvm_ffi_orcjit
pip install -e .
```

The `LLVM_PREFIX` environment variable tells CMake where to find LLVM. If
LLVM is installed in a conda env or a standard system path, CMake can
auto-discover it and `LLVM_PREFIX` is not needed.

## Usage

### Basic Example

```python
from tvm_ffi_orcjit import ExecutionSession

# Create an execution session
session = ExecutionSession()

# Create a dynamic library
lib = session.create_library()

# Load an object file
lib.add("example.o")

# Get and call a function
add_func = lib.get_function("add")
result = add_func(1, 2)
print(f"Result: {result}")  # Output: Result: 3
```

### Multiple Libraries with Symbol Isolation

```python
session = ExecutionSession()

lib1 = session.create_library("lib1")
lib2 = session.create_library("lib2")

lib1.add("implementation_v1.o")
lib2.add("implementation_v2.o")

add_v1 = lib1.get_function("add")
add_v2 = lib2.get_function("add")

print(add_v1(5, 3))  # Uses implementation from lib1
print(add_v2(5, 3))  # Uses implementation from lib2
```

### Cross-Library Linking

```python
session = ExecutionSession()

base_lib = session.create_library("base")
base_lib.add("math_ops.o")

caller_lib = session.create_library("caller")
caller_lib.set_link_order(base_lib)  # Can resolve symbols from base_lib
caller_lib.add("caller.o")

result = caller_lib.get_function("call_math")(10, 20)
```

## Writing Functions for OrcJIT

### C++ (Linux/macOS)

```cpp
#include <tvm/ffi/function.h>

TVM_FFI_DLL_EXPORT_TYPED_FUNC(add, [](int a, int b) {
    return a + b;
});
```

Compile: `clang++ -std=c++17 -fPIC -O2 -c -o example.o example.cc`

### Pure C (all platforms including Windows)

```c
#include <tvm/ffi/c_api.h>

TVM_FFI_DLL_EXPORT int __tvm_ffi_add(
    void* self, const TVMFFIAny* args, int32_t num_args, TVMFFIAny* result) {
  result->type_index = kTVMFFIInt;
  result->v_int64 = args[0].v_int64 + args[1].v_int64;
  return 0;
}
```

Compile: `clang -O2 -c -o example.o example.c`

## How It Works

- **LLJIT**: Built on LLVM's ORC JIT v2 with `ObjectLinkingLayer` (JITLink) for
  all platforms.
- **InitFiniPlugin**: Custom `ObjectLinkingLayer::Plugin` that collects function
  pointers from init/fini sections (ELF `.init_array`/`.ctors`/`.fini_array`/`.dtors`,
  Mach-O `__mod_init_func`/`__mod_term_func`, COFF `.CRT$XC*`/`.CRT$XT*`) and
  runs them in priority order at symbol lookup / library teardown.
- **DLL Import Stubs** (Windows): Custom `DefinitionGenerator` that resolves host
  process symbols from all loaded DLLs and creates `__imp_*` pointer stubs in
  JIT memory, keeping all fixups within PCRel32 range.
- **SEH Stripping** (Windows): `ObjectTransformLayer` strips `.pdata`/`.xdata`
  relocations from COFF objects before JITLink graph building, working around a
  JITLink limitation with COMDAT section symbols.

Please refers to [ORCJIT_PRIMER.md](./ORCJIT_PRIMER.md) to learn more about object file, linking, llvm orcjit v2, and how the addon works.

## Project Structure

```text
tvm_ffi_orcjit/
├── CMakeLists.txt              # Build configuration
├── pyproject.toml              # Python package metadata
├── src/ffi/
│   ├── orcjit_session.cc       # ExecutionSession (LLJIT setup, plugins)
│   ├── orcjit_session.h
│   ├── orcjit_dylib.cc         # DynamicLibrary (object loading, symbol lookup)
│   ├── orcjit_dylib.h
│   └── orcjit_utils.h          # LLVM error handling utilities
├── python/tvm_ffi_orcjit/
│   ├── __init__.py             # Module exports and library loading
│   ├── session.py              # Python ExecutionSession wrapper
│   └── dylib.py                # Python DynamicLibrary wrapper
├── tests/                      # See tests/README.md
└── examples/quick-start/       # Complete example with CMake
```

## CI

Runs on Linux (x86_64, aarch64), macOS (arm64), Windows (AMD64) via
`cibuildwheel`. Each platform builds test objects with multiple compilers
and runs the full test suite. See `.github/workflows/tvm_ffi_orcjit.yml`.

## Troubleshooting

### "Cannot find global function" error

The shared library wasn't loaded. Reinstall: `pip install --force-reinstall apache-tvm_ffi_orcjit`

### "Duplicate definition of symbol" error

Use separate libraries for different implementations of the same symbol.

### "Symbol not found" error

Ensure functions are exported with TVM-FFI macros (`TVM_FFI_DLL_EXPORT_TYPED_FUNC`
for C++, or `__tvm_ffi_` prefix for C).

### Relocation errors on Windows

MSVC/clang-cl objects must be compiled with `/GS-` to disable buffer security
checks (`__security_cookie`) which are CRT symbols the JIT cannot resolve.

### LLVM version mismatch

The package requires LLVM 22+. Set `LLVM_PREFIX` to the LLVM install prefix:

```bash
export LLVM_PREFIX=/path/to/llvm
```

## License

Apache License 2.0
