فراخوانی Rust

صدور یا Exporting توابع و تایپ‌ها از Rust به C آسان است:

interoperability/rust/libanalyze/analyze.rs

//! Rust FFI demo.
#![deny(improper_ctypes_definitions)]

use std::os::raw::c_int;

/// Analyze the numbers.
#[no_mangle]
pub extern "C" fn analyze_numbers(x: c_int, y: c_int) {
    if x < y {
        println!("x ({x}) is smallest!");
    } else {
        println!(" مقدارy  ({y}) احتمالا بزرگتر از x ({x}) است");
    }
}

interoperability/rust/libanalyze/analyze.h

#ifndef ANALYSE_H
#define ANALYSE_H

extern "C" {
void analyze_numbers(int x, int y);
}

#endif

interoperability/rust/libanalyze/Android.bp

rust_ffi {
    name: "libanalyze_ffi",
    crate_name: "analyze_ffi",
    srcs: ["analyze.rs"],
    include_dirs: ["."],
}

اکنون می توانیم این را از یک باینری C فراخوانی کنیم:

interoperability/rust/analyze/Android.bp

#include "analyze.h"

int main() {
  analyze_numbers(10, 20);
  analyze_numbers(123, 123);
  return 0;
}

interoperability/rust/analyze/Android.bp

cc_binary {
    name: "analyze_numbers",
    srcs: ["main.c"],
    static_libs: ["libanalyze_ffi"],
}

ساخت، push و اجرای باینری‌ها روی یک ماشین:

m analyze_numbers
adb push "$ANDROID_PRODUCT_OUT/system/bin/analyze_numbers" /data/local/tmp
adb shell /data/local/tmp/analyze_numbers

#[no_mangle] نام‌گذاری معمول Rust را غیرفعال می‌کند، بنابراین نماد صادر شده فقط نام تابع خواهد بود. همچنین می‌توانید از #[export_name = "some_name"] برای تعیین هر نامی استفاده کنید.