1

我正在尝试用 Node FFI 包装一个 Rust 库(它公开一个 C API)。我有以下代码包装了两个函数。一个是返回指针的“构造函数”。另一个接受一个指针并返回 C 字符串。

var libcomm = ffi.Library('lib/c_api/target/debug/libcomm', {
  'comm_address_for_content': ['pointer', ['string']],
  'comm_address_to_str': ['string', ['pointer']]
});

当我使用响应的异步调用comm_address_to_str是正确的。但是,当我尝试使用同步样式调用该函数时,它会返回垃圾,或者很少会返回正确的结果。以下 nodeunit 测试练习该场景:

const comm = require("../").libcomm;

exports.testAddressForContent = function (test) {
  const ptr = comm.comm_address_for_content('test');

  const result = comm.comm_address_to_str(ptr);
  test.equal(result, 'a94a8fe5ccb19ba61c4c0873d391e987982fbbd3'); // always fails
  console.log('sync', result); // random garbage

  comm.comm_address_to_str.async(ptr, function(err, result) {
    test.equal(result, 'a94a8fe5ccb19ba61c4c0873d391e987982fbbd3'); // always passes
    console.log('async', result); // 'a94a8fe5ccb19ba61c4c0873d391e987982fbbd3'
    test.done();
  });
}

我不知道是什么原因造成的,但我需要能够使用同步调用样式。我包装的 Rust 库的 C API 在这里

4

1 回答 1

3

恐怕问题来自comm库,特别是comm_address_to_str函数:

#[no_mangle]
pub extern "C" fn comm_address_to_str(address: *const Address) -> *const c_char {
    let string = unsafe { *address }.to_str();
    CString::new(string).unwrap().as_ptr()
}

文档CString::as_ptr特别提到了这种模式:

use std::ffi::{CString};

let ptr = CString::new("Hello").unwrap().as_ptr();
unsafe {
    // `ptr` is dangling
    *ptr;
}

CString最后一行创建的指针在将指针放入其内部后立即被破坏,从而导致悬空指针。

您正在使用的comm库需要修复,并且可能也值得审核其他功能。

这里的解决方法是泄漏CString,然后提供另一个函数来调用泄漏的指针,该指针将负责释放它。

于 2016-10-10T09:14:56.697 回答