6_synchronous_custom_postprocessing.cpp 10.6 KB
Newer Older
1
// ------------------------- OpenPose C++ API Tutorial - Example 6 - XXXXXXXXXXXXX -------------------------
G
gineshidalgo99 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14 15
// If the user wants to learn to use the OpenPose library, we highly recommend to start with the
// examples in `examples/tutorial_api_cpp/`.
// This example summarizes all the functionality of the OpenPose library:
    // 1. Read folder of images / video / webcam  (`producer` module)
    // 2. Extract and render body keypoint / heatmap / PAF of that image (`pose` module)
    // 3. Extract and render face keypoint / heatmap / PAF of that image (`face` module)
    // 4. Save the results on disk (`filestream` module)
    // 5. Display the rendered pose (`gui` module)
    // Everything in a multi-thread scenario (`thread` module)
    // Points 2 to 5 are included in the `wrapper` module
// In addition to the previous OpenPose modules, we also need to use:
    // 1. `core` module:
        // For the Array<float> class that the `pose` module needs
        // For the Datum struct that the `thread` module sends between the queues
G
gineshidalgo99 已提交
16
    // 2. `utilities` module: for the error & logging functions, i.e., op::error & op::log respectively
G
gineshidalgo99 已提交
17 18 19 20 21 22 23
// This file should only be used for the user to take specific examples.

// Command-line user intraface
#include <openpose/flags.hpp>
// OpenPose dependencies
#include <openpose/headers.hpp>

24
// If the user needs his own variables, he can inherit the op::Datum struct and add them in there.
G
gineshidalgo99 已提交
25
// UserDatum can be directly used by the OpenPose wrapper because it inherits from op::Datum, just define
G
gineshidalgo99 已提交
26 27
// WrapperT<std::vector<std::shared_ptr<UserDatum>>> instead of Wrapper
// (or equivalently WrapperT<std::vector<std::shared_ptr<UserDatum>>>)
G
gineshidalgo99 已提交
28 29 30 31 32 33 34 35 36 37 38 39 40 41
struct UserDatum : public op::Datum
{
    bool boolThatUserNeedsForSomeReason;

    UserDatum(const bool boolThatUserNeedsForSomeReason_ = false) :
        boolThatUserNeedsForSomeReason{boolThatUserNeedsForSomeReason_}
    {}
};

// The W-classes can be implemented either as a template or as simple classes given
// that the user usually knows which kind of data he will move between the queues,
// in this case we assume a std::shared_ptr of a std::vector of UserDatum

// This worker will just invert the image
G
gineshidalgo99 已提交
42
class WUserPostProcessing : public op::Worker<std::shared_ptr<std::vector<std::shared_ptr<UserDatum>>>>
G
gineshidalgo99 已提交
43 44 45 46 47 48 49 50 51
{
public:
    WUserPostProcessing()
    {
        // User's constructor here
    }

    void initializationOnThread() {}

G
gineshidalgo99 已提交
52
    void work(std::shared_ptr<std::vector<std::shared_ptr<UserDatum>>>& datumsPtr)
G
gineshidalgo99 已提交
53 54
    {
        // User's post-processing (after OpenPose processing & before OpenPose outputs) here
G
gineshidalgo99 已提交
55 56
            // datumPtr->cvOutputData: rendered frame with pose or heatmaps
            // datumPtr->poseKeypoints: Array<float> with the estimated pose
G
gineshidalgo99 已提交
57 58 59
        try
        {
            if (datumsPtr != nullptr && !datumsPtr->empty())
G
gineshidalgo99 已提交
60 61
                for (auto& datumPtr : *datumsPtr)
                    cv::bitwise_not(datumPtr->cvOutputData, datumPtr->cvOutputData);
G
gineshidalgo99 已提交
62 63 64 65 66 67 68 69 70
        }
        catch (const std::exception& e)
        {
            this->stop();
            op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
        }
    }
};

