Skip to content

Support for Exporting Template and Namespaced Functions in Rcpp #1368

Open
@ManosPapadakis95

Description

@ManosPapadakis95

Enhancement: Support for Exporting Template and Namespaced Functions in Rcpp

Rcpp has been an incredible tool for bridging R and C++, and its flexibility has inspired me to explore new ways to extend its functionality. Building on the existing framework, I have added support for exporting template functions and functions inside namespaces, expanding the possibilities for advanced C++ integration.

Key Features

1. Support for Exporting Template Functions

  • The Rcpp::export attribute can now be applied to explicit template instantiations.
  • The function must have an explicit instantiation in the .cpp file for it to be correctly recognized.
  • Works seamlessly with Rcpp::interfaces(cpp).

2. Support for Exporting Functions Inside Namespaces

  • Functions within namespaces can now be exported, as long as a forward declaration is provided with cpp_name in Rcpp::export.
  • This ensures proper symbol resolution while allowing structured code organization.

3. Handling of Declaration Specifiers

  • The system now accounts for one and only one declaration specifier:
    • template → Used for explicit template instantiations.
    • inline → Supports functions defined inside headers.
  • Combinations like "static inline" are not supported (e.g., static is explicitly disallowed).
  • An inline function can still be forward-declared without the inline keyword, and it will work seamlessly.

4. New includes Attribute for Header File Support

  • Since Rcpp does not scan for header files, a new Rcpp::includes attribute has been introduced.
  • This allows specifying one or more header file paths, ensuring necessary definitions are available in the generated .cpp file.
  • This is especially important for template functions, as the compiler needs access to their full definitions in order to instantiate them correctly.

Example Usage

// [[Rcpp::includes("helpers.h")]]

// [[Rcpp::export(name = "A1", cpp_name = "A1")]]
template NumericVector A<NumericVector>(size_t n);

// [[Rcpp::export(name = "A2", cpp_name = "A2")]]
template NumericVector A<NumericVector>(NumericVector x);

// [[Rcpp::export(name = "A3", cpp_name = "A3")]]
NumericVector Test::G();

Generated Rcpp Export File

// A<NumericVector>
template NumericVector A<NumericVector>(size_t n);
RcppExport SEXP _testRcpp_A1(SEXP nSEXP) {
BEGIN_RCPP
    Rcpp::RObject rcpp_result_gen;
    Rcpp::RNGScope rcpp_rngScope_gen;
    Rcpp::traits::input_parameter< size_t >::type n(nSEXP);
    rcpp_result_gen = Rcpp::wrap(A<NumericVector>(n));
    return rcpp_result_gen;
END_RCPP
}

// A<NumericVector>
template NumericVector A<NumericVector>(NumericVector x);
RcppExport SEXP _testRcpp_A2(SEXP xSEXP) {
BEGIN_RCPP
    Rcpp::RObject rcpp_result_gen;
    Rcpp::RNGScope rcpp_rngScope_gen;
    Rcpp::traits::input_parameter< NumericVector >::type x(xSEXP);
    rcpp_result_gen = Rcpp::wrap(A<NumericVector>(x));
    return rcpp_result_gen;
END_RCPP
}

// Test::G
NumericVector Test::G();
RcppExport SEXP _testRcpp_A3() {
BEGIN_RCPP
    Rcpp::RObject rcpp_result_gen;
    Rcpp::RNGScope rcpp_rngScope_gen;
    rcpp_result_gen = Rcpp::wrap(Test::G());
    return rcpp_result_gen;
END_RCPP
}

static const R_CallMethodDef CallEntries[] = {
    {"_testRcpp_A1", (DL_FUNC) &_testRcpp_A1, 1},
    {"_testRcpp_A2", (DL_FUNC) &_testRcpp_A2, 1},
    {"_testRcpp_A3", (DL_FUNC) &_testRcpp_A3, 0},
    {NULL, NULL, 0}
};

Testing & Future Considerations

I have run extensive tests to ensure this new feature works correctly. The ability to export template and namespaced functions integrates well with the existing Rcpp framework. However, given the complexity of C++ templates and namespaces, there might still be edge cases that I haven’t encountered.

Would this be a feature that the maintainers would be interested in? If so, I would be happy to prepare a pull request.

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions