LoginSignup
6
8

More than 5 years have passed since last update.

OpenCV は物体認識などのライブラリが含まれているので、Windows IoT Core で使えると利用できる幅が広がりそうですよね。Linux 系の Raspbian の場合はソースコードをダウンロードしてビルドして導入という手順になりますが、Windows IoT Core の場合は UWP アプリにしないといけない難点があってちょっとハードルが高くなっています。が、できないことはないです。

Windows 10ユニバーサルアプリ(Universal Windows Application)でOpenCVを使う(その2) - embeddedなブログ

なところに、導入&プロジェクト作成手順が書いてあるので、それに従うとうまくいきます。

いくつかのポイントがあって、

  • github の Microsoft/opencv の vs2015-samples というブランチを使う。
  • このブランチが OpenCV 3.0 ベースになっている(現時点で最新が 3.2)
  • コードは C++/CLI で記述する。

となっています。Microsoft/opencv は OpenCV/opencv のフォークになっているのですが、どちらも Microsoft のほうは1年位更新されていません。ただし、x86/x64版(Intel版)のほうは、本家の OpenCV/opencv で動作させることができました。ARM のほうがうまくビルドできないのは、ippicv 絡みなので、cmake するときに外してやるとライブラリはうまくビルドできるようにになります。が、UWP on RPi への組み込みは失敗してしまいます。この経緯は後で自前のブログにでも記録として残しておきます。

Microsoft/opencv の vs2015-samples をダウンロードする

Microsoft/opencv at vs2015-samples で ブランチを「vs2015-samples」に変更して zip でダウンロードします。

001.jpg

zip を解凍します。

環境変数 OCV2015_ROOT を設定する

ARM 版の OpenCV をビルドすることになるのですが、インクルードパスが「$(OCV2015_ROOT)\vs2015\WS\10.0\ARM;...」のように設定されいるので、環境変数 OCV2015_ROOT を解凍先の zip に合わせて設定します。

002.jpg

ユーザー環境変数に設定したら、一度ログアウトしてログインし直すと、環境変数が設定されます。

ARM 版の OpenCV をビルドする

本来であれば、cmake を使って make を作って、という手順になるのですが、vs2015-samples は cmake まで済ませてあります。なので、$OCV2015_ROOT\vs2015\WS\10.0\ARM\OpenCV.sln を開いて、ALL_BUILD をビルドします。

003.jpg

無事ビルドが終わると、$OCV2015_ROOT\vs2015\WS\10.0\ARM\bin\Debug\ にアセンブリ(.dll)が、$OCV2015_ROOT\vs2015\WS\10.0\ARM\lib\Debug\ にエクスポートされたライブラリ(.lib)が作成されます。デバッグの場合は Debug、リリース版は Release を使えば ok です。

OpenCVExample をビルドしてみる

ms-iot のサンプル samples/OpenCVExample at develop · ms-iot/samples をビルドしてみましょう。ちなみに、OpenCV | Windows IoT に OpenCV をビルドするドキュメントがあるのですが、肝心の opencv をダウンロードするリンクが切れている(間違っている?)ので、頓挫してしまいます。このリンク先は、先の vs2015-samples ブランチに変更されています。

OpenCV の DLL をプロジェクトにごっそり追加します。本来は選んだほうがサイズが小さくなるのですが、面倒なので一気に追加します。

004.jpg

DLL のプロパティでコンテンツを「True」に変更しておきます。こうすると、DLL がごっそり *.exe と同じところにコピーされてアプリに追加されます。

005.jpg

あとは、ソリューションプラットフォームを「ARM」に変更してビルドをします。いくつかの警告がでますが、無事ビルドができます。

ちなみに、DLL は ARM 版を追加するので、x64 にしたときは動作しません。普通の PC の UWP アプリを使いたい時は、x64 のほうにある opencv の DLL をコピーします。ここは面倒なので2つプロジェクトを作ってしまったほうがいいでしょうね。あるいは、ビルド前プロセスなどで自動化できると思います。

実機で動かしてみる。

いままでと同じように「リモートコンピュータ」の先を minwinpc にして動かすと、こんな感じで動作します。

006.jpg

ちなみにアプリケーションやDLLは、Uドライブのほうに保存されています。

007.jpg

試しに、x64 でビルドして PC で動かしたものはこんな感じです。

008.jpg

OpenCV on Windows IoT Core の使いどころ

多少バージョンが古くなってしまいますが、これで Windows IoT Core でも OpenCV が使えることが分かります。まあ、本格的な画像処理を RPi でやるには非力なので、写真の前処理とかちょっとした色の加工とか、物体認識とか簡単なテンプレートマッチングとかに使うといいんじゃないでしょうか。

多くの方にとっての難点は C++/CLI で記述するというところです。

// takes an image (inputImg), runs face and body classifiers on it, and stores the results in objectVector and objectVectorBodies, respectively
void InternalDetectObjects(cv::Mat& inputImg, std::vector<cv::Rect> & objectVector, std::vector<cv::Rect> & objectVectorBodies, cv::CascadeClassifier& face_cascade, cv::CascadeClassifier& body_cascade)
{
    cv::Mat frame_gray;

    cvtColor(inputImg, frame_gray, CV_BGR2GRAY);
    cv::equalizeHist(frame_gray, frame_gray);

    // Detect faces
    face_cascade.detectMultiScale(frame_gray, objectVector, 1.1, 2, 0 | CV_HAAR_SCALE_IMAGE, cv::Size(30, 30));
    //detect bodies
    body_cascade.detectMultiScale(frame_gray, objectVectorBodies, 1.1, 2, 0 | CV_HAAR_SCALE_IMAGE, cv::Size(30, 300));
}

このあたりを適当に C# でラップするだけでも使い勝手はよくなるかなと(本格的なものは 2.4 のほうにあるのだけど、3.0 にはない)。

あと、画像まわりが、WriteableBitmap と cv::Mat の相互変換とか、Bitmap 形式とかが混在するので、結構ややこしいかもしれません。OpenCVExample では、下記のようにベタで memcpy していました。

void  OpenCVExample::MainPage::UpdateImage(const cv::Mat& image)
{
    // Create the WriteableBitmap
    WriteableBitmap^ bitmap = ref new WriteableBitmap(image.cols, image.rows);

    // Get access to the pixels
    IBuffer^ buffer = bitmap->PixelBuffer;
    unsigned char* dstPixels = nullptr;

    // Obtain IBufferByteAccess
    ComPtr<IBufferByteAccess> pBufferByteAccess;
    ComPtr<IInspectable> pBuffer((IInspectable*)buffer);
    pBuffer.As(&pBufferByteAccess);

    // Get pointer to pixel bytes
    HRESULT get_bytes = pBufferByteAccess->Buffer(&dstPixels);
    if (get_bytes == S_OK) {
        memcpy(dstPixels, image.data, image.step.buf[1] * image.cols*image.rows);

        // Set the bitmap to the Image element
        storedImage->Source = bitmap;
    }
    else {
        printf("Error loading image into buffer\n");
    }
}
6
8
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
6
8