--- title: "Native C API" output: rmarkdown::html_vignette vignette: > %\VignetteIndexEntry{Native C API} %\VignetteEngine{knitr::rmarkdown} %\VignetteEncoding{UTF-8} --- ```{r, include = FALSE} knitr::opts_chunk$set(collapse = TRUE, comment = "#>") ``` Ropendal exposes a pure C API in `inst/include/ropendal.h` for downstream native packages. The purpose is direct OpenDAL-backed byte I/O from native code: submit async work, wait on Aio handles, and read borrowed results or fill caller-owned buffers without calling R while background I/O is running. The exported C symbols are implemented in Rust and documented by the installed header. ## Async-first surface C calls mirror the package’s async behavior: - `*_aio()` for submission, - `ropendal_aio_wait()` / monitor helpers for completion, - explicit status codes plus optional error objects for operation errors. ## Usage from another R package A downstream package uses the normal R package mechanism to get the header: ```text LinkingTo: Ropendal Imports: Ropendal ``` Native code can then include the installed header: ```c #include ``` If the downstream shared library calls Ropendal symbols directly, its build system must also arrange for Ropendal's native library to be loaded/linked before those calls. The C interface itself remains pure C and does not require R headers or `SEXP` values. ## Minimal usage pattern ```c // allocate a filesystem handle ropendal_fs_t *fs = NULL; ropendal_error_t *err = NULL; ropendal_kv_t cfg = { .struct_size = sizeof(cfg), .key = "root", .value = "/tmp/ropendal-example" }; ropendal_fs_open("fs", &cfg, 1, &fs, &err); // submit async read and wait ropendal_read_options_t opts = { .struct_size = sizeof(opts), .path = "object.bin" }; ropendal_aio_t *aio = NULL; ropendal_status_t status = ropendal_read_aio(fs, &opts, &aio, &err); if (status == ROPENDAL_OK) { ropendal_aio_wait(aio, -1, &err); const uint8_t *data = NULL; size_t len = 0; ropendal_aio_result_bytes(aio, &data, &len, &err); } // always release native resources ropendal_aio_release(aio); ropendal_fs_release(fs); ropendal_error_release(err); ``` ## Why this exists Downstream packages can perform byte-oriented async operations without going through R-level raw allocation. This keeps the native path fast and explicit while using the same transport semantics as the user-facing R API.