Rust Vec для Ruby Array с FFI Segfaults
Я пытаюсь вернуть структуру, которая может быть преобразована в массив Ruby из внешней функции ржавчины, но когда я пытаюсь вызвать метод структурирования #to_a, я получаю segfault.
use libc::size_t;
#[repr(C)]
pub struct Array {
len: libc::size_t,
data: *const libc::c_void,
}
impl Array {
fn from_vec<T>(mut vec: Vec<T>) -> Array {
vec.shrink_to_fit();
let array = Array { data: vec.as_ptr() as *const libc::c_void, len: vec.len() as libc::size_t };
mem::forget(vec);
array
}
}
#[no_mangle]
pub extern fn get_links(url: *const libc::c_char) -> Array {
// Get links
let mut urls: Vec<String> = vec![];
// push strings into urls vec
// urls => collections::vec::Vec<collections::string::String>
Array::from_vec(urls)
}
require 'ffi'
module Rust
extend FFI::Library
ffi_lib './bin/libembed.dylib'
class NodesArray < FFI::Struct
layout :len, :size_t, # dynamic array layout
:data, :pointer #
def to_a
self[:data].get_array_of_string(0, self[:len]).compact
end
end
attach_function :get_links, [:string], NodesArray.by_value
end
Когда я пытаюсь использовать эту функцию в ruby, она возвращает Fii::NodesArray. Я также могу получить len и данные из структуры. Только когда я вызываю #to_a, segfaults.
2 ответа
Решение
Проблема, на которую указал Adrian, заключалась в том, что я проталкивал струны в Vec. FFI нуждается *const libc::c_char
, который может быть преобразован из String
,
let mut urls: Vec<*const libc::c_char> = vec![];
urls.push(CString::new(string_var.value.to_string()).unwrap().into_raw());
Это похоже на FFI::Pointer#get_array_of_string
прослушивается (или он просто не делает то, что я думаю, что делает). Ваш код работает для меня, если я изменю эту строку:
self[:data].get_array_of_string(0, self[:len]).compact
к этому:
Array.new(self[:len]) {|i| self[:data].read_pointer[i * FFI::Pointer::SIZE].read_string }