1. 程式人生 > >DetNet或Faster R-CNN實時視訊流demo(pytorch版)

DetNet或Faster R-CNN實時視訊流demo(pytorch版)

        最近要使用faster-rcnn,DetNet-FPN以及Light-Head三種目標檢測方案訓練自己的資料集,並做一個比較。在GitHub上搜羅了一番,發現下面三個開源專案一脈相承,正合我意。期中DetNet_pytorch和pytorch-lighthead我認為都是基於faster-rcnn.pytorch這個專案改的,作者貌似是佐治亞理工的Ph.D.,確實是高手中的高高手,程式碼寫得很流暢。我三個專案都fork了下來,做了不少的修改以適配自己的資料集(不能一天到晚玩coco、voc嘛),期中faster-rcnn.pytorch和DetNet_pytorch都成功地進行了適配並做了大量實驗,唯獨pytorch-lighthead死活沒有搞成功。哎,暫且不管了,考慮到我自己的資料集分類數很有限,用lighthead理論上並不會有啥優勢。事實上,我後面嘗試了LightHead的tensorflow版本,也印證了我的想法,mAP差異甚小。

        以使用的較多的DetNet_pytorch為例,配合程式碼介紹視訊流demo(輸入為視訊檔案而非單張圖片)。對於DetNet的實現原理,不再做過多闡述,這一部分網路上已有大量詳細介紹的文章。

1. 準備

        按照作者文件上的要求安裝好一些依賴包,這個過程可能會有一點小挫折,別一開始就放棄。

  • Python 2.7 or 3.6
  • Pytorch 0.2.0 or higher(not support pytorch version >=0.4.0)
  • CUDA 8.0 or higher

        從Git上下載程式碼,完成編譯。編譯也有可能遇到錯誤,繼續鍛鍊你Linux運維能力。

cd lib
sh make.sh

2.程式碼調整

        作者已經提供了一個demo.py的程式,雖然不是我想要的視訊流demo,但主體架構沒問題,只要在這上面進行二次開發即可。以下是我修改後的demo.py程式,以供參考。如果是想要修改為faster rcnn版的視訊流demo也是很容易的,可自行修改。

# coding: utf-8
# --------------------------------------------------------
# Tensorflow Faster R-CNN
# Licensed under The MIT License [see LICENSE for details]
# Written by Jiasen Lu, Jianwei Yang, based on code from Ross Girshick
# --------------------------------------------------------
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function

import _init_paths
import os
import sys
import numpy as np
import argparse
import pprint
import pdb
import time
import cv2
import cPickle
import torch
from torch.autograd import Variable
import torch.nn as nn
import torch.optim as optim
import glob
import torchvision.transforms as transforms
import torchvision.datasets as dset
from PIL import Image
from scipy.misc import imread
from roi_data_layer.roidb import combined_roidb
from roi_data_layer.roibatchLoader import roibatchLoader
from model.utils.config import cfg, cfg_from_file, cfg_from_list, get_output_dir

from model.rpn.bbox_transform import clip_boxes
from model.nms.nms_wrapper import nms
from model.rpn.bbox_transform import bbox_transform_inv
from model.utils.net_utils import save_net, load_net, vis_detections
from model.utils.blob import prep_im_for_blob,im_list_to_blob
import pdb
from model.fpn.detnet_backbone import detnet

import warnings
warnings.filterwarnings("ignore")

def parse_args():
  """
  Parse input arguments
  """
  #減少篇幅,此處省略若干

  args = parser.parse_args()
  return args

lr = cfg.TRAIN.LEARNING_RATE
momentum = cfg.TRAIN.MOMENTUM
weight_decay = cfg.TRAIN.WEIGHT_DECAY

def get_image_blob(im):
  """Converts an image into a network input.
  Arguments:
    im: data of image
  Returns:
    blob (ndarray): a data blob holding an image pyramid
    im_scale_factors (list): list of image scales (relative to im) used
      in the image pyramid
  """
  im_scales = []
  processed_ims = []
  scale_inds = np.random.randint(0, high=len(cfg.TRAIN.SCALES), size=1)

  target_size = cfg.TRAIN.SCALES[scale_inds[0]]
  im, im_scale = prep_im_for_blob(im, cfg.PIXEL_MEANS, cfg.PIXEL_STDS, target_size, cfg.TRAIN.MAX_SIZE)

  im_scales.append(im_scale)
  processed_ims.append(im)

  # Create a blob to hold the input images
  blob = im_list_to_blob(processed_ims)

  return blob, im_scales

