OpenCVを使ったC++コードをコンパイルする(CMake, GCC, pkg-config)


(修正)
CMakeLists.txt中の変数SOURCE_CODESOURCE_CORDEになっていました。
以前のままでも問題なく動きますが一応訂正しておきます。

(追記 2020/02/18)
2.3 pkg-configでg++にインクルードパスとライブラリパスを渡す の最後に行うテストファイル実行時、
"error while loading shared libraries: libopencv_core.so.*.*: cannot open shared object file: No such file or directory"
とエラー出てしまう場合があるらしいので2.4に対処法を載せました。

はじめに

(CMake)WSLに公式OpenCVをやさしくインストール (C/C++, Python2, Python3)の続きです。
OpenCVのバージョンを出力するC++のソースコードをコンパイルします。
方法としては

1. CMakeを使ってMakefileを作る
2. pkg-configを使ってパスを通す

を説明します。

前提
1. WSLでUbuntuを入れてある
2. CMakeとpkg-configをインストールしてある
3. OpenCVをCMakeでインストールしている

これ以降エディタとしてGNU nanoを使用しますが各自お好みのものを使用してください。(GNU nano は editor コマンドで呼び出します)

準備

テストコード(opencv_test)の作成

~/opencv直下にcpp_version_testというディレクトリを作成します。

$ mkdir cpp_version_test
$ cd cpp_version_test # 今cpp_version_testにいる

GNU nano でOpenCVのバージョンを出力するコード(opencv_test)を作成します。

$ editor opencv_test.cpp # 今もcpp_version_testにいる

GNU nanoが開かれたら以下のコードを貼り付けてください。
注意: GNU nanoでは貼り付けを^U(Ctrl + u)で行いますが、これはUbuntu上のバッファ領域にあるデータを貼り付けるものなのでWindowsでコピーしたデータには適用されません。
今まで通り右クリックで貼り付けてください。

opencv_test.cpp
// This file is part of OpenCV project.
// It is subject to the license terms in the LICENSE file found in the top-level directory
// of this distribution and at http://opencv.org/license.html

#include <opencv2/core/utility.hpp>
#include <iostream>

static const std::string keys = "{ b build | | print complete build info }"
                                "{ h help  | | print this help           }";

int main(int argc, const char* argv[])
{
    cv::CommandLineParser parser(argc, argv, keys);
    parser.about("This sample outputs OpenCV version and build configuration.");
    if (parser.has("help"))
    {
        parser.printMessage();
    }
    else if (!parser.check())
    {
        parser.printErrors();
    }
    else if (parser.has("build"))
    {
        std::cout << cv::getBuildInformation() << std::endl;
    }
    else
    {
        std::cout << "Welcome to OpenCV " << CV_VERSION << std::endl;
    }
    return 0;
}

貼り付けたら^X(Ctrl + x)で終了して、y(はい)→ Enterの順に打ち込んでください。

1 CMakeを使う方法

公式ページ(Using OpenCV with gcc and CMake)を参考に進めていきます。

1.1 CMakeLists.txtの作成

CMakeLists.txtを作成します。
準備と同様にGNU nanoを使っていきます。

$ editor CMakeLists.txt # 今cpp_version_testにいる

以下のコードを貼り付けて、保存してください。

CMakeLists.txt
# 変数SOURCE_CORDEを宣言し、opencv_testという値を入れる。
# cmake -D SOURCE_CODE=(ソース名)で上書き可
set(SOURCE_CODE opencv_test CACHE NAME "Target object name")

# CMakeの最低バージョンを記述
cmake_minimum_required(VERSION 2.8)

# ソリューション名を指定
project( ${SOURCE_CODE} )

# OpenCVのパッケージを探す
find_package( OpenCV REQUIRED )

#ヘッダファイルのパスを指定
include_directories( ${OpenCV_INCLUDE_DIRS} )

# 実行ファイル名とソース指定(ここではソースと同じ名前の実行ファイルを作ります)
add_executable( ${SOURCE_CODE} ${SOURCE_CODE}.cpp )

#リンク先のライブラリを指定
target_link_libraries( ${SOURCE_CODE} ${OpenCV_LIBS} )

これで準備完了です!

1.2 ビルドして実行

ここからはopencvをインストールしたときと同じ手順です。
まず、buildディレクトリを作成します。

$ mkdir build
$ cd build # buildに移動

つぎに、cmakeでmakefileを作成します。
もちろんtwo dots .. を忘れずに。

$ cmake .. # 今buildにいる

これでbuildディレクトリ直下にmakefileが作成されたのでmakeします。

$ make # 今もbuildにいる

実行ファイルが生成されたので実行します!

$ ./opencv_test # 今もbuildにいる

Welcome to OpenCV (version)と表示されれば動作確認完了です!

opencv_test以外の名前のコードをコンパイルしたい

