Невозможно связать с библиотеками повышения

Я не могу заставить свой g ++ скомпилировать и связать программу минимального ускорения (а также другие компоненты) на моей машине.

// example.cpp

#include <boost/coroutine/all.hpp>
#include <iostream>
using namespace boost::coroutines;

void mycorofunc(coroutine<void>::push_type &sink){
std::cout << "1";
sink();
std::cout << " 3";
}

int main() {
coroutine<void>::pull_type source{mycorofunc};
std::cout << " 2";
source();
std::cout << " 4!" << std::endl;
}

Я использовал apt-get для установки boost (libboost-all-dev), но ни

g++ example.cpp -lboost_coroutine -lboost_system

ни

g++ example.cpp -I/usr/include -L/usr/lib/x86_64-linux-gnu -lboost_coroutine -lboost_system

работает. Вывод g ++ выглядит следующим образом:

In file included from /usr/include/boost/coroutine/v1/coroutine.hpp:19:0,
from /usr/include/boost/coroutine/coroutine.hpp:13,
from /usr/include/boost/coroutine/all.hpp:11,
from example.cpp:3:
/usr/include/boost/type_traits/function_traits.hpp: In instantiation of ‘struct boost::function_traits<void>’:
example.cpp:7:31:   required from here
/usr/include/boost/type_traits/function_traits.hpp:168:8: error: invalid use of incomplete type ‘struct boost::detail::function_traits_helper<void*>’
struct function_traits :
^
/usr/include/boost/type_traits/function_traits.hpp:21:36: error: declaration of ‘struct boost::detail::function_traits_helper<void*>’
template<typename Function> struct function_traits_helper;
^
example.cpp:7:31: error: ‘arity’ is not a member of ‘boost::function_traits<void>’
void mycorofunc(coroutine<void>::push_type &sink){
^
example.cpp:7:31: error: template argument 2 is invalid
example.cpp:7:44: error: expected ‘,’ or ‘...’ before ‘&’ token
void mycorofunc(coroutine<void>::push_type &sink){
^
example.cpp: In function ‘void mycorofunc(int)’:
example.cpp:9:8: error: ‘sink’ was not declared in this scope
sink();
^
example.cpp: In function ‘int main()’:
example.cpp:14:17: error: ‘arity’ is not a member of ‘boost::function_traits<void>’
coroutine<void>::pull_type source{mycorofunc};
^
example.cpp:14:17: error: template argument 2 is invalid
example.cpp:14:30: error: expected initializer before ‘source’
coroutine<void>::pull_type source{mycorofunc};
^
example.cpp:16:10: error: ‘source’ was not declared in this scope
source();

Еще немного информации о моей среде:

$$ locate coroutine/all.hpp
/usr/include/boost/coroutine/all.hpp
/usr/local/boost_1_58_0/boost/coroutine/all.hpp

$$ locate boost_system
/usr/lib/x86_64-linux-gnu/libboost_system.a
/usr/lib/x86_64-linux-gnu/libboost_system.so
/usr/lib/x86_64-linux-gnu/libboost_system.so.1.54.0

$$ locate boost_coroutine
/usr/lib/x86_64-linux-gnu/libboost_coroutine.a

$$ g++ -E -x c++ - -v < /dev/null  # printing g++ include path
Using built-in specs.
COLLECT_GCC=g++
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.8.2-19ubuntu1' --with-bugurl=file:///usr/share/doc/gcc-4.8/README.Bugs --enable-languages=c,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.8 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.8 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --disable-libmudflap --enable-plugin --with-system-zlib --disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64/jre --enable-java-home --with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64 --with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-4.8-amd64 --with-arch-directory=amd64 --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --enable-objc-gc --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.8.2 (Ubuntu 4.8.2-19ubuntu1)
COLLECT_GCC_OPTIONS='-E' '-v' '-shared-libgcc' '-mtune=generic' '-march=x86-64'
/usr/lib/gcc/x86_64-linux-gnu/4.8/cc1plus -E -quiet -v -imultiarch x86_64-linux-gnu -D_GNU_SOURCE - -mtune=generic -march=x86-64 -fstack-protector -Wformat -Wformat-security
ignoring duplicate directory "/usr/include/x86_64-linux-gnu/c++/4.8"ignoring nonexistent directory "/usr/local/include/x86_64-linux-gnu"ignoring nonexistent directory "/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../../x86_64-linux-gnu/include"#include "..." search starts here:
#include <...> search starts here:
/usr/include/c++/4.8
/usr/include/x86_64-linux-gnu/c++/4.8
/usr/include/c++/4.8/backward
/usr/lib/gcc/x86_64-linux-gnu/4.8/include
/usr/local/include
/usr/lib/gcc/x86_64-linux-gnu/4.8/include-fixed
/usr/include/x86_64-linux-gnu
/usr/include
End of search list.
# 1 "<stdin>"# 1 "<command-line>"# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "<stdin>"COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../../lib/:/lib/x86_64-linux-gnu/:/lib/../lib/:/usr/lib/x86_64-linux-gnu/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../:/lib/:/usr/lib/
COLLECT_GCC_OPTIONS='-E' '-v' '-shared-libgcc' '-mtune=generic' '-march=x86-64'

Компилятор: пробовал как g ++ (v. 4.8.2), так и g ++ — 5

Distro: Ubuntu 14.04.1 LTS

3

Решение

После нескольких дней отладки это решение я нашел. Вполне возможно, что даже подмножество следующих инструкций одинаково достаточно для решения проблемы.

Быстрые инструкции без особых подробностей

Выполните следующие инструкции:

  1. устанавливать g++-6 (любая версия новее, чем 5.2 который поддерживает c++14 будут
    скорее всего тоже работает, хотя я их не проверял).
  2. Установить буст 1.58 или новее (поддержка coroutine2). По состоянию на дату
    в этом посте последняя версия, доступная в Ubuntu PPA
    1.54, Поэтому вам нужно вручную установить буст. Если вы не знаете как, следуйте эта ссылка.
  3. Запустите вашу программу, используя -std=c++14 флаг. Кроме того, если вы работаете с сопрограммами, как я, не забудьте связать с libboost_context, Я заметил, что для сопрограмм 2, libboost_system не нужно, возможно, потому что он автоматически втягивается libboost_contextХотя я не уверен в причине.
  4. Удостоверьтесь, что вы ссылаетесь на свои недавно установленные библиотеки boost, а не на старую. Увидеть этот ТАК вопрос чтобы увидеть, что может пойти не так. В моем случае мне пришлось удалить все ранее установленные буст с моих машин.

Чтобы объединить эти моменты, вот команда, которую я использовал для компиляции и компоновки своего кода (скажем, я установил boost в /usr/local/):

g++-6 -std=c++14 example.cpp -I/usr/local/include -L/usr/local/lib -lboost_coroutine -lboost_context -lboost_system

Подробнее

По-видимому, boost.coroutine библиотека в бусте 1.54 реализован с устаревшим C-подобным fcontext-API из boost.context [источник]. Я заметил, что у моей машины были проблемы с компиляцией этих частей. К сожалению, Ubuntu PPA не предлагает более новую версию, чем 1.54 в этот момент. Boost.coroutine2 рекомендуемая библиотека, которую можно найти в версии 1.59 или новее. Поэтому вам нужно вручную загрузить подходящую версию boost с ее сайта и установить ее.

После этого мой код успешно compilied, но нет связанный:

... undefined reference to `ontop_fcontext'  `boost::context::basic_fixedsize_stack<boost::context::stack_traits>::deallocate(boost::context::stack_context&)':
example.cpp:(...): undefined reference to `boost::context::stack_traits::minimum_size()'
example.cpp:(...): undefined reference to `boost::context::stack_traits::is_unbounded()'
example.cpp:(...): undefined reference to `boost::context::stack_traits::maximum_size()'

Удивительно, добавляя -lboost_context флаг команды компиляции не удалил эту ошибку ссылки, предполагая, что libboost_context не правильно связан с.

Потом наткнулся вопрос в форуме повышения, где у оригинального плаката была проблема, похожая на мою. Короче говоря, существует ряд макросов C, которые препятствуют правильному включению execute_context, и, следовательно, проблема с libboost_context, Код, позволяющий узнать, включены ли некоторые из этих макросов в вашем компиляторе, выглядит следующим образом (я не уверен, что все из этих макросов актуальны):

#include <boost/config.hpp>

#if defined(BOOST_NO_CXX11_AUTO_DECLARATIONS) || \
defined(BOOST_NO_CXX11_CONSTEXPR) || \
defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) || \
defined(BOOST_NO_CXX11_FINAL) || \
defined(BOOST_NO_CXX11_HDR_TUPLE) || \
defined(BOOST_NO_CXX11_NOEXCEPT) || \
defined(BOOST_NO_CXX11_NULLPTR) || \
defined(BOOST_NO_CXX11_RVALUE_REFERENCES) || \
defined(BOOST_NO_CXX11_TEMPLATE_ALIASES) || \
defined(BOOST_NO_CXX11_UNIFIED_INITIALISATION_SYNTAX) || \
defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || \
defined(BOOST_NO_HDR_ATOMIC) || \
defined(BOOST_NO_HDR_TUPLE)
#error "execution_context is prevented to be included in this compiler";
#endif

Решение простое: установка g++-6 (или я Угадай 5.2 или новее), и компилирование кода с std=-c++14,

1

Другие решения

Других решений пока нет …