Ошибка связывания пакета R с Rcpp: «неопределенный символ: LAPACKE_dgels» - PullRequest
3 голосов
/ 15 марта 2019

Я создаю пакет R 'lapacker', чтобы обеспечить интерфейс C для внутренней библиотеки LAPACK, предоставляемой и используемой R (с двойной точностью и только двойной сложностью), используя заголовочный файл R API "R_ext / Lapack.h".Исходный код: https://github.com/ypan1988/lapacker/

И структура проекта:

/lapacker
  /inst
    /include
      /lapacke.h
      /someother header files
  /R
    /zzz.R
  /src
    /lapacke_dgetrf.c
    /lapacke_dgetrf_work.c
    /loads of other utility functions provided by LAPACKE 
    /rcpp_hello.cpp
  DESCRIPTION
  NAMESPACE

Внутри проекта я пробовал тестовую функцию в файле rcpp_hello.cpp (обратите внимание, этот пример взят из https://www.netlib.org/lapack/lapacke.html#_calling_code_dgels_code):

//'@export
// [[Rcpp::export]]
void example_lapacke_dgels()
{
  double a[5][3] = {{1,1,1},{2,3,4},{3,5,2},{4,2,5},{5,4,3}};
  double b[5][2] = {{-10,-3},{12,14},{14,12},{16,16},{18,16}};
  lapack_int info,m,n,lda,ldb,nrhs;
  int i,j;

  m = 5;
  n = 3;
  nrhs = 2;
  lda = 3;
  ldb = 2;

  info = LAPACKE_dgels(LAPACK_ROW_MAJOR,'N',m,n,nrhs,*a,lda,*b,ldb);

  for(i=0;i<n;i++)
  {
    for(j=0;j<nrhs;j++)
    {
      printf("%lf ",b[i][j]);
    }
    printf("\n");
  }
}

Весь пакет может быть правильно скомпилирован без ошибок, и в R он дает правильный ответ (с указанием символа LAPACKE_dgels можно найти):

> example_lapacke_dgels()
2.000000 1.000000 
1.000000 1.000000 
1.000000 2.000000 

Однако, когда ясоздайте отдельный файл C ++, скажем, demo3.cpp с точно такой же функцией,

#include <Rcpp.h>

#include <lapacke.h>
// [[Rcpp::depends(lapacker)]]

// [[Rcpp::export]]
void lapacke_dgels_test()
{
  double a[5][3] = {{1,1,1},{2,3,4},{3,5,2},{4,2,5},{5,4,3}};
  double b[5][2] = {{-10,-3},{12,14},{14,12},{16,16},{18,16}};
  lapack_int info,m,n,lda,ldb,nrhs;
  int i,j;

  m = 5;
  n = 3;
  nrhs = 2;
  lda = 3;
  ldb = 2;

  info = LAPACKE_dgels(LAPACK_ROW_MAJOR,'N',m,n,nrhs,*a,lda,*b,ldb);

  for(i=0;i<n;i++)
  {
    for(j=0;j<nrhs;j++)
    {
      printf("%lf ",b[i][j]);
    }
    printf("\n");
  }
}

, он больше не компилируется должным образом (на самом деле я пытался и в MacOS, и в Ubuntu, та же проблема с линковкой) и выдает сообщения об ошибках компоновки(Не удается найти символ LAPACKE_dgels):

> Rcpp::sourceCpp("~/Desktop/demo3.cpp", showOutput = TRUE)
/usr/lib/R/bin/R CMD SHLIB -o 'sourceCpp_6.so'  'demo3.cpp'  
g++  -I/usr/share/R/include -DNDEBUG   -I"/home/yipan/R/x86_64-pc-linux-gnu-library/3.4/Rcpp/include" -I"/home/yipan/R/x86_64-pc-linux-gnu-library/3.4/lapacker/include" -I"/home/yipan/Desktop"    -fpic  -g -O2 -fdebug-prefix-map=/build/r-base-AitvI6/r-base-3.4.4=. -fstack-protector-strong -Wformat -Werror=format-security -Wdate-time -D_FORTIFY_SOURCE=2 -g  -c demo3.cpp -o demo3.o
g++ -shared -L/usr/lib/R/lib -Wl,-Bsymbolic-functions -Wl,-z,relro -o sourceCpp_6.so demo3.o -L/usr/lib/R/lib -lR
Error in dyn.load("/tmp/RtmpUsASwK/sourceCpp-x86_64-pc-linux-gnu-1.0.0/sourcecpp_159e6145591d/sourceCpp_6.so") : 
  unable to load shared object '/tmp/RtmpUsASwK/sourceCpp-x86_64-pc-linux-gnu-1.0.0/sourcecpp_159e6145591d/sourceCpp_6.so':
  /tmp/RtmpUsASwK/sourceCpp-x86_64-pc-linux-gnu-1.0.0/sourcecpp_159e6145591d/sourceCpp_6.so: undefined symbol: LAPACKE_dgels

Я также проверил файл lapacker.so в /R/x86_64-pc-linux-gnu-library/3.4/lapacker/libs и нашел:

000000000000c6b0 g    DF .text  00000000000001bf  Base        LAPACKE_dgels

Я что-то пропустил, чтобы правильно скомпилировать demo3.cpp? Большое спасибо за ваше терпение и время!

1 Ответ

4 голосов
/ 15 марта 2019

Вы столкнулись с трудной проблемой здесь. Символ, который вы пытаетесь разрешить LAPACKE_dgels, является частью lapacker.so, созданного во время установки пакета. Проблема в том, что библиотеки для пакетов R не предназначены для компоновки. Вместо этого они загружаются R динамически во время выполнения. В принципе, я вижу четыре варианта:

  1. Преобразуйте lapacke в библиотеку только для заголовков и установите ее в inst/include (c.f. RcppArmadillo).
  2. Связь с системой установки lapacke (просто в Linux ...)
  3. Зарегистрируйте все функции с помощью R и используйте методы, предоставленные R, для связи с ними (см. WRE и nloptr).
  4. Скомпилируйте библиотеку, предназначенную для компоновки, и установите ее с помощью пакета R. Для этого вам все еще понадобится плагин, так как вам нужно добавить -L<path/to/lib> -l<libname> .... к PKG_LIBS.

Я уверен, что есть примеры на CRAN, которые используют метод 4, но ни один не приходит на ум прямо сейчас. Однако, как «кодовое ката», я преобразовал свой последний тестовый пакет для использования этой структуры, c.f. https://github.com/rstub/levmaR/tree/static.


(Оригинальный неполный ответ.)

В src/Makevars у вас есть

PKG_LIBS = $(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS)

Вам понадобится аналоговая настройка при компиляции файла cpp через атрибуты Rcpp. Лучший способ добиться этого - использовать плагин Rcpp, c.f. Решение RcppArmadillo (корректировки не проверены!):

inlineCxxPlugin <- function(...) {
    plugin <-
        Rcpp::Rcpp.plugin.maker(
                  include.before = "#include <lapacke.h>",
                  libs           = "$(LAPACK_LIBS) $(BLAS_LIBS) $(FLIBS)",
                  package        = "lapacker"
              )
    settings <- plugin()
    settings$env$PKG_CPPFLAGS <- "-I../inst/include"
    settings
}

Кстати, почему вы хотите напрямую взаимодействовать с LAPACK, когда RcppArmadillo уже делает это?

...