[lldb][NFC] Fix all formatting errors in .cpp file headers
Summary:
A *.cpp file header in LLDB (and in LLDB) should like this:
```
//===-- TestUtilities.cpp -------------------------------------------------===//
```
However in LLDB most of our source files have arbitrary changes to this format and
these changes are spreading through LLDB as folks usually just use the existing
source files as templates for their new files (most notably the unnecessary
editor language indicator `-*- C++ -*-` is spreading and in every review
someone is pointing out that this is wrong, resulting in people pointing out that this
is done in the same way in other files).
This patch removes most of these inconsistencies including the editor language indicators,
all the different missing/additional '-' characters, files that center the file name, missing
trailing `===//` (mostly caused by clang-format breaking the line).
Reviewers: aprantl, espindola, jfb, shafik, JDevlieghere
Reviewed By: JDevlieghere
Subscribers: dexonsmith, wuzish, emaste, sdardis, nemanjai, kbarton, MaskRay, atanasyan, arphaman, jfb, abidh, jsji, JDevlieghere, usaxena95, lldb-commits
Tags: #lldb
Differential Revision: https://reviews.llvm.org/D73258
2020-01-24 15:23:27 +08:00
|
|
|
//===-- BreakpointOptions.cpp ---------------------------------------------===//
|
2010-06-09 00:52:24 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2010-06-09 00:52:24 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-10-31 02:50:12 +08:00
|
|
|
#include "lldb/Breakpoint/BreakpointOptions.h"
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Breakpoint/StoppointCallbackContext.h"
|
2010-10-15 07:45:03 +08:00
|
|
|
#include "lldb/Core/Value.h"
|
2016-09-13 07:10:56 +08:00
|
|
|
#include "lldb/Interpreter/CommandInterpreter.h"
|
|
|
|
#include "lldb/Interpreter/CommandReturnObject.h"
|
2010-10-15 07:45:03 +08:00
|
|
|
#include "lldb/Target/Process.h"
|
2010-11-19 10:52:21 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
2010-06-16 10:00:15 +08:00
|
|
|
#include "lldb/Target/ThreadSpec.h"
|
2017-02-03 05:39:50 +08:00
|
|
|
#include "lldb/Utility/Stream.h"
|
2017-03-22 02:25:04 +08:00
|
|
|
#include "lldb/Utility/StringList.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-14 01:53:38 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
2016-09-15 09:47:22 +08:00
|
|
|
const char
|
|
|
|
*BreakpointOptions::CommandData::g_option_names[static_cast<uint32_t>(
|
|
|
|
BreakpointOptions::CommandData::OptionNames::LastOptionName)]{
|
2016-09-13 07:10:56 +08:00
|
|
|
"UserSource", "ScriptSource", "StopOnError"};
|
|
|
|
|
|
|
|
StructuredData::ObjectSP
|
|
|
|
BreakpointOptions::CommandData::SerializeToStructuredData() {
|
|
|
|
size_t num_strings = user_source.GetSize();
|
|
|
|
if (num_strings == 0 && script_source.empty()) {
|
|
|
|
// We shouldn't serialize commands if there aren't any, return an empty sp
|
|
|
|
// to indicate this.
|
|
|
|
return StructuredData::ObjectSP();
|
|
|
|
}
|
|
|
|
|
|
|
|
StructuredData::DictionarySP options_dict_sp(
|
|
|
|
new StructuredData::Dictionary());
|
|
|
|
options_dict_sp->AddBooleanItem(GetKey(OptionNames::StopOnError),
|
|
|
|
stop_on_error);
|
|
|
|
|
|
|
|
StructuredData::ArraySP user_source_sp(new StructuredData::Array());
|
2016-09-13 09:58:08 +08:00
|
|
|
for (size_t i = 0; i < num_strings; i++) {
|
|
|
|
StructuredData::StringSP item_sp(
|
|
|
|
new StructuredData::String(user_source[i]));
|
|
|
|
user_source_sp->AddItem(item_sp);
|
|
|
|
options_dict_sp->AddItem(GetKey(OptionNames::UserSource), user_source_sp);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
2016-09-27 03:47:37 +08:00
|
|
|
options_dict_sp->AddStringItem(
|
|
|
|
GetKey(OptionNames::Interpreter),
|
|
|
|
ScriptInterpreter::LanguageToString(interpreter));
|
2016-09-13 07:10:56 +08:00
|
|
|
return options_dict_sp;
|
|
|
|
}
|
|
|
|
|
2016-09-13 09:58:08 +08:00
|
|
|
std::unique_ptr<BreakpointOptions::CommandData>
|
2016-09-13 07:10:56 +08:00
|
|
|
BreakpointOptions::CommandData::CreateFromStructuredData(
|
2017-05-12 12:51:55 +08:00
|
|
|
const StructuredData::Dictionary &options_dict, Status &error) {
|
2016-09-21 06:54:49 +08:00
|
|
|
std::unique_ptr<CommandData> data_up(new CommandData());
|
|
|
|
bool found_something = false;
|
|
|
|
|
2016-09-13 07:10:56 +08:00
|
|
|
bool success = options_dict.GetValueForKeyAsBoolean(
|
2016-09-21 06:54:49 +08:00
|
|
|
GetKey(OptionNames::StopOnError), data_up->stop_on_error);
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
found_something = true;
|
2016-09-13 07:10:56 +08:00
|
|
|
|
2017-05-12 13:49:54 +08:00
|
|
|
llvm::StringRef interpreter_str;
|
2016-09-27 03:47:37 +08:00
|
|
|
ScriptLanguage interp_language;
|
2016-09-13 07:10:56 +08:00
|
|
|
success = options_dict.GetValueForKeyAsString(
|
2016-09-27 03:47:37 +08:00
|
|
|
GetKey(OptionNames::Interpreter), interpreter_str);
|
2016-09-21 06:54:49 +08:00
|
|
|
|
2016-09-27 03:47:37 +08:00
|
|
|
if (!success) {
|
|
|
|
error.SetErrorString("Missing command language value.");
|
|
|
|
return data_up;
|
|
|
|
}
|
|
|
|
|
|
|
|
found_something = true;
|
|
|
|
interp_language = ScriptInterpreter::StringToLanguage(interpreter_str);
|
|
|
|
if (interp_language == eScriptLanguageUnknown) {
|
2016-12-16 12:27:00 +08:00
|
|
|
error.SetErrorStringWithFormatv("Unknown breakpoint command language: {0}.",
|
|
|
|
interpreter_str);
|
2016-09-27 03:47:37 +08:00
|
|
|
return data_up;
|
|
|
|
}
|
|
|
|
data_up->interpreter = interp_language;
|
2016-09-13 07:10:56 +08:00
|
|
|
|
|
|
|
StructuredData::Array *user_source;
|
|
|
|
success = options_dict.GetValueForKeyAsArray(GetKey(OptionNames::UserSource),
|
|
|
|
user_source);
|
|
|
|
if (success) {
|
2016-09-21 06:54:49 +08:00
|
|
|
found_something = true;
|
2016-09-13 07:10:56 +08:00
|
|
|
size_t num_elems = user_source->GetSize();
|
|
|
|
for (size_t i = 0; i < num_elems; i++) {
|
2017-05-12 13:49:54 +08:00
|
|
|
llvm::StringRef elem_string;
|
2016-09-13 07:10:56 +08:00
|
|
|
success = user_source->GetItemAtIndexAsString(i, elem_string);
|
|
|
|
if (success)
|
2016-09-21 06:54:49 +08:00
|
|
|
data_up->user_source.AppendString(elem_string);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
}
|
2016-09-21 06:54:49 +08:00
|
|
|
|
|
|
|
if (found_something)
|
|
|
|
return data_up;
|
|
|
|
else
|
|
|
|
return std::unique_ptr<BreakpointOptions::CommandData>();
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
2016-09-23 06:00:59 +08:00
|
|
|
const char *BreakpointOptions::g_option_names[(
|
|
|
|
size_t)BreakpointOptions::OptionNames::LastOptionName]{
|
2017-08-04 02:13:24 +08:00
|
|
|
"ConditionText", "IgnoreCount",
|
|
|
|
"EnabledState", "OneShotState", "AutoContinue"};
|
2016-09-13 07:10:56 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
bool BreakpointOptions::NullCallback(void *baton,
|
|
|
|
StoppointCallbackContext *context,
|
|
|
|
lldb::user_id_t break_id,
|
|
|
|
lldb::user_id_t break_loc_id) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// BreakpointOptions constructor
|
2017-08-02 08:16:10 +08:00
|
|
|
BreakpointOptions::BreakpointOptions(bool all_flags_set)
|
2010-06-09 00:52:24 +08:00
|
|
|
: m_callback(BreakpointOptions::NullCallback), m_callback_baton_sp(),
|
2016-09-13 07:10:56 +08:00
|
|
|
m_baton_is_command_baton(false), m_callback_is_synchronous(false),
|
2019-02-13 14:25:41 +08:00
|
|
|
m_enabled(true), m_one_shot(false), m_ignore_count(0), m_thread_spec_up(),
|
2017-08-04 02:13:24 +08:00
|
|
|
m_condition_text(), m_condition_text_hash(0), m_auto_continue(false),
|
2017-09-15 04:22:49 +08:00
|
|
|
m_set_flags(0) {
|
2019-02-13 14:25:41 +08:00
|
|
|
if (all_flags_set)
|
|
|
|
m_set_flags.Set(~((Flags::ValueType)0));
|
|
|
|
}
|
2016-09-13 07:10:56 +08:00
|
|
|
|
|
|
|
BreakpointOptions::BreakpointOptions(const char *condition, bool enabled,
|
2017-08-04 02:13:24 +08:00
|
|
|
int32_t ignore, bool one_shot,
|
|
|
|
bool auto_continue)
|
2016-09-13 07:10:56 +08:00
|
|
|
: m_callback(nullptr), m_baton_is_command_baton(false),
|
|
|
|
m_callback_is_synchronous(false), m_enabled(enabled),
|
2017-09-15 04:22:49 +08:00
|
|
|
m_one_shot(one_shot), m_ignore_count(ignore),
|
2017-08-04 02:13:24 +08:00
|
|
|
m_condition_text_hash(0), m_auto_continue(auto_continue)
|
2017-08-02 08:16:10 +08:00
|
|
|
{
|
2017-08-04 02:13:24 +08:00
|
|
|
m_set_flags.Set(eEnabled | eIgnoreCount | eOneShot
|
2017-09-15 04:22:49 +08:00
|
|
|
| eAutoContinue);
|
|
|
|
if (condition && *condition != '\0') {
|
|
|
|
SetCondition(condition);
|
|
|
|
}
|
2017-08-02 08:16:10 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
// BreakpointOptions copy constructor
|
|
|
|
BreakpointOptions::BreakpointOptions(const BreakpointOptions &rhs)
|
|
|
|
: m_callback(rhs.m_callback), m_callback_baton_sp(rhs.m_callback_baton_sp),
|
2016-09-13 07:10:56 +08:00
|
|
|
m_baton_is_command_baton(rhs.m_baton_is_command_baton),
|
2010-06-09 00:52:24 +08:00
|
|
|
m_callback_is_synchronous(rhs.m_callback_is_synchronous),
|
2012-10-06 03:16:31 +08:00
|
|
|
m_enabled(rhs.m_enabled), m_one_shot(rhs.m_one_shot),
|
2019-02-13 14:25:41 +08:00
|
|
|
m_ignore_count(rhs.m_ignore_count), m_thread_spec_up(),
|
|
|
|
m_auto_continue(rhs.m_auto_continue), m_set_flags(rhs.m_set_flags) {
|
|
|
|
if (rhs.m_thread_spec_up != nullptr)
|
2020-06-25 08:44:33 +08:00
|
|
|
m_thread_spec_up = std::make_unique<ThreadSpec>(*rhs.m_thread_spec_up);
|
2013-04-19 15:09:15 +08:00
|
|
|
m_condition_text = rhs.m_condition_text;
|
|
|
|
m_condition_text_hash = rhs.m_condition_text_hash;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// BreakpointOptions assignment operator
|
|
|
|
const BreakpointOptions &BreakpointOptions::
|
|
|
|
operator=(const BreakpointOptions &rhs) {
|
|
|
|
m_callback = rhs.m_callback;
|
|
|
|
m_callback_baton_sp = rhs.m_callback_baton_sp;
|
2016-09-13 07:10:56 +08:00
|
|
|
m_baton_is_command_baton = rhs.m_baton_is_command_baton;
|
2010-06-09 00:52:24 +08:00
|
|
|
m_callback_is_synchronous = rhs.m_callback_is_synchronous;
|
|
|
|
m_enabled = rhs.m_enabled;
|
2012-10-06 03:16:31 +08:00
|
|
|
m_one_shot = rhs.m_one_shot;
|
2010-06-09 00:52:24 +08:00
|
|
|
m_ignore_count = rhs.m_ignore_count;
|
2019-02-13 14:25:41 +08:00
|
|
|
if (rhs.m_thread_spec_up != nullptr)
|
2020-06-25 08:44:33 +08:00
|
|
|
m_thread_spec_up = std::make_unique<ThreadSpec>(*rhs.m_thread_spec_up);
|
2013-04-19 15:09:15 +08:00
|
|
|
m_condition_text = rhs.m_condition_text;
|
|
|
|
m_condition_text_hash = rhs.m_condition_text_hash;
|
2017-08-04 02:13:24 +08:00
|
|
|
m_auto_continue = rhs.m_auto_continue;
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags = rhs.m_set_flags;
|
2010-06-09 00:52:24 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2017-09-15 04:22:49 +08:00
|
|
|
void BreakpointOptions::CopyOverSetOptions(const BreakpointOptions &incoming)
|
|
|
|
{
|
|
|
|
if (incoming.m_set_flags.Test(eEnabled))
|
|
|
|
{
|
|
|
|
m_enabled = incoming.m_enabled;
|
|
|
|
m_set_flags.Set(eEnabled);
|
|
|
|
}
|
|
|
|
if (incoming.m_set_flags.Test(eOneShot))
|
|
|
|
{
|
|
|
|
m_one_shot = incoming.m_one_shot;
|
|
|
|
m_set_flags.Set(eOneShot);
|
|
|
|
}
|
|
|
|
if (incoming.m_set_flags.Test(eCallback))
|
|
|
|
{
|
|
|
|
m_callback = incoming.m_callback;
|
|
|
|
m_callback_baton_sp = incoming.m_callback_baton_sp;
|
|
|
|
m_callback_is_synchronous = incoming.m_callback_is_synchronous;
|
|
|
|
m_baton_is_command_baton = incoming.m_baton_is_command_baton;
|
|
|
|
m_set_flags.Set(eCallback);
|
|
|
|
}
|
|
|
|
if (incoming.m_set_flags.Test(eIgnoreCount))
|
|
|
|
{
|
|
|
|
m_ignore_count = incoming.m_ignore_count;
|
|
|
|
m_set_flags.Set(eIgnoreCount);
|
|
|
|
}
|
|
|
|
if (incoming.m_set_flags.Test(eCondition))
|
|
|
|
{
|
|
|
|
// If we're copying over an empty condition, mark it as unset.
|
|
|
|
if (incoming.m_condition_text.empty()) {
|
|
|
|
m_condition_text.clear();
|
|
|
|
m_condition_text_hash = 0;
|
|
|
|
m_set_flags.Clear(eCondition);
|
|
|
|
} else {
|
|
|
|
m_condition_text = incoming.m_condition_text;
|
|
|
|
m_condition_text_hash = incoming.m_condition_text_hash;
|
|
|
|
m_set_flags.Set(eCondition);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (incoming.m_set_flags.Test(eAutoContinue))
|
|
|
|
{
|
|
|
|
m_auto_continue = incoming.m_auto_continue;
|
|
|
|
m_set_flags.Set(eAutoContinue);
|
|
|
|
}
|
2019-02-13 14:25:41 +08:00
|
|
|
if (incoming.m_set_flags.Test(eThreadSpec) && incoming.m_thread_spec_up) {
|
|
|
|
if (!m_thread_spec_up)
|
2020-06-25 08:44:33 +08:00
|
|
|
m_thread_spec_up =
|
|
|
|
std::make_unique<ThreadSpec>(*incoming.m_thread_spec_up);
|
2017-09-15 04:22:49 +08:00
|
|
|
else
|
2019-02-13 14:25:41 +08:00
|
|
|
*m_thread_spec_up = *incoming.m_thread_spec_up;
|
2017-09-15 04:22:49 +08:00
|
|
|
m_set_flags.Set(eThreadSpec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// Destructor
|
2015-10-31 02:50:12 +08:00
|
|
|
BreakpointOptions::~BreakpointOptions() = default;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-13 09:58:08 +08:00
|
|
|
std::unique_ptr<BreakpointOptions> BreakpointOptions::CreateFromStructuredData(
|
2016-09-27 03:47:37 +08:00
|
|
|
Target &target, const StructuredData::Dictionary &options_dict,
|
2017-05-12 12:51:55 +08:00
|
|
|
Status &error) {
|
2016-09-13 07:10:56 +08:00
|
|
|
bool enabled = true;
|
|
|
|
bool one_shot = false;
|
2017-08-04 02:13:24 +08:00
|
|
|
bool auto_continue = false;
|
2016-09-13 07:10:56 +08:00
|
|
|
int32_t ignore_count = 0;
|
2017-08-02 08:16:10 +08:00
|
|
|
llvm::StringRef condition_ref("");
|
|
|
|
Flags set_options;
|
|
|
|
|
|
|
|
const char *key = GetKey(OptionNames::EnabledState);
|
|
|
|
bool success;
|
2019-02-23 07:54:11 +08:00
|
|
|
if (key && options_dict.HasKey(key)) {
|
2017-08-02 08:16:10 +08:00
|
|
|
success = options_dict.GetValueForKeyAsBoolean(key, enabled);
|
|
|
|
if (!success) {
|
2019-02-23 07:54:11 +08:00
|
|
|
error.SetErrorStringWithFormat("%s key is not a boolean.", key);
|
2017-08-02 08:16:10 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
set_options.Set(eEnabled);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
2017-08-02 08:16:10 +08:00
|
|
|
key = GetKey(OptionNames::OneShotState);
|
2019-02-23 07:54:11 +08:00
|
|
|
if (key && options_dict.HasKey(key)) {
|
2017-08-02 08:16:10 +08:00
|
|
|
success = options_dict.GetValueForKeyAsBoolean(key, one_shot);
|
|
|
|
if (!success) {
|
2019-02-23 07:54:11 +08:00
|
|
|
error.SetErrorStringWithFormat("%s key is not a boolean.", key);
|
2017-08-02 08:16:10 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
set_options.Set(eOneShot);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
2017-08-02 08:16:10 +08:00
|
|
|
|
2017-08-04 02:13:24 +08:00
|
|
|
key = GetKey(OptionNames::AutoContinue);
|
2019-02-23 07:54:11 +08:00
|
|
|
if (key && options_dict.HasKey(key)) {
|
2017-08-04 02:13:24 +08:00
|
|
|
success = options_dict.GetValueForKeyAsBoolean(key, auto_continue);
|
|
|
|
if (!success) {
|
2019-02-23 07:54:11 +08:00
|
|
|
error.SetErrorStringWithFormat("%s key is not a boolean.", key);
|
2017-08-04 02:13:24 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
set_options.Set(eAutoContinue);
|
|
|
|
}
|
|
|
|
|
2017-08-02 08:16:10 +08:00
|
|
|
key = GetKey(OptionNames::IgnoreCount);
|
2019-02-23 07:54:11 +08:00
|
|
|
if (key && options_dict.HasKey(key)) {
|
2017-08-02 08:16:10 +08:00
|
|
|
success = options_dict.GetValueForKeyAsInteger(key, ignore_count);
|
|
|
|
if (!success) {
|
2019-02-23 07:54:11 +08:00
|
|
|
error.SetErrorStringWithFormat("%s key is not an integer.", key);
|
2017-08-02 08:16:10 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
set_options.Set(eIgnoreCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
key = GetKey(OptionNames::ConditionText);
|
2019-02-23 07:54:11 +08:00
|
|
|
if (key && options_dict.HasKey(key)) {
|
2017-08-02 08:16:10 +08:00
|
|
|
success = options_dict.GetValueForKeyAsString(key, condition_ref);
|
|
|
|
if (!success) {
|
2019-02-23 07:54:11 +08:00
|
|
|
error.SetErrorStringWithFormat("%s key is not an string.", key);
|
2017-08-02 08:16:10 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
set_options.Set(eCondition);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
2016-09-21 06:54:49 +08:00
|
|
|
std::unique_ptr<CommandData> cmd_data_up;
|
2016-09-13 07:10:56 +08:00
|
|
|
StructuredData::Dictionary *cmds_dict;
|
|
|
|
success = options_dict.GetValueForKeyAsDictionary(
|
|
|
|
CommandData::GetSerializationKey(), cmds_dict);
|
|
|
|
if (success && cmds_dict) {
|
2017-05-12 12:51:55 +08:00
|
|
|
Status cmds_error;
|
2016-09-21 06:54:49 +08:00
|
|
|
cmd_data_up = CommandData::CreateFromStructuredData(*cmds_dict, cmds_error);
|
2016-09-13 07:10:56 +08:00
|
|
|
if (cmds_error.Fail()) {
|
|
|
|
error.SetErrorStringWithFormat(
|
|
|
|
"Failed to deserialize breakpoint command options: %s.",
|
|
|
|
cmds_error.AsCString());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-15 06:19:23 +08:00
|
|
|
auto bp_options = std::make_unique<BreakpointOptions>(
|
2017-08-04 02:13:24 +08:00
|
|
|
condition_ref.str().c_str(), enabled,
|
|
|
|
ignore_count, one_shot, auto_continue);
|
2019-02-12 11:47:39 +08:00
|
|
|
if (cmd_data_up) {
|
2016-09-27 03:47:37 +08:00
|
|
|
if (cmd_data_up->interpreter == eScriptLanguageNone)
|
|
|
|
bp_options->SetCommandDataCallback(cmd_data_up);
|
|
|
|
else {
|
2019-04-27 06:43:16 +08:00
|
|
|
ScriptInterpreter *interp = target.GetDebugger().GetScriptInterpreter();
|
2016-09-27 03:47:37 +08:00
|
|
|
if (!interp) {
|
2020-07-21 13:57:06 +08:00
|
|
|
error.SetErrorString(
|
2016-09-27 03:47:37 +08:00
|
|
|
"Can't set script commands - no script interpreter");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (interp->GetLanguage() != cmd_data_up->interpreter) {
|
|
|
|
error.SetErrorStringWithFormat(
|
|
|
|
"Current script language doesn't match breakpoint's language: %s",
|
|
|
|
ScriptInterpreter::LanguageToString(cmd_data_up->interpreter)
|
|
|
|
.c_str());
|
|
|
|
return nullptr;
|
|
|
|
}
|
2017-05-12 12:51:55 +08:00
|
|
|
Status script_error;
|
2016-09-27 03:47:37 +08:00
|
|
|
script_error =
|
2021-06-12 08:00:46 +08:00
|
|
|
interp->SetBreakpointCommandCallback(*bp_options, cmd_data_up);
|
2016-09-27 03:47:37 +08:00
|
|
|
if (script_error.Fail()) {
|
|
|
|
error.SetErrorStringWithFormat("Error generating script callback: %s.",
|
|
|
|
error.AsCString());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-09-23 06:00:59 +08:00
|
|
|
|
|
|
|
StructuredData::Dictionary *thread_spec_dict;
|
|
|
|
success = options_dict.GetValueForKeyAsDictionary(
|
|
|
|
ThreadSpec::GetSerializationKey(), thread_spec_dict);
|
|
|
|
if (success) {
|
2017-05-12 12:51:55 +08:00
|
|
|
Status thread_spec_error;
|
2016-09-23 06:00:59 +08:00
|
|
|
std::unique_ptr<ThreadSpec> thread_spec_up =
|
|
|
|
ThreadSpec::CreateFromStructuredData(*thread_spec_dict,
|
|
|
|
thread_spec_error);
|
|
|
|
if (thread_spec_error.Fail()) {
|
|
|
|
error.SetErrorStringWithFormat(
|
|
|
|
"Failed to deserialize breakpoint thread spec options: %s.",
|
|
|
|
thread_spec_error.AsCString());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
bp_options->SetThreadSpec(thread_spec_up);
|
|
|
|
}
|
2016-09-14 01:53:38 +08:00
|
|
|
return bp_options;
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
StructuredData::ObjectSP BreakpointOptions::SerializeToStructuredData() {
|
|
|
|
StructuredData::DictionarySP options_dict_sp(
|
|
|
|
new StructuredData::Dictionary());
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eEnabled))
|
2017-08-02 08:16:10 +08:00
|
|
|
options_dict_sp->AddBooleanItem(GetKey(OptionNames::EnabledState),
|
|
|
|
m_enabled);
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eOneShot))
|
2017-08-02 08:16:10 +08:00
|
|
|
options_dict_sp->AddBooleanItem(GetKey(OptionNames::OneShotState),
|
|
|
|
m_one_shot);
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eAutoContinue))
|
2017-08-04 02:13:24 +08:00
|
|
|
options_dict_sp->AddBooleanItem(GetKey(OptionNames::AutoContinue),
|
|
|
|
m_auto_continue);
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eIgnoreCount))
|
2017-08-02 08:16:10 +08:00
|
|
|
options_dict_sp->AddIntegerItem(GetKey(OptionNames::IgnoreCount),
|
|
|
|
m_ignore_count);
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eCondition))
|
2017-08-02 08:16:10 +08:00
|
|
|
options_dict_sp->AddStringItem(GetKey(OptionNames::ConditionText),
|
|
|
|
m_condition_text);
|
|
|
|
|
2017-09-15 04:22:49 +08:00
|
|
|
if (m_set_flags.Test(eCallback) && m_baton_is_command_baton) {
|
2016-09-14 01:53:38 +08:00
|
|
|
auto cmd_baton =
|
|
|
|
std::static_pointer_cast<CommandBaton>(m_callback_baton_sp);
|
2016-09-13 07:10:56 +08:00
|
|
|
StructuredData::ObjectSP commands_sp =
|
2016-09-14 01:53:38 +08:00
|
|
|
cmd_baton->getItem()->SerializeToStructuredData();
|
2016-09-13 07:10:56 +08:00
|
|
|
if (commands_sp) {
|
|
|
|
options_dict_sp->AddItem(
|
|
|
|
BreakpointOptions::CommandData::GetSerializationKey(), commands_sp);
|
|
|
|
}
|
|
|
|
}
|
2019-02-13 14:25:41 +08:00
|
|
|
if (m_set_flags.Test(eThreadSpec) && m_thread_spec_up) {
|
2016-09-23 06:00:59 +08:00
|
|
|
StructuredData::ObjectSP thread_spec_sp =
|
2019-02-13 14:25:41 +08:00
|
|
|
m_thread_spec_up->SerializeToStructuredData();
|
2016-09-23 06:00:59 +08:00
|
|
|
options_dict_sp->AddItem(ThreadSpec::GetSerializationKey(), thread_spec_sp);
|
|
|
|
}
|
|
|
|
|
2016-09-13 07:10:56 +08:00
|
|
|
return options_dict_sp;
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// Callbacks
|
|
|
|
void BreakpointOptions::SetCallback(BreakpointHitCallback callback,
|
2016-09-13 07:10:56 +08:00
|
|
|
const lldb::BatonSP &callback_baton_sp,
|
2010-06-09 00:52:24 +08:00
|
|
|
bool callback_is_synchronous) {
|
2016-09-14 01:53:38 +08:00
|
|
|
// FIXME: This seems unsafe. If BatonSP actually *is* a CommandBaton, but
|
2018-05-01 00:49:04 +08:00
|
|
|
// in a shared_ptr<Baton> instead of a shared_ptr<CommandBaton>, then we will
|
|
|
|
// set m_baton_is_command_baton to false, which is incorrect. One possible
|
|
|
|
// solution is to make the base Baton class provide a method such as:
|
2016-09-14 01:53:38 +08:00
|
|
|
// virtual StringRef getBatonId() const { return ""; }
|
|
|
|
// and have CommandBaton override this to return something unique, and then
|
|
|
|
// check for it here. Another option might be to make Baton using the llvm
|
|
|
|
// casting infrastructure, so that we could write something like:
|
|
|
|
// if (llvm::isa<CommandBaton>(callback_baton_sp))
|
|
|
|
// at relevant callsites instead of storing a boolean.
|
2010-06-09 00:52:24 +08:00
|
|
|
m_callback_is_synchronous = callback_is_synchronous;
|
|
|
|
m_callback = callback;
|
|
|
|
m_callback_baton_sp = callback_baton_sp;
|
2016-09-13 07:10:56 +08:00
|
|
|
m_baton_is_command_baton = false;
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Set(eCallback);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void BreakpointOptions::SetCallback(
|
|
|
|
BreakpointHitCallback callback,
|
|
|
|
const BreakpointOptions::CommandBatonSP &callback_baton_sp,
|
|
|
|
bool callback_is_synchronous) {
|
|
|
|
m_callback_is_synchronous = callback_is_synchronous;
|
|
|
|
m_callback = callback;
|
|
|
|
m_callback_baton_sp = callback_baton_sp;
|
|
|
|
m_baton_is_command_baton = true;
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Set(eCallback);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void BreakpointOptions::ClearCallback() {
|
2010-10-15 07:45:03 +08:00
|
|
|
m_callback = BreakpointOptions::NullCallback;
|
|
|
|
m_callback_is_synchronous = false;
|
2010-06-09 00:52:24 +08:00
|
|
|
m_callback_baton_sp.reset();
|
2016-09-13 07:10:56 +08:00
|
|
|
m_baton_is_command_baton = false;
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Clear(eCallback);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Baton *BreakpointOptions::GetBaton() { return m_callback_baton_sp.get(); }
|
|
|
|
|
2010-06-16 10:00:15 +08:00
|
|
|
const Baton *BreakpointOptions::GetBaton() const {
|
|
|
|
return m_callback_baton_sp.get();
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
bool BreakpointOptions::InvokeCallback(StoppointCallbackContext *context,
|
|
|
|
lldb::user_id_t break_id,
|
|
|
|
lldb::user_id_t break_loc_id) {
|
2018-02-24 05:10:42 +08:00
|
|
|
if (m_callback) {
|
|
|
|
if (context->is_synchronous == IsCallbackSynchronous()) {
|
|
|
|
return m_callback(m_callback_baton_sp ? m_callback_baton_sp->data()
|
2015-10-31 02:50:12 +08:00
|
|
|
: nullptr,
|
|
|
|
context, break_id, break_loc_id);
|
2018-02-24 05:10:42 +08:00
|
|
|
} else if (IsCallbackSynchronous()) {
|
2021-03-20 03:38:41 +08:00
|
|
|
return false;
|
2018-02-24 05:10:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2014-04-09 04:24:45 +08:00
|
|
|
bool BreakpointOptions::HasCallback() const {
|
2010-06-18 09:00:58 +08:00
|
|
|
return m_callback != BreakpointOptions::NullCallback;
|
|
|
|
}
|
|
|
|
|
2016-09-21 06:54:49 +08:00
|
|
|
bool BreakpointOptions::GetCommandLineCallbacks(StringList &command_list) {
|
|
|
|
if (!HasCallback())
|
|
|
|
return false;
|
|
|
|
if (!m_baton_is_command_baton)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
auto cmd_baton = std::static_pointer_cast<CommandBaton>(m_callback_baton_sp);
|
|
|
|
CommandData *data = cmd_baton->getItem();
|
|
|
|
if (!data)
|
|
|
|
return false;
|
|
|
|
command_list = data->user_source;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-10-15 07:45:03 +08:00
|
|
|
void BreakpointOptions::SetCondition(const char *condition) {
|
2017-08-02 08:16:10 +08:00
|
|
|
if (!condition || condition[0] == '\0') {
|
2013-05-11 05:58:45 +08:00
|
|
|
condition = "";
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Clear(eCondition);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
m_set_flags.Set(eCondition);
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2013-04-19 15:09:15 +08:00
|
|
|
m_condition_text.assign(condition);
|
|
|
|
std::hash<std::string> hasher;
|
|
|
|
m_condition_text_hash = hasher(m_condition_text);
|
2010-10-15 07:45:03 +08:00
|
|
|
}
|
|
|
|
|
2013-04-19 15:09:15 +08:00
|
|
|
const char *BreakpointOptions::GetConditionText(size_t *hash) const {
|
|
|
|
if (!m_condition_text.empty()) {
|
|
|
|
if (hash)
|
|
|
|
*hash = m_condition_text_hash;
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2013-04-19 15:09:15 +08:00
|
|
|
return m_condition_text.c_str();
|
2010-10-15 07:45:03 +08:00
|
|
|
} else {
|
2015-10-31 02:50:12 +08:00
|
|
|
return nullptr;
|
2013-04-19 15:09:15 +08:00
|
|
|
}
|
2010-10-15 07:45:03 +08:00
|
|
|
}
|
|
|
|
|
2010-06-23 05:12:54 +08:00
|
|
|
const ThreadSpec *BreakpointOptions::GetThreadSpecNoCreate() const {
|
2019-02-13 14:25:41 +08:00
|
|
|
return m_thread_spec_up.get();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2010-06-16 10:00:15 +08:00
|
|
|
ThreadSpec *BreakpointOptions::GetThreadSpec() {
|
2019-02-13 14:25:41 +08:00
|
|
|
if (m_thread_spec_up == nullptr) {
|
2018-02-24 05:10:42 +08:00
|
|
|
m_set_flags.Set(eThreadSpec);
|
2020-06-25 08:44:33 +08:00
|
|
|
m_thread_spec_up = std::make_unique<ThreadSpec>();
|
2018-02-24 05:10:42 +08:00
|
|
|
}
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2019-02-13 14:25:41 +08:00
|
|
|
return m_thread_spec_up.get();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2010-06-16 10:00:15 +08:00
|
|
|
void BreakpointOptions::SetThreadID(lldb::tid_t thread_id) {
|
|
|
|
GetThreadSpec()->SetTID(thread_id);
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Set(eThreadSpec);
|
2010-06-16 10:00:15 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2016-09-23 06:00:59 +08:00
|
|
|
void BreakpointOptions::SetThreadSpec(
|
|
|
|
std::unique_ptr<ThreadSpec> &thread_spec_up) {
|
2019-02-13 14:25:41 +08:00
|
|
|
m_thread_spec_up = std::move(thread_spec_up);
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Set(eThreadSpec);
|
2016-09-23 06:00:59 +08:00
|
|
|
}
|
|
|
|
|
2010-06-18 09:00:58 +08:00
|
|
|
void BreakpointOptions::GetDescription(Stream *s,
|
|
|
|
lldb::DescriptionLevel level) const {
|
|
|
|
// Figure out if there are any options not at their default value, and only
|
2018-05-01 00:49:04 +08:00
|
|
|
// print anything if there are:
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2017-08-04 02:13:24 +08:00
|
|
|
if (m_ignore_count != 0 || !m_enabled || m_one_shot || m_auto_continue ||
|
2015-10-31 02:50:12 +08:00
|
|
|
(GetThreadSpecNoCreate() != nullptr &&
|
|
|
|
GetThreadSpecNoCreate()->HasSpecification())) {
|
2010-06-18 09:00:58 +08:00
|
|
|
if (level == lldb::eDescriptionLevelVerbose) {
|
|
|
|
s->EOL();
|
|
|
|
s->IndentMore();
|
|
|
|
s->Indent();
|
|
|
|
s->PutCString("Breakpoint Options:\n");
|
|
|
|
s->IndentMore();
|
|
|
|
s->Indent();
|
|
|
|
} else
|
|
|
|
s->PutCString(" Options: ");
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2010-06-18 09:00:58 +08:00
|
|
|
if (m_ignore_count > 0)
|
|
|
|
s->Printf("ignore: %d ", m_ignore_count);
|
|
|
|
s->Printf("%sabled ", m_enabled ? "en" : "dis");
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2012-10-06 03:16:31 +08:00
|
|
|
if (m_one_shot)
|
|
|
|
s->Printf("one-shot ");
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2017-08-04 02:13:24 +08:00
|
|
|
if (m_auto_continue)
|
|
|
|
s->Printf("auto-continue ");
|
|
|
|
|
2019-02-13 14:25:41 +08:00
|
|
|
if (m_thread_spec_up)
|
|
|
|
m_thread_spec_up->GetDescription(s, level);
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2010-06-18 09:00:58 +08:00
|
|
|
if (level == lldb::eDescriptionLevelFull) {
|
|
|
|
s->IndentLess();
|
|
|
|
s->IndentMore();
|
|
|
|
}
|
2016-09-07 04:57:50 +08:00
|
|
|
}
|
|
|
|
|
2011-06-21 02:59:49 +08:00
|
|
|
if (m_callback_baton_sp.get()) {
|
|
|
|
if (level != eDescriptionLevelBrief) {
|
|
|
|
s->EOL();
|
2019-11-30 22:30:08 +08:00
|
|
|
m_callback_baton_sp->GetDescription(s->AsRawOstream(), level,
|
|
|
|
s->GetIndentLevel());
|
2016-09-07 04:57:50 +08:00
|
|
|
}
|
2011-06-21 02:59:49 +08:00
|
|
|
}
|
2013-04-19 15:09:15 +08:00
|
|
|
if (!m_condition_text.empty()) {
|
2010-10-15 07:45:03 +08:00
|
|
|
if (level != eDescriptionLevelBrief) {
|
|
|
|
s->EOL();
|
2013-04-19 15:09:15 +08:00
|
|
|
s->Printf("Condition: %s\n", m_condition_text.c_str());
|
2011-06-21 02:59:49 +08:00
|
|
|
}
|
2010-10-15 07:45:03 +08:00
|
|
|
}
|
2010-06-18 09:00:58 +08:00
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void BreakpointOptions::CommandBaton::GetDescription(
|
2019-11-30 22:30:08 +08:00
|
|
|
llvm::raw_ostream &s, lldb::DescriptionLevel level,
|
|
|
|
unsigned indentation) const {
|
2016-09-14 01:53:38 +08:00
|
|
|
const CommandData *data = getItem();
|
2010-06-18 09:00:58 +08:00
|
|
|
|
|
|
|
if (level == eDescriptionLevelBrief) {
|
2019-11-30 22:30:08 +08:00
|
|
|
s << ", commands = "
|
|
|
|
<< ((data && data->user_source.GetSize() > 0) ? "yes" : "no");
|
2010-06-18 09:00:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2019-11-30 22:30:08 +08:00
|
|
|
indentation += 2;
|
|
|
|
s.indent(indentation);
|
|
|
|
s << "Breakpoint commands";
|
2016-09-27 03:47:37 +08:00
|
|
|
if (data->interpreter != eScriptLanguageNone)
|
2019-11-30 22:30:08 +08:00
|
|
|
s << llvm::formatv(" ({0}):\n",
|
|
|
|
ScriptInterpreter::LanguageToString(data->interpreter));
|
2016-09-27 03:47:37 +08:00
|
|
|
else
|
2019-11-30 22:30:08 +08:00
|
|
|
s << ":\n";
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2019-11-30 22:30:08 +08:00
|
|
|
indentation += 2;
|
2010-06-09 00:52:24 +08:00
|
|
|
if (data && data->user_source.GetSize() > 0) {
|
2019-11-30 22:30:08 +08:00
|
|
|
for (llvm::StringRef str : data->user_source) {
|
|
|
|
s.indent(indentation);
|
|
|
|
s << str << "\n";
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2019-11-30 22:30:08 +08:00
|
|
|
} else
|
|
|
|
s << "No commands.\n";
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2016-09-13 07:10:56 +08:00
|
|
|
|
2016-09-14 01:53:38 +08:00
|
|
|
void BreakpointOptions::SetCommandDataCallback(
|
2016-09-21 06:54:49 +08:00
|
|
|
std::unique_ptr<CommandData> &cmd_data) {
|
2016-09-27 03:47:37 +08:00
|
|
|
cmd_data->interpreter = eScriptLanguageNone;
|
2016-09-14 01:53:38 +08:00
|
|
|
auto baton_sp = std::make_shared<CommandBaton>(std::move(cmd_data));
|
2016-09-13 07:10:56 +08:00
|
|
|
SetCallback(BreakpointOptions::BreakpointOptionsCallbackFunction, baton_sp);
|
2017-08-02 08:16:10 +08:00
|
|
|
m_set_flags.Set(eCallback);
|
2016-09-13 07:10:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool BreakpointOptions::BreakpointOptionsCallbackFunction(
|
|
|
|
void *baton, StoppointCallbackContext *context, lldb::user_id_t break_id,
|
|
|
|
lldb::user_id_t break_loc_id) {
|
|
|
|
bool ret_value = true;
|
|
|
|
if (baton == nullptr)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
CommandData *data = (CommandData *)baton;
|
|
|
|
StringList &commands = data->user_source;
|
|
|
|
|
|
|
|
if (commands.GetSize() > 0) {
|
|
|
|
ExecutionContext exe_ctx(context->exe_ctx_ref);
|
|
|
|
Target *target = exe_ctx.GetTargetPtr();
|
|
|
|
if (target) {
|
|
|
|
Debugger &debugger = target->GetDebugger();
|
2020-06-10 01:21:09 +08:00
|
|
|
CommandReturnObject result(debugger.GetUseColor());
|
|
|
|
|
2016-09-13 07:10:56 +08:00
|
|
|
// Rig up the results secondary output stream to the debugger's, so the
|
2018-05-01 00:49:04 +08:00
|
|
|
// output will come out synchronously if the debugger is set up that way.
|
2016-09-13 07:10:56 +08:00
|
|
|
StreamSP output_stream(debugger.GetAsyncOutputStream());
|
|
|
|
StreamSP error_stream(debugger.GetAsyncErrorStream());
|
|
|
|
result.SetImmediateOutputStream(output_stream);
|
|
|
|
result.SetImmediateErrorStream(error_stream);
|
|
|
|
|
|
|
|
CommandInterpreterRunOptions options;
|
|
|
|
options.SetStopOnContinue(true);
|
|
|
|
options.SetStopOnError(data->stop_on_error);
|
|
|
|
options.SetEchoCommands(true);
|
|
|
|
options.SetPrintResults(true);
|
2019-05-08 09:23:47 +08:00
|
|
|
options.SetPrintErrors(true);
|
2016-09-13 07:10:56 +08:00
|
|
|
options.SetAddToHistory(false);
|
|
|
|
|
2020-12-18 21:36:15 +08:00
|
|
|
debugger.GetCommandInterpreter().HandleCommands(commands, exe_ctx,
|
2016-09-13 07:10:56 +08:00
|
|
|
options, result);
|
|
|
|
result.GetImmediateOutputStream()->Flush();
|
|
|
|
result.GetImmediateErrorStream()->Flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret_value;
|
|
|
|
}
|
2017-09-15 04:22:49 +08:00
|
|
|
|
|
|
|
void BreakpointOptions::Clear()
|
|
|
|
{
|
|
|
|
m_set_flags.Clear();
|
2019-02-13 14:25:41 +08:00
|
|
|
m_thread_spec_up.release();
|
2017-09-15 04:22:49 +08:00
|
|
|
m_one_shot = false;
|
|
|
|
m_ignore_count = 0;
|
|
|
|
m_auto_continue = false;
|
|
|
|
m_callback = nullptr;
|
|
|
|
m_callback_baton_sp.reset();
|
|
|
|
m_baton_is_command_baton = false;
|
|
|
|
m_callback_is_synchronous = false;
|
|
|
|
m_enabled = false;
|
|
|
|
m_condition_text.clear();
|
|
|
|
}
|