2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

【OpenVINO】顔マスク検出モデル

2
Last updated at Posted at 2021-07-04

はじめに

AIがホットワードとして注目を集めています。
特に画像解析の学習済みモデルが容易に手に入り、学習コストが低く導入できることから敷居が低くなっているかと思います。(自分が導入できたのはそのおかげですが、、)

しかし、昨今のコロナ禍によりマスクを装着することが常態化し、顔を検出しようとしてもマスクがあることで精度が低くなってしまうことが多々あるかと思います。

そこで、拾い物ではありますが、顔マスクを検出するモデルを見つけて実装してみたので、その例をご紹介したいと思います。

関連リンク

以下のGitHubを参考にさせていただきました。必要であれば参考にしてください。。

前提

  • Ubuntu 18.04
  • python3
  • OpenVINO 2020.04 LTS を推論エンジンとして使用しています

OpenVINOとは

OpenVINOツールキット(Open Visual Inference and Neural network Optimization)は、フレームワークからの深層学習モデルの最適化と、推論エンジンを使用したIntelハードウェアへの展開を容易にする無料のツールキットです。

導入方法などについては、他の方が記事にしていたりするので割愛します。
導入後は以下の記事などを例にして実装してみると理解が早いと思います。

使い方

学習済みモデルを読み込ませて、APIとして呼び出しをします。

マスク検出モデル

学習済みモデルをあげてくださっているので、こちらを使用していきます。
以下、「実装」部分のパスをこちらの学習済みモデルに指定してください。

image.png

実装

Python で実装しました。
簡単にコメントを残していますので、参考までに・・・。

import cv2
import numpy as np
from openvino.inference_engine import IECore

iecore = IECore()

# モデルの読み込み(顔検出)
# パスは変更してください (上記「マスク検出モデル」で取得した学習済みモデルを使用)
model_path = './face_mask_detection_openvino-master/models/face-detection-adas-0001.xml'
weights_path = './face_mask_detection_openvino-master/models/face-detection-adas-0001.bin'
face_net = iecore.read_network(model=model_path, weights=weights_path)
face_exec_net = iecore.load_network(network=face_net, device_name='CPU', num_requests=2)

# モデルの読み込み(マスク検出)
# パスは変更してください (上記「マスク検出モデル」で取得した学習済みモデルを使用)
origin_path = './face_mask_detection_openvino-master/models/'
mask_net = iecore.read_network(model=origin_path+'face_mask.xml', weights=origin_path+'face_mask.bin')
mask_exec_net = iecore.load_network(network=mask_net, device_name='CPU', num_requests=2)
mask_input_blob =  next(iter(mask_exec_net.inputs))
mask_output_blob = next(iter(mask_exec_net.outputs))

# カメラ準備
cap = cv2.VideoCapture(0)

# メインループ
while cap.isOpened():
    ret, frame = cap.read()

    # Reload on error
    if ret == False:
        continue

    # 顔検出用の入力データフォーマットへ変換
    img = cv2.resize(frame, (300, 300))   # サイズ変更
    img = img.transpose((2, 0, 1))    # HWC > CHW
    img = np.expand_dims(img, axis=0) # 次元合せ

    # 顔検出 推論実行
    out = face_exec_net.infer(inputs={'data': img})

    # 出力から必要なデータのみ取り出し
    out = out['detection_out']
    out = np.squeeze(out) #サイズ1の次元を全て削除

    # 検出されたすべての顔領域に対して1つずつ処理
    for detection in out:
        # conf値の取得
        confidence = float(detection[2])

        # conf値が0.5より大きい場合のみ感情推論とバウンディングボックス表示
        if confidence > 0.5:
            # バウンディングボックス座標を入力画像のスケールに変換
            xmin = int(detection[3] * frame.shape[1])
            ymin = int(detection[4] * frame.shape[0])
            xmax = int(detection[5] * frame.shape[1])
            ymax = int(detection[6] * frame.shape[0])

           # 顔検出領域はカメラ範囲内に補正する。特にminは補正しないとエラーになる
            if xmin < 0:
                xmin = 0
            if ymin < 0:
                ymin = 0
            if xmax > frame.shape[1]:
                xmax = frame.shape[1]
            if ymax > frame.shape[0]:
                ymax = frame.shape[0]

            # 顔領域のみ切り出し
            frame_face = frame[ ymin:ymax, xmin:xmax ]

            # マスク検出モデル 入力データフォーマットへ変換
            img = cv2.resize(frame_face, (224, 224))   # サイズ変更
            img = img.transpose((2, 0, 1))    # HWC > CHW
            img = np.expand_dims(img, axis=0) # 次元合せ

            # マスク検出 推論実行
            out = mask_exec_net.infer(inputs={mask_input_blob: img})

            # 出力から必要なデータのみ取り出し
            mask_out = out[mask_output_blob]
            mask_out = np.squeeze(mask_out) #不要な次元の削減

            # 文字列描画
            if int(mask_out) > 0:
                display_text = 'With Face Mask'
            else:
                display_text = 'No Face Mask'

            # 文字列描画
            cv2.putText(frame, display_text, (xmin, ymin), cv2.FONT_HERSHEY_SIMPLEX, 1, (255, 255, 255), 2)
            # バウンディングボックス表示
            cv2.rectangle(frame, (xmin, ymin), (xmax, ymax), color=(240, 180, 0), thickness=3)


    # 画像表示
    cv2.imshow('frame', frame)

    # 何らかのキーが押されたら終了
    key = cv2.waitKey(1)
    if key == 27:
        break

# 終了処理
cap.release()
cv2.destroyAllWindows()

評価(感想)

遊びがてらに使ってみたので、精度の評価などは行っていません。
使ってみた感想として、以下の点が少し気になりました。

  • intel が提供している学習済みモデルと比べると重い
  • 精度が思ったよりも出ていない気がする
  • 複数人だと判定できない?

まとめ

マスク常備がいつまで続くかわからないので、マスク着用判定モデルを作成する人は少ないのかもしれませんね。
自分が調べ切れていないだけの可能性もあるので、何か情報をお持ちであれば教えていただきたいです。

また、自分で学習させるスキルをつけた方がいいのか悩みどころ。。


会社紹介

株式会社 Mosaica
最先端テクノロジーで社会課題を解決し、持続可能な未来を創造する IT カンパニー。
AI ソリューション、クラウド統合、DX 推進、経営コンサルティングなど包括的なサービスでビジネス変革を支援しています。

詳しくは 公式サイト までお気軽にご相談ください。
公式サイト: https://mosaica.co.jp/

2
2
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
2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?