Взаимодействие C ++ с Rust — возвращение паники CString

Я пытаюсь вызвать некоторые функции, написанные на Rust из C ++. До сих пор я был довольно успешным, но у меня все еще есть одна маленькая проблема с CStringсвязанная с паникой во время выполнения.

Функция hello предполагается взять входную строку, объединить ее с другой строкой и вернуть продукт.

Вот мой fun.rs:

use std::ffi::CString;

#[no_mangle]
pub extern "C" fn add(a: i32, b: i32) -> i32 {
a + b
}

#[no_mangle]
pub extern "C" fn hello(cs: CString) -> CString {
let slice = cs.to_str().unwrap();
let mut s = "Hello, ".to_string();
s = s + slice;

CString::new(&s[..]).unwrap() // runtime error
// CString::new(cs).unwrap()  // empty string if no borrow
// cs                         // works if no borrow, but this is not what I meant
}

Вот main.cpp:

#include <iostream>
using namespace std;

extern "C" {
int add(int a, int b);
const char* hello(const char*x);
}

int main()
{
int a, b;
cin >> a >> b;
cout << add(a,b) << ";" << hello("Pawel") << std::endl;
return 0;
}

А также makefile:

rust:
rustc --crate-type=staticlib -C panic=abort fun.rs

cpp:
g++ -c main.cpp

link:
g++ main.o -L . libfun.a -o main -lpthread -ldl -lgcc_s -lc -lm -lrt -lutil

Команды для запуска исполняемого файла:

$ make rust
$ make cpp
$ make link
$ ./main
1 2

Выполняемый вывод:

1 2
thread '<unnamed>' panicked at 'index 18446744073709551615 out of range for slice of length 0', ../src/libcore/slice.rs:549
note: Run with `RUST_BACKTRACE=1` for a backtrace..

Backtrace:

stack backtrace:
1:           0x435d4f - std::sys::backtrace::tracing::imp::write::h46e546df6e4e4fe6
2:           0x44405b - std::panicking::default_hook::_$u7b$$u7b$closure$u7d$$u7d$::h077deeda8b799591
3:           0x443c8f - std::panicking::default_hook::heb8b6fd640571a4f
4:           0x4099fe - std::panicking::rust_panic_with_hook::hd7b83626099d3416
5:           0x4442a1 - std::panicking::begin_panic::h941ea76fc945d925
6:           0x40b74a - std::panicking::begin_panic_fmt::h30280d4dd3f149f5
7:           0x44423e - rust_begin_unwind
8:           0x451d8f - core::panicking::panic_fmt::h2d3cc8234dde51b4
9:           0x452073 - core::slice::slice_index_len_fail::ha4faf37254d75f20
10:           0x40e903 - std::ffi::c_str::CStr::to_str::ha9642252376bab15
11:           0x4048e0 - hello
12:           0x40476f - main
13:     0x7f78ff688f44 - __libc_start_main
14:           0x404678 - <unknown>
15:                0x0 - <unknown>

Есть идеи, почему Руст паникует?

2

Решение

Руста CString не совместим с С const char *, Вот определение CString из стандартной библиотеки:

pub struct CString {
inner: Box<[u8]>,
}

это Box<[u8]> тип является толстый указатель, т.е. кортеж, который содержит указатель на элементы среза и длину среза в виде usize,

Вместо этого вы должны сделать так, чтобы ваша функция Rust *const c_char аргумент, а затем позвонить CStr::from_ptr с этим указателем в качестве аргумента для получения CStr значение.

Что касается возвращаемого значения, есть небольшая проблема: ваша функция выделяет новую строку, а затем возвращает указатель на нее. Опять же, вы должны вернуть *const c_char, что вы можете сделать, позвонив CString::into_raw на вашем соединенном CString значение. Но чтобы избежать утечек памяти, вы также должны предоставить функцию Rust, которая будет возвращать указатель, возвращаемый hello и позвонить CString::from_raw на нем, который воссоздаст CString, CStringЗатем будет запущен деструктор, освобождающий память.

5

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]