grabcut_nodelet.cpp
Go to the documentation of this file.
1 // -*- mode: c++ -*-
2 /*********************************************************************
3  * Software License Agreement (BSD License)
4  *
5  * Copyright (c) 2014, JSK Lab
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  *
12  * * Redistributions of source code must retain the above copyright
13  * notice, this list of conditions and the following disclaimer.
14  * * Redistributions in binary form must reproduce the above
15  * copyright notice, this list of conditions and the following
16  * disclaimer in the documentation and/or other materials provided
17  * with the distribution.
18  * * Neither the name of the JSK Lab nor the names of its
19  * contributors may be used to endorse or promote products derived
20  * from this software without specific prior written permission.
21  *
22  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
23  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
24  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
25  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
26  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
27  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
28  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
30  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
32  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33  * POSSIBILITY OF SUCH DAMAGE.
34  *********************************************************************/
35 
36 #include "jsk_perception/grabcut.h"
37 #include <cv_bridge/cv_bridge.h>
39 #include <opencv2/opencv.hpp>
40 
41 namespace jsk_perception
42 {
43  void GrabCut::onInit()
44  {
45  DiagnosticNodelet::onInit();
46  srv_ = boost::make_shared <dynamic_reconfigure::Server<Config> > (*pnh_);
47  dynamic_reconfigure::Server<Config>::CallbackType f =
48  boost::bind (
49  &GrabCut::configCallback, this, _1, _2);
50  srv_->setCallback (f);
52  = advertise<sensor_msgs::Image>(*pnh_, "output/foreground", 1);
54  = advertise<sensor_msgs::Image>(*pnh_, "output/background", 1);
56  = advertise<sensor_msgs::Image>(*pnh_, "output/foreground_mask", 1);
58  = advertise<sensor_msgs::Image>(*pnh_, "output/background_mask", 1);
59  onInitPostProcess();
60  }
61 
63  // message_filters::Synchronizer needs to be called reset
64  // before message_filters::Subscriber is freed.
65  // Calling reset fixes the following error on shutdown of the nodelet:
66  // terminate called after throwing an instance of
67  // 'boost::exception_detail::clone_impl<boost::exception_detail::error_info_injector<boost::lock_error> >'
68  // what(): boost: mutex lock failed in pthread_mutex_lock: Invalid argument
69  // Also see https://github.com/ros/ros_comm/issues/720 .
70  sync_.reset();
71  }
72 
73  void GrabCut::subscribe()
74  {
75  sub_image_.subscribe(*pnh_, "input", 1);
76  sub_foreground_.subscribe(*pnh_, "input/foreground", 1);
77  sub_background_.subscribe(*pnh_, "input/background", 1);
78  sync_ = boost::make_shared<message_filters::Synchronizer<SyncPolicy> >(100);
80  sync_->registerCallback(boost::bind(&GrabCut::segment,
81  this, _1, _2, _3));
82  }
83 
85  {
89  }
90 
93  {
94  // foreground and background is not continuous,
95  // so just say OK
96  stat.summary(diagnostic_msgs::DiagnosticStatus::OK,
97  "GrabCut running");
98  }
99 
100  void GrabCut::segment(
101  const sensor_msgs::Image::ConstPtr& image_msg,
102  const sensor_msgs::Image::ConstPtr& foreground_msg,
103  const sensor_msgs::Image::ConstPtr& background_msg)
104  {
105  boost::mutex::scoped_lock lock(mutex_);
106  cv::Mat input;
107  cv::Mat in_image = cv_bridge::toCvShare(image_msg,
108  image_msg->encoding)->image;
109  if (in_image.channels() == 3) {
110  input = in_image;
111  }
112  else if (in_image.channels() == 1) {
113  input = cv::Mat::zeros(in_image.rows, in_image.cols, CV_8UC3);
114  for (size_t j = 0; j < in_image.rows; ++j) {
115  for (size_t i = 0; i < in_image.cols; ++i) {
116  input.at<cv::Vec3b>(j, i) = cv::Vec3b(in_image.at<uchar>(j, i),
117  in_image.at<uchar>(j, i),
118  in_image.at<uchar>(j, i));
119  }
120  }
121  }
122  cv::Mat foreground = cv_bridge::toCvCopy(
123  foreground_msg, sensor_msgs::image_encodings::MONO8)->image;
124  cv::Mat background = cv_bridge::toCvCopy(
125  background_msg, sensor_msgs::image_encodings::MONO8)->image;
126  if (!(input.cols == foreground.cols &&
127  input.rows == foreground.rows &&
128  background.cols == foreground.cols &&
129  background.rows == foreground.rows)) {
130  NODELET_WARN("size of image is not corretct");
131  return;
132  }
133  cv::Mat mask = cv::Mat::zeros(input.size(), CV_8UC1);
134  mask.setTo(cv::Scalar::all(cv::GC_PR_BGD));
135  for (size_t j = 0; j < input.rows; j++) {
136  for (size_t i = 0; i < input.cols; i++) {
137  if (foreground.at<uchar>(j, i) == 255) {
139  mask.at<uchar>(j, i) = cv::GC_PR_FGD;
140  }
141  else {
142  mask.at<uchar>(j, i) = cv::GC_FGD;
143  }
144  }
145  if (background.at<uchar>(j, i) == 255) {
147  mask.at<uchar>(j, i) = cv::GC_PR_BGD;
148  }
149  else {
150  mask.at<uchar>(j, i) = cv::GC_BGD;
151  }
152  }
153  }
154  }
155  cv::Rect roi;
156  cv::Mat bgd_model;
157  cv::Mat fgd_model;
158 
159  cv::grabCut(input, mask, roi, bgd_model, fgd_model, 5, cv::GC_INIT_WITH_MASK);
160  cv::Mat bgd, fgd, bgd_mask, fgd_mask;
161  // model -> mask
162  //cv::compare(mask, cv::GC_BGD, bgd_mask, cv::CMP_EQ);
163  bgd_mask = (mask == cv::GC_BGD) | (mask == cv::GC_PR_BGD);
164  //cv::compare(mask, cv::GC_FGD, fgd_mask, cv::CMP_EQ);
165  fgd_mask = (mask == cv::GC_FGD) | (mask == cv::GC_PR_FGD);
166  // mask -> image
167  input.copyTo(bgd, bgd_mask);
168  input.copyTo(fgd, fgd_mask);
169  cv_bridge::CvImage fg_bridge(
170  image_msg->header, sensor_msgs::image_encodings::BGR8, fgd);
171  cv_bridge::CvImage bg_bridge(
172  image_msg->header, sensor_msgs::image_encodings::BGR8, bgd);
173  cv_bridge::CvImage fg_mask_bridge(
174  image_msg->header, sensor_msgs::image_encodings::MONO8, fgd_mask);
175  cv_bridge::CvImage bg_mask_bridge(
176  image_msg->header, sensor_msgs::image_encodings::MONO8, bgd_mask);
177  pub_foreground_.publish(fg_bridge.toImageMsg());
178  pub_background_.publish(bg_bridge.toImageMsg());
179  pub_foreground_mask_.publish(fg_mask_bridge.toImageMsg());
180  pub_background_mask_.publish(bg_mask_bridge.toImageMsg());
181  }
182 
183  void GrabCut::configCallback(Config &config, uint32_t level)
184  {
185  boost::mutex::scoped_lock lock(mutex_);
186  use_probable_pixel_seed_ = (config.seed_pixel_policy == 1);
187  }
188 
189 }
190 
image_encodings.h
i
int i
jsk_perception::GrabCut::pub_background_
ros::Publisher pub_background_
Definition: grabcut.h:147
jsk_perception::GrabCut::segment
virtual void segment(const sensor_msgs::Image::ConstPtr &image_msg, const sensor_msgs::Image::ConstPtr &foreground_msg, const sensor_msgs::Image::ConstPtr &background_msg)
Definition: grabcut_nodelet.cpp:132
jsk_perception::GrabCut::pub_foreground_
ros::Publisher pub_foreground_
Definition: grabcut.h:146
NODELET_WARN
#define NODELET_WARN(...)
jsk_perception::GrabCut::mutex_
boost::mutex mutex_
Definition: grabcut.h:155
jsk_perception::GrabCut::unsubscribe
virtual void unsubscribe()
Definition: grabcut_nodelet.cpp:116
grabcut.h
jsk_perception::GrabCut::pub_foreground_mask_
ros::Publisher pub_foreground_mask_
Definition: grabcut.h:148
ros::Publisher::publish
void publish(const boost::shared_ptr< M > &message) const
message_filters::Subscriber::unsubscribe
void unsubscribe()
PLUGINLIB_EXPORT_CLASS
PLUGINLIB_EXPORT_CLASS(jsk_perception::GrabCut, nodelet::Nodelet)
input
char input[LINE_MAX_LEN]
class_list_macros.h
jsk_perception
Definition: add_mask_image.h:48
jsk_perception::GrabCut::sub_foreground_
message_filters::Subscriber< sensor_msgs::Image > sub_foreground_
Definition: grabcut.h:152
diagnostic_updater::DiagnosticStatusWrapper::summary
void summary(const diagnostic_msgs::DiagnosticStatus &src)
jsk_perception::GrabCut::sub_image_
message_filters::Subscriber< sensor_msgs::Image > sub_image_
Definition: grabcut.h:151
jsk_perception::GrabCut::updateDiagnostic
virtual void updateDiagnostic(diagnostic_updater::DiagnosticStatusWrapper &stat)
Definition: grabcut_nodelet.cpp:123
cv_bridge::toCvCopy
CvImagePtr toCvCopy(const sensor_msgs::CompressedImage &source, const std::string &encoding=std::string())
jsk_perception::GrabCut::srv_
boost::shared_ptr< dynamic_reconfigure::Server< Config > > srv_
Definition: grabcut.h:154
lock
mutex_t * lock
jsk_perception::GrabCut
Definition: grabcut.h:84
jsk_perception::GrabCut::onInit
virtual void onInit()
Definition: grabcut_nodelet.cpp:75
jsk_perception::GrabCut::use_probable_pixel_seed_
bool use_probable_pixel_seed_
Definition: grabcut.h:160
jsk_perception::GrabCut::pub_background_mask_
ros::Publisher pub_background_mask_
Definition: grabcut.h:149
jsk_perception::GrabCut::sub_background_
message_filters::Subscriber< sensor_msgs::Image > sub_background_
Definition: grabcut.h:153
message_filters::Subscriber::subscribe
void subscribe()
f
f
nodelet::Nodelet
jsk_perception::GrabCut::~GrabCut
virtual ~GrabCut()
Definition: grabcut_nodelet.cpp:94
jsk_perception::GrabCut::sync_
boost::shared_ptr< message_filters::Synchronizer< SyncPolicy > > sync_
Definition: grabcut.h:150
sensor_msgs::image_encodings::MONO8
const std::string MONO8
cv_bridge.h
cv_bridge::CvImage
diagnostic_updater::DiagnosticStatusWrapper
sensor_msgs::image_encodings::BGR8
const std::string BGR8
cv_bridge::toCvShare
CvImageConstPtr toCvShare(const sensor_msgs::Image &source, const boost::shared_ptr< void const > &tracked_object, const std::string &encoding=std::string())
jsk_perception::GrabCut::subscribe
virtual void subscribe()
Definition: grabcut_nodelet.cpp:105
config
config
jsk_perception::GrabCut::configCallback
virtual void configCallback(Config &config, uint32_t level)
Definition: grabcut_nodelet.cpp:215


jsk_perception
Author(s): Manabu Saito, Ryohei Ueda
autogenerated on Fri May 16 2025 03:11:17