if __name__ == '__main__':

  args = parse_args()

  print('Called with args:')
  print(args)

  args.cfg_file = "cfgs/{}.yml".format(args.net)
  if args.cfg_file is not None:
    cfg_from_file(args.cfg_file)
  if args.set_cfgs is not None:
    cfg_from_list(args.set_cfgs)

  #print('Using config:')
  #pprint.pprint(cfg)
  np.random.seed(cfg.RNG_SEED)
  cfg.TRAIN.USE_FLIPPED = False

  # train set
  # -- Note: Use validation set and disable the flipped to enable faster loading.

  if args.exp_name is not None:
    input_dir = args.load_dir + "/" + args.net + "/" + args.dataset + '/' + args.exp_name
  else:
    input_dir = args.load_dir + "/" + args.net + "/" + args.dataset
  if not os.path.exists(input_dir):
    raise Exception('There is no input directory for loading network from ' + input_dir)
  load_name = os.path.join(input_dir,
    'fpn_{}_{}_{}.pth'.format(args.checksession, args.checkepoch, args.checkpoint))

  classes = cfg.TRAIN.CLASSES
  fpn = detnet(classes, 59, pretrained=False, class_agnostic=args.class_agnostic)
  fpn.create_architecture()

  print('load checkpoint %s' % (load_name))
  checkpoint = torch.load(load_name)

  fpn.load_state_dict(checkpoint['model'])
  if 'pooling_mode' in checkpoint.keys():
      cfg.POOLING_MODE = checkpoint['pooling_mode']

  # initilize the tensor holder here.
  im_data = torch.FloatTensor(1)
  im_info = torch.FloatTensor(1)
  num_boxes = torch.LongTensor(1)
  gt_boxes = torch.FloatTensor(1)

  # ship to cuda
  if args.ngpu > 0:
    im_data = im_data.cuda()
    im_info = im_info.cuda()
    num_boxes = num_boxes.cuda()
    gt_boxes = gt_boxes.cuda()

  # make variable
  im_data = Variable(im_data, volatile=True)
  im_info = Variable(im_info, volatile=True)
  num_boxes = Variable(num_boxes, volatile=True)
  gt_boxes = Variable(gt_boxes, volatile=True)

  if args.ngpu > 0:
    cfg.CUDA = True

  if args.ngpu > 0:
    fpn.cuda()

  fpn.eval()

  max_per_image = 100
  thresh = 0.05
  vis_thresh = 0.8
  vis = True

  if not os.path.exists(args.video_file):
      raise Exception("video %s not exist".format(args.video_file))

  vc = cv2.VideoCapture(args.video_file)
  i = 0
  while True:
      i += 1
      _, im = vc.read()
      if im is None:
          break

      blobs, im_scales = get_image_blob(im)
      assert len(im_scales) == 1, "Only single-image batch implemented"
      im_blob = blobs
      # (h,w,scale)
      im_info_np = np.array([[im_blob.shape[1], im_blob.shape[2], im_scales[0]]], dtype=np.float32)
      im_data_pt = torch.from_numpy(im_blob)
      # exchange dimension->(b,c,h,w)
      im_data_pt = im_data_pt.permute(0, 3, 1, 2)

      im_info_pt = torch.from_numpy(im_info_np)
      #im_info_pt = im_info_pt.view(3)

      im_data.data.resize_(im_data_pt.size()).copy_(im_data_pt)
      im_info.data.resize_(im_info_pt.size()).copy_(im_info_pt)
      gt_boxes.data.resize_(1, 1, 5).zero_()
      num_boxes.data.resize_(1).zero_()

      det_tic = time.time()
      rois, cls_prob, bbox_pred, \
          _, _, _, _, _ = fpn(im_data, im_info, gt_boxes, num_boxes)
      #pdb.set_trace()

      scores = cls_prob.data
      boxes = rois.data[:, :, 1:5]

      if cfg.TEST.BBOX_REG:
            # Apply bounding-box regression deltas
            box_deltas = bbox_pred.data
            if cfg.TRAIN.BBOX_NORMALIZE_TARGETS_PRECOMPUTED:
                # Optionally normalize targets by a precomputed mean and stdev
                if args.class_agnostic:
                    box_deltas = box_deltas.view(-1, 4) * torch.FloatTensor(cfg.TRAIN.BBOX_NORMALIZE_STDS).cuda() \
                                 + torch.FloatTensor(cfg.TRAIN.BBOX_NORMALIZE_MEANS).cuda()
                    box_deltas = box_deltas.view(1, -1, 4)
                else:
                    box_deltas = box_deltas.view(-1, 4) * torch.FloatTensor(cfg.TRAIN.BBOX_NORMALIZE_STDS).cuda() \
                                 + torch.FloatTensor(cfg.TRAIN.BBOX_NORMALIZE_MEANS).cuda()
                    box_deltas = box_deltas.view(1, -1, 4 * len(classes))

            #pdb.set_trace()
            pred_boxes = bbox_transform_inv(boxes, box_deltas, 1)
            pred_boxes = clip_boxes(pred_boxes, im_info.data, 1)
      else:
          # Simply repeat the boxes, once for each class
          pred_boxes = np.tile(boxes, (1, scores.shape[1]))

      pred_boxes /= im_scales[0]

      scores = scores.squeeze()
      pred_boxes = pred_boxes.squeeze()
      det_toc = time.time()
      detect_time = det_toc - det_tic

      if vis:
          im2show = np.copy(im)

      sys.stdout.write('im_detect: {:d} {:.3f}s   \r'.format(i, detect_time))
      sys.stdout.flush()

      for j in xrange(1, len(classes)): # 0 for background
          inds = torch.nonzero(scores[:,j] > thresh).view(-1)
          # if there is det
          if inds.numel() > 0:
            cls_scores = scores[:,j][inds] # confidence of the specified class
            _, order = torch.sort(cls_scores, 0, True) # sorted scores and indexes
            if args.class_agnostic:
              cls_boxes = pred_boxes[inds, :]
            else:
              cls_boxes = pred_boxes[inds][:, j * 4:(j + 1) * 4]

            cls_dets = torch.cat((cls_boxes, cls_scores.unsqueeze(1)), 1)
            # cls_dets = torch.cat((cls_boxes, cls_scores), 1)
            cls_dets = cls_dets[order]
            keep = nms(cls_dets, cfg.TEST.NMS) # after nms
            cls_dets = cls_dets[keep.view(-1).long()] # keep shape is ?x1
            if vis:
              # cls_dets.cpu().numpy() make tensor->numpy array
              im2show = vis_detections(im2show, classes[j], cls_dets.cpu().numpy(), vis_thresh)
              #drawpath = os.path.join('images', "{}.jpg".format(i))
              #cv2.imwrite(drawpath, im2show)
      cv2.imshow('demo', im2show)
      if (cv2.waitKey(25) & 0xFF) == ord('q'):
          break
  vc.release()
  cv2.destroyAllWindows()

3.執行

【程式碼】