G
gineshidalgo99 已提交
71
int tutorialApiCpp6()
G
gineshidalgo99 已提交
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
{
    try
    {
        op::log("Starting OpenPose demo...", op::Priority::High);
        const auto timerBegin = std::chrono::high_resolution_clock::now();

        // logging_level
        op::check(0 <= FLAGS_logging_level && FLAGS_logging_level <= 255, "Wrong logging_level value.",
                  __LINE__, __FUNCTION__, __FILE__);
        op::ConfigureLog::setPriorityThreshold((op::Priority)FLAGS_logging_level);
        op::Profiler::setDefaultX(FLAGS_profile_speed);
        // // For debugging
        // // Print all logging messages
        // op::ConfigureLog::setPriorityThreshold(op::Priority::None);
        // // Print out speed values faster
        // op::Profiler::setDefaultX(100);

        // Applying user defined configuration - GFlags to program variables
G
gineshidalgo99 已提交
90 91
        // cameraSize
        const auto cameraSize = op::flagsToPoint(FLAGS_camera_resolution, "-1x-1");
G
gineshidalgo99 已提交
92 93 94 95 96 97 98 99 100
        // outputSize
        const auto outputSize = op::flagsToPoint(FLAGS_output_resolution, "-1x-1");
        // netInputSize
        const auto netInputSize = op::flagsToPoint(FLAGS_net_resolution, "-1x368");
        // faceNetInputSize
        const auto faceNetInputSize = op::flagsToPoint(FLAGS_face_net_resolution, "368x368 (multiples of 16)");
        // handNetInputSize
        const auto handNetInputSize = op::flagsToPoint(FLAGS_hand_net_resolution, "368x368 (multiples of 16)");
        // producerType
G
gineshidalgo99 已提交
101 102 103 104
        op::ProducerType producerType;
        std::string producerString;
        std::tie(producerType, producerString) = op::flagsToProducer(
            FLAGS_image_dir, FLAGS_video, FLAGS_ip_camera, FLAGS_camera, FLAGS_flir_camera, FLAGS_flir_camera_index);
G
gineshidalgo99 已提交
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
        // poseModel
        const auto poseModel = op::flagsToPoseModel(FLAGS_model_pose);
        // JSON saving
        if (!FLAGS_write_keypoint.empty())
            op::log("Flag `write_keypoint` is deprecated and will eventually be removed."
                    " Please, use `write_json` instead.", op::Priority::Max);
        // keypointScale
        const auto keypointScale = op::flagsToScaleMode(FLAGS_keypoint_scale);
        // heatmaps to add
        const auto heatMapTypes = op::flagsToHeatMaps(FLAGS_heatmaps_add_parts, FLAGS_heatmaps_add_bkg,
                                                      FLAGS_heatmaps_add_PAFs);
        const auto heatMapScale = op::flagsToHeatMapScaleMode(FLAGS_heatmaps_scale);
        // >1 camera view?
        const auto multipleView = (FLAGS_3d || FLAGS_3d_views > 1 || FLAGS_flir_camera);
        // Enabling Google Logging
        const bool enableGoogleLogging = true;

        // OpenPose wrapper
G
gineshidalgo99 已提交
123
        op::log("Configuring OpenPose...", op::Priority::High);
G
gineshidalgo99 已提交
124
        op::WrapperT<UserDatum> opWrapperT;
G
gineshidalgo99 已提交
125 126 127 128 129 130

        // Initializing the user custom classes
        // Processing
        auto wUserPostProcessing = std::make_shared<WUserPostProcessing>();
        // Add custom processing
        const auto workerProcessingOnNewThread = true;
G
gineshidalgo99 已提交
131
        opWrapperT.setWorker(op::WorkerType::PostProcessing, wUserPostProcessing, workerProcessingOnNewThread);
G
gineshidalgo99 已提交
132 133 134 135 136 137 138

        // Pose configuration (use WrapperStructPose{} for default and recommended configuration)
        const op::WrapperStructPose wrapperStructPose{
            !FLAGS_body_disable, netInputSize, outputSize, keypointScale, FLAGS_num_gpu, FLAGS_num_gpu_start,
            FLAGS_scale_number, (float)FLAGS_scale_gap, op::flagsToRenderMode(FLAGS_render_pose, multipleView),
            poseModel, !FLAGS_disable_blending, (float)FLAGS_alpha_pose, (float)FLAGS_alpha_heatmap,
            FLAGS_part_to_show, FLAGS_model_folder, heatMapTypes, heatMapScale, FLAGS_part_candidates,
139
            (float)FLAGS_render_threshold, FLAGS_number_people_max, FLAGS_maximize_positives, FLAGS_fps_max,
140
            FLAGS_prototxt_path, FLAGS_caffemodel_path, enableGoogleLogging};
G
gineshidalgo99 已提交
141
        opWrapperT.configure(wrapperStructPose);
G
gineshidalgo99 已提交
142 143 144 145
        // Face configuration (use op::WrapperStructFace{} to disable it)
        const op::WrapperStructFace wrapperStructFace{
            FLAGS_face, faceNetInputSize, op::flagsToRenderMode(FLAGS_face_render, multipleView, FLAGS_render_pose),
            (float)FLAGS_face_alpha_pose, (float)FLAGS_face_alpha_heatmap, (float)FLAGS_face_render_threshold};
G
gineshidalgo99 已提交
146
        opWrapperT.configure(wrapperStructFace);
G
gineshidalgo99 已提交
147 148 149 150 151
        // Hand configuration (use op::WrapperStructHand{} to disable it)
        const op::WrapperStructHand wrapperStructHand{
            FLAGS_hand, handNetInputSize, FLAGS_hand_scale_number, (float)FLAGS_hand_scale_range, FLAGS_hand_tracking,
            op::flagsToRenderMode(FLAGS_hand_render, multipleView, FLAGS_render_pose), (float)FLAGS_hand_alpha_pose,
            (float)FLAGS_hand_alpha_heatmap, (float)FLAGS_hand_render_threshold};
G
gineshidalgo99 已提交
152 153 154 155 156
        opWrapperT.configure(wrapperStructHand);
        // Extra functionality configuration (use op::WrapperStructExtra{} to disable it)
        const op::WrapperStructExtra wrapperStructExtra{
            FLAGS_3d, FLAGS_3d_min_views, FLAGS_identification, FLAGS_tracking, FLAGS_ik_threads};
        opWrapperT.configure(wrapperStructExtra);
G
gineshidalgo99 已提交
157 158
        // Producer (use default to disable any input)
        const op::WrapperStructInput wrapperStructInput{
G
gineshidalgo99 已提交
159 160
            producerType, producerString, FLAGS_frame_first, FLAGS_frame_step, FLAGS_frame_last,
            FLAGS_process_real_time, FLAGS_frame_flip, FLAGS_frame_rotate, FLAGS_frames_repeat,
161
            cameraSize, FLAGS_camera_parameter_path, FLAGS_frame_undistort, FLAGS_3d_views};
G
gineshidalgo99 已提交
162
        opWrapperT.configure(wrapperStructInput);
163
        // Output (comment or use default argument to disable any output)
G
gineshidalgo99 已提交
164
        const op::WrapperStructOutput wrapperStructOutput{
G
gineshidalgo99 已提交
165 166
            FLAGS_cli_verbose, FLAGS_write_keypoint, op::stringToDataFormat(FLAGS_write_keypoint_format),
            FLAGS_write_json, FLAGS_write_coco_json, FLAGS_write_coco_foot_json, FLAGS_write_coco_json_variant,
167
            FLAGS_write_images, FLAGS_write_images_format, FLAGS_write_video, FLAGS_write_video_fps,
168 169
            FLAGS_write_heatmaps, FLAGS_write_heatmaps_format, FLAGS_write_video_3d, FLAGS_write_video_adam,
            FLAGS_write_bvh, FLAGS_udp_host, FLAGS_udp_port};
G
gineshidalgo99 已提交
170
        opWrapperT.configure(wrapperStructOutput);
171 172 173 174
        // GUI (comment or use default argument to disable any visual output)
        const op::WrapperStructGui wrapperStructGui{
            op::flagsToDisplayMode(FLAGS_display, FLAGS_3d), !FLAGS_no_gui_verbose, FLAGS_fullscreen};
        opWrapperT.configure(wrapperStructGui);
G
gineshidalgo99 已提交
175 176
        // Set to single-thread (for sequential processing and/or debugging and/or reducing latency)
        if (FLAGS_disable_multi_thread)
G
gineshidalgo99 已提交
177
            opWrapperT.disableMultiThreading();
G
gineshidalgo99 已提交
178

G
gineshidalgo99 已提交
179
        // Start, run, and stop processing - exec() blocks this thread until OpenPose wrapper has finished
G
gineshidalgo99 已提交
180
        op::log("Starting thread(s)...", op::Priority::High);
G
gineshidalgo99 已提交
181
        opWrapperT.exec();
G
gineshidalgo99 已提交
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204

        // Measuring total time
        const auto now = std::chrono::high_resolution_clock::now();
        const auto totalTimeSec = (double)std::chrono::duration_cast<std::chrono::nanoseconds>(now-timerBegin).count()
                                * 1e-9;
        const auto message = "OpenPose demo successfully finished. Total time: "
                           + std::to_string(totalTimeSec) + " seconds.";
        op::log(message, op::Priority::High);

        // Return successful message
        return 0;
    }
    catch (const std::exception& e)
    {
        return -1;
    }
}

int main(int argc, char *argv[])
{
    // Parsing command line flags
    gflags::ParseCommandLineFlags(&argc, &argv, true);

G
gineshidalgo99 已提交
205 206
    // Running tutorialApiCpp6
    return tutorialApiCpp6();
G
gineshidalgo99 已提交
207
}