如何使用 Rcpp 创建依赖于共享库的 R 包?

How do I create an R package that depends on a shared library with Rcpp?

我有一个从 C 源代码生成的共享库 libbcd.so。 我还有一些接口代码Rinterface.h, Rinterface.cpp,如下所示:

Rinterface.h:

#pragma once

#include <R.h>
#include <Rcpp.h>
extern "C" {
#include "bcd.h"
}

void call_bcd();

Rinterface.cpp

#include "Rinterface.h"

// [[Rcpp::export]]
void call_bcd() {
    mm_problem* pr = mm_problem_alloc();
    bcd_vars* vars = bcd_vars_alloc();
    bcd(pr, vars);
}

其中 mm_problem_alloc(), bcd_vars_alloc(), bcd(mm_problem*, bcd_vars*)libbcd.so 中的所有函数,mm_problembcd_varslibbcd.so 中的用户定义类型。

我用 Rcpp.package.skeleton("bcd") 创建了一个包,将 Rinterface 文件放在 src 文件夹下,其中包含以下 Makevars

PKG_CPPFLAGS=-I/usr/local/lib/R/site-library/Rcpp/include/ -I/path/to/bcd/headers/
PKG_LIBS=-L/path/to/bcd/library/file -lbcd

如果我在编译步骤后尝试使用 install.packages("/path/to/bcd_1.0.tar.gz", type = "source") 从源安装包,当它尝试加载 libbcd.so:

时出现错误
** R
** byte-compile and prepare package for lazy loading
** help
*** installing help indices
** building package indices
** testing if installed package can be loaded from temporary location
Error: package or namespace load failed for ‘bcd’ in dyn.load(file, DLLpath = DLLpath, ...):
 unable to load shared object '/home/gomfy/R/x86_64-pc-linux-gnu-library/3.6/00LOCK-bcd/00new/bcd/libs/bcd.so':
  libbcd.so: cannot open shared object file: No such file or directory

基于 在我看来,解决方法是将 libbcd.so.c 源文件重命名为 .cpp,然后将它们放在 src文件夹。这是正确的吗?

一般来说,利用 Rcpp 能够从 R 调用第三方共享库的推荐方法是什么?我发现了一个相关的问题 但它没有涉及构建包的细节。我看过 Writing R extensions vignette("Rcpp-introduction"), vignette("Rcpp-package") 但我无法得出明确的结论。所以,我很感激专家的帮助。谢谢!

简而言之,你不能。

系统级共享库将由ld.so按需加载(在Linux上,在其他操作系统上类似,想法相同) .因此,如果您的包依赖于非标准或不常见的共享库,那就倒霉了。 这只有在 ld.so 知道的情况下才有效,而 R 包无法做到这一点。

可以做的是将您的libbcd的源代码与您的包捆绑在一起,并将它们构建为静态库 链接到您的包。那可行。更简单的方法可能只是将 libbcd 的源文件放入包 src/ 目录中——R 将处理其余部分。