基本的に今回作成したCMakeLists.txtを利用すればできます。
1. 上記と同様の手順でソースと同じディレクトリにCMakeLists.txtを作成し、buildディレクトリに移動
2. cmake -D SOURCE_CODE=(コード名(拡張子なし)) ..と実行すれば、ターゲットが指定したコードに代わります。
3. あとはmakeビルドして実行してください。

2. pkg-configでパスを通す方法

2.1 pkg-configとは

そもそもpkg-configとは何なんでしょうか。

Wikipedia(pkg-config)にはこうあります。

pkg-configは、環境変数PKG_CONFIG_PATHのパスに存在する *.pc ファイルに記録された情報を元に、ビルドの際に必要な文字列を返す。

ビルドの際に必要な文字列とは、今回でいうopencvのヘッダファイル#include <opencv2/core/utility.hpp>や、その実行に必要なライブラリのパスを示しています。

さらに、この*.pcファイルですが、実は3.2のCMake実行時のオプション指定で生成するように設定してあります。
公式ページ(Installation in Linux) にもこうあります。

つまり-DOPENCV_GENERATE_PKGCONFIG=ONを設定していればCMakeを使わない場合でもコンパイルできるということですね。

2.2 pkg-configを使うには

もう一度Wikipedia(pkg-config)を見てみましょう。

pkg-configは、環境変数PKG_CONFIG_PATHのパスに存在する *.pc ファイルに記録された情報を元に、ビルドの際に必要な文字列を返す。

pkg-configを使うには*.pcファイルが存在するパスをPKG_CONFIG_PATHに教えてあげないといけないんですね。

OpenCVインストール時に/usr/local/lib/pkgconfigopencv4.pcというファイルが作成されているのでこれをPKG_CONFIG_PATHに与えます。
以下のコマンドを実行してください。

$ export PKG_CONFIG_PATH==/usr/local/lib/pkgconfig

これでpkg-configを使えます。

注意: exportで値を追加した環境変数はターミナルを閉じるまでは値を保持し続けます。
逆に言えば、ターミナルを開くたびに上記のexport (以下省略)を実行しなければいけません。

いちいちパスを通したくないよ

いちいちパスを通すのが面倒だという人は~/.basrcにさきほどのコマンドを追記しておきましょう。
.bashrcは記述されたコマンドをターミナル起動時に自動で実行しておいてくれる設定ファイルです)

以下のコマンドを実行するか

$ echo export PKG_CONFIG_PATH==/usr/local/lib/pkgconfig >> ~/.bashrc

または、エディタを使って~/.bashrcの一番下に
export PKG_CONFIG_PATH==/usr/local/lib/pkgconfigをコピペしてください。
[例]

$ editor ~/.bashrc #GNU nanoで.bashrcを開く

一番下の行にexport PKG_CONFIG_PATH==/usr/local/lib/pkgconfigをコピペして保存すれば、次回からUbuntuを起動するときに自動で実行されます。

2.3 pkg-configでg++にインクルードパスとライブラリパスを渡す

さて、そろそろコンパイルの時間です。
まず、以下のコマンドを実行してopencvのインクルードパスを見てみましょう

$ pkg-config --cflags opencv4

おそらく-I/usr/local/include/opencv4/opencv-I/usr/local/include/opencv4などと出てきたのではないでしょうか。
この-Iはg++にインクルードパスを渡すオプションです。

つぎに、以下のコマンドを実行してopencvのライブラリパスを見てみましょう

$ pkg-config --libs opencv4

おそらく-L/usr/local/libとその横に-lから始まる文字列がたくさん出てきたのではないでしょうか。
この-L-lはそれぞれ、g++にライブラリパスを渡すオプションと、使用するライブラリ指定するオプションです。

g++でコンパイルするときは上記2つの実行結果をg++に渡さなければいけません。しかし、これはかなり面倒くさいとわかると思います。

そこで、例えばopencv_test.cppをコンパイルするときは以下のように実行します

$ g++ opencv_test.cpp `pkg-config --cflags --libs opencv4` -o opencv_test

シングルクォート`で囲むことでその部分を実行結果に置き換えることができるのです。

実行しましょう!

$ ./opencv_test

バージョンが表示されれば成功です!
お疲れ様です!

2.4 エラーが出る場合

2.3 の最後に行うテストファイル実行時

error while loading shared libraries: libopencv_core.so.*.*: cannot open shared object file: No such file or directory

とエラーが出てしまう場合があるらしいので、対処法を載せておきます。

そもそものエラーの原因は、/etc/ld.so.cache中のキャッシュが書き換わっていないことが原因です。

以下のコマンドを実行してください。

$ sudo ldconfig

以上で解消します。

3. 参考

CMakeの使い方 その1~3(https://qiita.com/shohirose/items/45fb49c6b429e8b204ac#_reference-27dce2bd429efef9488e)