944 lines
31 KiB
C++
944 lines
31 KiB
C++
|
|
/* Copyright Statement:
|
|
* *
|
|
* * This software/firmware and related documentation ("MediaTek Software") are
|
|
* * protected under relevant copyright laws. The information contained herein
|
|
* * is confidential and proprietary to MediaTek Inc. and/or its licensors.
|
|
* * Without the prior written permission of MediaTek inc. and/or its licensors,
|
|
* * any reproduction, modification, use or disclosure of MediaTek Software,
|
|
* * and information contained herein, in whole or in part, shall be strictly prohibited.
|
|
* *
|
|
* * MediaTek Inc. (C) 2016. All rights reserved.
|
|
* *
|
|
* * BY OPENING THIS FILE, RECEIVER HEREBY UNEQUIVOCALLY ACKNOWLEDGES AND AGREES
|
|
* * THAT THE SOFTWARE/FIRMWARE AND ITS DOCUMENTATIONS ("MEDIATEK SOFTWARE")
|
|
* * RECEIVED FROM MEDIATEK AND/OR ITS REPRESENTATIVES ARE PROVIDED TO RECEIVER ON
|
|
* * AN "AS-IS" BASIS ONLY. MEDIATEK EXPRESSLY DISCLAIMS ANY AND ALL WARRANTIES,
|
|
* * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED WARRANTIES OF
|
|
* * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR NONINFRINGEMENT.
|
|
* * NEITHER DOES MEDIATEK PROVIDE ANY WARRANTY WHATSOEVER WITH RESPECT TO THE
|
|
* * SOFTWARE OF ANY THIRD PARTY WHICH MAY BE USED BY, INCORPORATED IN, OR
|
|
* * SUPPLIED WITH THE MEDIATEK SOFTWARE, AND RECEIVER AGREES TO LOOK ONLY TO SUCH
|
|
* * THIRD PARTY FOR ANY WARRANTY CLAIM RELATING THERETO. RECEIVER EXPRESSLY ACKNOWLEDGES
|
|
* * THAT IT IS RECEIVER'S SOLE RESPONSIBILITY TO OBTAIN FROM ANY THIRD PARTY ALL PROPER LICENSES
|
|
* * CONTAINED IN MEDIATEK SOFTWARE. MEDIATEK SHALL ALSO NOT BE RESPONSIBLE FOR ANY MEDIATEK
|
|
* * SOFTWARE RELEASES MADE TO RECEIVER'S SPECIFICATION OR TO CONFORM TO A PARTICULAR
|
|
* * STANDARD OR OPEN FORUM. RECEIVER'S SOLE AND EXCLUSIVE REMEDY AND MEDIATEK'S ENTIRE AND
|
|
* * CUMULATIVE LIABILITY WITH RESPECT TO THE MEDIATEK SOFTWARE RELEASED HEREUNDER WILL BE,
|
|
* * AT MEDIATEK'S OPTION, TO REVISE OR REPLACE THE MEDIATEK SOFTWARE AT ISSUE,
|
|
* * OR REFUND ANY SOFTWARE LICENSE FEES OR SERVICE CHARGE PAID BY RECEIVER TO
|
|
* * MEDIATEK FOR SUCH MEDIATEK SOFTWARE AT ISSUE.
|
|
* *
|
|
* * The following software/firmware and/or related documentation ("MediaTek Software")
|
|
* * have been modified by MediaTek Inc. All revisions are subject to any receiver's
|
|
* * applicable license agreements with MediaTek Inc.
|
|
* */
|
|
#if defined(MTK_STACK_CONFIG_LOG) && (MTK_STACK_CONFIG_LOG == TRUE)
|
|
|
|
#define LOG_TAG "bt_log_snoop_twrite"
|
|
|
|
#include "twrite.h"
|
|
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <limits.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/time.h>
|
|
#include <unistd.h>
|
|
|
|
#include <atomic>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <queue>
|
|
#include <vector>
|
|
|
|
#include <base/bind.h>
|
|
#include <base/logging.h>
|
|
#include <base/files/file.h>
|
|
#include <base/files/file_enumerator.h>
|
|
#include <base/files/file_util.h>
|
|
#include <base/memory/singleton.h>
|
|
#include <base/message_loop/message_loop.h>
|
|
#include <base/run_loop.h>
|
|
#include <base/sequenced_task_runner.h>
|
|
#include <base/strings/string_util.h>
|
|
#include <base/synchronization/waitable_event.h>
|
|
#include <base/threading/thread.h>
|
|
#include <chrono>
|
|
#include <utils/misc.h>
|
|
|
|
#include "bt_hci_bdroid.h"
|
|
#include "bt_log_controller.h"
|
|
#include "bt_log_tool.h"
|
|
#include "btsnoop_mem.h"
|
|
#include "hci_inbound_data_monitor.h"
|
|
#include "log_file_controller.h"
|
|
#include "log_mode.h"
|
|
#include "log_time.h"
|
|
#include "log_tree_mgr.h"
|
|
#include "logs_stats.h"
|
|
#include "module.h"
|
|
#include "mtk_stack_config.h"
|
|
#include "mtk_util.h"
|
|
#include "snoop_log_config.h"
|
|
#include "osi/include/alarm.h"
|
|
#include "osi/include/log.h"
|
|
#include "osi/include/osi.h"
|
|
#include "osi/include/properties.h"
|
|
#include "osi/include/thread.h"
|
|
#include "snoop_packetizer.h"
|
|
#include "stack_config.h"
|
|
|
|
#if defined(BT_NET_DEBUG) && (BT_NET_DEBUG == TRUE)
|
|
void btsnoop_net_open();
|
|
void btsnoop_net_close();
|
|
void btsnoop_net_write(const void* data, size_t length);
|
|
#endif
|
|
|
|
// Module life cycle functions
|
|
static future_t* start_up(void) {
|
|
vendor::mediatek::bt::stack::BTSnoop::GetInstance()->StartUp();
|
|
return NULL;
|
|
}
|
|
|
|
static future_t* shut_down(void) {
|
|
vendor::mediatek::bt::stack::BTSnoop::GetInstance()->Stop();
|
|
return NULL;
|
|
}
|
|
|
|
EXPORT_SYMBOL extern const module_t mtk_btsnoop_module = {
|
|
.name = MTK_BTSNOOP_MODULE,
|
|
.init = NULL,
|
|
.start_up = start_up,
|
|
.shut_down = shut_down,
|
|
.clean_up = NULL,
|
|
.dependencies = {STACK_CONFIG_MODULE, NULL}};
|
|
|
|
namespace vendor {
|
|
namespace mediatek {
|
|
namespace bt {
|
|
namespace stack {
|
|
|
|
const std::string kMTKLogDefaultRootPathProperty("/sdcard");
|
|
const std::string kDefaultBTSnoopRelativePath(
|
|
"/debuglogger/connsyslog/bthci");
|
|
const std::string kBTSnoopLoggingProperty("vendor.bthcisnoop.running");
|
|
|
|
constexpr char kLogDirPrefix[] = "CsLog_";
|
|
constexpr char kLogFilePrefix[] = "BT_HCI_";
|
|
constexpr char kLogFileCurrentSuffix[] = ".curf";
|
|
constexpr char kLogFileSuffix[] = ".cfa";
|
|
|
|
std::string GetBTSnoopLogPath(const std::string& root_path) {
|
|
return (std::string(root_path) + kDefaultBTSnoopRelativePath);
|
|
}
|
|
|
|
std::string GetBTSnoopLogInitPath() {
|
|
// AOSP way
|
|
if (IsPropertySet(kBluetoothSnoopRootPath)) {
|
|
return GetPropertyValue(kBluetoothSnoopRootPath);
|
|
} else {
|
|
return std::string(kDefaultBtsnoopPath);
|
|
}
|
|
}
|
|
|
|
class BTSnoopBootBackupHelper {
|
|
public:
|
|
BTSnoopBootBackupHelper() : is_backup_in_flight_(false) {}
|
|
~BTSnoopBootBackupHelper() {
|
|
// Add mutex protection for avoid race condition
|
|
// between data Flush and shutdown
|
|
std::lock_guard<std::mutex> lock(data_wrapper_mutex_);
|
|
}
|
|
|
|
bool is_backup_in_flight() const {
|
|
return is_backup_in_flight_;
|
|
}
|
|
|
|
void set_is_backup_in_flight(bool is_backup_in_flight) {
|
|
is_backup_in_flight_ = is_backup_in_flight;
|
|
}
|
|
|
|
std::mutex& GetDataWrapperMutex() {
|
|
return data_wrapper_mutex_;
|
|
}
|
|
|
|
void Capture(const BT_HDR* buffer, bool is_received) {
|
|
uint64_t timestamp_us = LogTime::GetInstance()->GetEpochTime();
|
|
btsnoop_mem_capture(buffer, timestamp_us);
|
|
BTSnoopPacketWrapper packet(buffer, is_received, timestamp_us);
|
|
LOG_DEBUG(
|
|
"%s: is_received: %d, buffer: event: 0x%x, len: %u, data: %s",
|
|
__func__, is_received, buffer->event, buffer->len,
|
|
DataArrayToString<uint8_t>(buffer->data, buffer->len).c_str());
|
|
LOG_DEBUG(
|
|
"%s: wrapper: is_received: %d, buffer: event: 0x%x, len: %u, data: %s",
|
|
__func__,
|
|
packet.is_received(), packet.buffer()->event, packet.buffer()->len,
|
|
DataArrayToString<uint8_t>(
|
|
packet.buffer()->data,
|
|
packet.buffer()->len).c_str());
|
|
data_wrapper_.push(packet);
|
|
}
|
|
|
|
void Flush(const LogFileKeeper* log_file_keeper,
|
|
SnoopPacketizer* packetizer) {
|
|
std::vector<BTSnoopParcel> parcels;
|
|
if (packetizer) {
|
|
std::lock_guard<std::mutex> lock(data_wrapper_mutex_);
|
|
while (data_wrapper_.size() > 0) {
|
|
LOG_DEBUG("%s: size %zu", __func__, data_wrapper_.size());
|
|
parcels.push_back(
|
|
packetizer->ConvertToParcel(data_wrapper_.front()));
|
|
data_wrapper_.pop();
|
|
}
|
|
}
|
|
if (log_file_keeper && (!parcels.empty())) {
|
|
log_file_keeper->LogData(parcels.data(), parcels.size());
|
|
parcels.clear();
|
|
}
|
|
}
|
|
|
|
// Return the last back up one
|
|
std::string Backup(const base::FilePath& to_path) {
|
|
base::FilePath from_path(GetBTSnoopLogInitPath());
|
|
LOG_INFO("%s try backup from %s to %s",
|
|
__func__, from_path.value().c_str(), to_path.value().c_str());
|
|
if (!from_path.empty() && !to_path.empty()) {
|
|
base::FilePath log_file(BackupBootLogs(from_path, to_path));
|
|
if (base::PathExists(log_file)) {
|
|
return RenameAsCurrentLoggingFile(log_file.value());
|
|
} else {
|
|
LOG_ERROR("%s invalid %s for backup",
|
|
__func__, log_file.value().c_str());
|
|
return std::string();
|
|
}
|
|
} else {
|
|
LOG_ERROR("%s invalid %s for backup",
|
|
__func__, to_path.value().c_str());
|
|
return std::string();
|
|
}
|
|
}
|
|
|
|
void RemoveLastTimeInitLogs(const std::string& base_dir) {
|
|
// Don't need to do capture sync here because it is guaranteed by
|
|
// module start up life cycle controlled by stack_manager.
|
|
// In another word, mtk_btsnoop will be started before hci for sure
|
|
if(base_dir.empty()) {
|
|
// there may be something wrong in StartUp if run into here
|
|
LOG_ERROR("%s empty base_dir!!", __func__);
|
|
return;
|
|
}
|
|
LOG_INFO("%s base_dir: %s", __func__, base_dir.c_str());
|
|
base::FileEnumerator file_enum(base::FilePath(base_dir), false, base::FileEnumerator::FILES);
|
|
for (base::FilePath name = file_enum.Next(); !name.empty(); name = file_enum.Next()) {
|
|
if (base::StartsWith(GetFileName(name), kLogFilePrefix, base::CompareCase::SENSITIVE)) {
|
|
if (base::DeleteFile(name, false)) {
|
|
LOG_WARN("%s %s is deleted", __func__, name.value().c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
base::FilePath ConvertToNewFilePath(const base::FilePath& file_path,
|
|
const base::FilePath& logging_dir) {
|
|
std::vector<base::FilePath::StringType> comps;
|
|
file_path.GetComponents(&comps);
|
|
if (comps.size() > 0) {
|
|
return logging_dir.Append(comps[comps.size()-1]);
|
|
} else {
|
|
LOG_ERROR("%s invalid %s",
|
|
__func__, file_path.value().c_str());
|
|
return file_path;
|
|
}
|
|
}
|
|
|
|
bool MoveBootLogs(const base::FilePath& file_path,
|
|
const base::FilePath& new_path) {
|
|
if (!base::PathExists(file_path)) {
|
|
LOG_WARN("%s %s NOT exist!",
|
|
__func__, file_path.value().c_str());
|
|
return false;
|
|
}
|
|
// Do rename just after the file being created may cause rename hang
|
|
// which causes thread (join) quit timeout exception
|
|
LOG_INFO("%s try move from %s to %s", __func__,
|
|
file_path.value().c_str(), new_path.value().c_str());
|
|
if (_MoveFile(file_path.value().c_str(), new_path.value().c_str(),
|
|
FILESYS_FILE_PERMISSION, FILESYS_USER, FILESYS_GROUP)) {
|
|
LOG_WARN("%s %s is moved to %s.",
|
|
__func__, file_path.value().c_str(), new_path.value().c_str());
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
std::string BackupBootLogs(const base::FilePath& from_dir,
|
|
const base::FilePath& to_dir) {
|
|
CHECK(!from_dir.empty());
|
|
CHECK(!to_dir.empty());
|
|
std::string last_init_file;
|
|
base::FileEnumerator file_enum(base::FilePath(from_dir),
|
|
false, base::FileEnumerator::FILES);
|
|
for (base::FilePath name = file_enum.Next();
|
|
!name.empty(); name = file_enum.Next()) {
|
|
if (base::StartsWith(GetFileName(name),
|
|
kLogFilePrefix, base::CompareCase::SENSITIVE)) {
|
|
base::FilePath new_path = ConvertToNewFilePath(name, to_dir);
|
|
if (MoveBootLogs(name, new_path)) {
|
|
last_init_file = new_path.value();
|
|
}
|
|
}
|
|
}
|
|
LOG_INFO("%s: the last one is %s",
|
|
__func__, last_init_file.c_str());
|
|
return last_init_file;
|
|
}
|
|
|
|
std::string GetFileName(const base::FilePath& file_path) const {
|
|
CHECK(!file_path.empty());
|
|
std::vector<base::FilePath::StringType> comps;
|
|
file_path.GetComponents(&comps);
|
|
return comps[comps.size()-1];
|
|
}
|
|
|
|
std::string RenameAsCurrentLoggingFile(const std::string& file) {
|
|
base::FilePath cur_path(file);
|
|
if (std::string(kLogFileSuffix) == cur_path.FinalExtension()) {
|
|
base::FilePath to_path(cur_path.AddExtension(
|
|
std::string(kLogFileCurrentSuffix)));
|
|
if (base::Move(cur_path, to_path)) {
|
|
LOG_WARN("%s %s is moved to %s.",
|
|
__func__,
|
|
cur_path.value().c_str(),
|
|
to_path.value().c_str());
|
|
return to_path.value();
|
|
} else {
|
|
return file;
|
|
}
|
|
} else {
|
|
return file;
|
|
}
|
|
}
|
|
|
|
std::queue<BTSnoopPacketWrapper> data_wrapper_;
|
|
bool is_backup_in_flight_;
|
|
std::mutex data_wrapper_mutex_;
|
|
};
|
|
|
|
using WriteDataCallback = std::function<void(ssize_t)>;
|
|
|
|
struct BtSnoopStartUpArg {
|
|
public:
|
|
std::string base_dir_;
|
|
bool is_need_append_tag_dir_;
|
|
BTSnoopBootBackupHelper* boot_log_backup_helper_;
|
|
};
|
|
|
|
struct BtSnoopStartUpHelperArg {
|
|
public:
|
|
explicit BtSnoopStartUpHelperArg(base::FilePath base_dir,
|
|
BTSnoopBootBackupHelper* boot_log_backup_helper,
|
|
void *context)
|
|
: base_dir_(base_dir),
|
|
boot_log_backup_helper_(boot_log_backup_helper),
|
|
context_(context) {}
|
|
|
|
base::FilePath base_dir_;
|
|
BTSnoopBootBackupHelper* boot_log_backup_helper_;
|
|
void *context_;
|
|
};
|
|
|
|
class BtSnoopSubject {
|
|
public:
|
|
BtSnoopSubject() = default;
|
|
virtual ~BtSnoopSubject() = default;
|
|
|
|
virtual bool IsLogging() const {
|
|
return false;
|
|
}
|
|
virtual void StartUp(const BtSnoopStartUpArg& arg) = 0;
|
|
virtual void Capture(const BT_HDR* buffer, bool is_received) = 0;
|
|
virtual void Stop() = 0;
|
|
};
|
|
|
|
class BtSnoopHelper {
|
|
public:
|
|
explicit BtSnoopHelper(
|
|
LogFileKeeper* log_keeper,
|
|
LogStatistics* log_stats,
|
|
LogFileTreeManager* log_tree_mgr)
|
|
: log_keeper_(log_keeper),
|
|
log_stats_(log_stats),
|
|
log_tree_mgr_(log_tree_mgr),
|
|
btsnoop_packetizer_(new BtSnoopPacketizer(
|
|
[this](const std::vector<BTSnoopParcel>& parcels,
|
|
base::WaitableEvent* event) {
|
|
OnParcelsReady(parcels, event);
|
|
},
|
|
[this](const BTSnoopParcel& parcel) {
|
|
OnPacketReady(parcel);
|
|
})),
|
|
is_logging_(false),
|
|
thread_(nullptr),
|
|
message_loop_(nullptr),
|
|
run_loop_(nullptr) {}
|
|
~BtSnoopHelper() = default;
|
|
|
|
static void RunMessageLoop(void* context) {
|
|
BtSnoopStartUpHelperArg* arg =
|
|
static_cast<BtSnoopStartUpHelperArg*>(context);
|
|
BtSnoopHelper* impl = static_cast<BtSnoopHelper*>(arg->context_);
|
|
{
|
|
std::lock_guard<std::mutex> lock(impl->message_loop_mutex_);
|
|
impl->message_loop_ = new base::MessageLoop();
|
|
impl->run_loop_ = new base::RunLoop();
|
|
}
|
|
|
|
impl->message_loop_->task_runner()->PostTask(
|
|
FROM_HERE, base::Bind(&BtSnoopHelper::InitTask,
|
|
base::Owned(arg)));
|
|
impl->run_loop_->Run();
|
|
|
|
{
|
|
std::lock_guard<std::mutex> lock(impl->message_loop_mutex_);
|
|
delete impl->message_loop_;
|
|
impl->message_loop_ = nullptr;
|
|
delete impl->run_loop_;
|
|
impl->run_loop_ = nullptr;
|
|
}
|
|
}
|
|
|
|
static void InitTask(BtSnoopStartUpHelperArg* arg) {
|
|
CHECK(arg);
|
|
BtSnoopHelper* impl = static_cast<BtSnoopHelper*>(arg->context_);
|
|
impl->CheckReady();
|
|
impl->BTSnoopInitialize(arg->base_dir_, arg->boot_log_backup_helper_);
|
|
}
|
|
|
|
static void WriteTask(const BtSnoopHelper* impl,
|
|
std::vector<BTSnoopParcel> parcels,
|
|
WriteDataCallback write_cb,
|
|
base::WaitableEvent* event) {
|
|
CHECK(impl);
|
|
CHECK(write_cb);
|
|
|
|
if (!impl->is_logging()) return;
|
|
|
|
ssize_t ret = impl->log_keeper_->LogData(parcels.data(), parcels.size());
|
|
write_cb(ret);
|
|
if (event) {
|
|
event->Signal();
|
|
}
|
|
}
|
|
|
|
static void ShutdownTask(BtSnoopHelper* impl) {
|
|
CHECK(impl);
|
|
#if defined(BT_NET_DEBUG) && (BT_NET_DEBUG == TRUE)
|
|
btsnoop_net_close();
|
|
#endif
|
|
impl->log_keeper_->StopLogging();
|
|
impl->is_logging_ = false;
|
|
impl->UpdateLoggingState(impl->is_logging_);
|
|
}
|
|
|
|
bool is_logging() const {
|
|
return is_logging_;
|
|
}
|
|
|
|
void StartUp(BtSnoopStartUpHelperArg* arg) {
|
|
thread_ = thread_new("bt_log_snoop_helper");
|
|
if (!thread_) {
|
|
LOG_ERROR("%s unable to create thread.", __func__);
|
|
} else {
|
|
if (!thread_set_priority(thread_, 0)) {
|
|
LOG_ERROR("%s unable to make thread priority.", __func__);
|
|
}
|
|
thread_post(thread_, RunMessageLoop, arg);
|
|
}
|
|
}
|
|
|
|
void Capture(const BT_HDR* buffer, bool is_received) {
|
|
uint64_t timestamp_us = LogTime::GetInstance()->GetEpochTime();
|
|
btsnoop_mem_capture(buffer, timestamp_us);
|
|
|
|
if (!is_logging_) return;
|
|
|
|
btsnoop_packetizer_->OnDataReady(buffer, is_received, timestamp_us);
|
|
}
|
|
|
|
void Stop() {
|
|
// DO NOT post FlushData() into task thread, will dead lock!
|
|
FlushData();
|
|
LOG_INFO("%s: BtSnoopHelper Stop", __func__);
|
|
{
|
|
std::lock_guard<std::mutex> lock(message_loop_mutex_);
|
|
// Error handling for messge_loop_ is still NULL during thread creation
|
|
// but stop event is coming too soon
|
|
if (message_loop_ != nullptr) {
|
|
message_loop_->task_runner()->PostTask(
|
|
FROM_HERE, base::Bind(&BtSnoopHelper::ShutdownTask, this));
|
|
message_loop_->task_runner()->PostTask(
|
|
FROM_HERE, run_loop_->QuitClosure());
|
|
}
|
|
}
|
|
|
|
// Stop the thread to prevent Send() calls.
|
|
if (thread_) {
|
|
thread_stop(thread_);
|
|
thread_join(thread_);
|
|
thread_free(thread_);
|
|
thread_ = NULL;
|
|
}
|
|
}
|
|
|
|
private:
|
|
void CheckReady() {
|
|
CHECK(btsnoop_packetizer_);
|
|
CHECK(log_keeper_);
|
|
CHECK(log_stats_);
|
|
CHECK(log_tree_mgr_);
|
|
|
|
CHECK(thread_);
|
|
CHECK(message_loop_);
|
|
CHECK(run_loop_);
|
|
}
|
|
|
|
void InitLoggedDataSize(LogStatistics* log_stats,
|
|
LogFileTreeManager* log_tree_mgr) {
|
|
log_stats->UpdateTotalLogedSize(log_tree_mgr->GetTotalLoggedSize(), true);
|
|
log_stats->Dump();
|
|
}
|
|
|
|
void BTSnoopInitialize(const base::FilePath& dir,
|
|
BTSnoopBootBackupHelper* boot_backup_helper) {
|
|
if (log_keeper_->MakeDir(dir)) {
|
|
if (boot_backup_helper) {
|
|
std::string logging_file = boot_backup_helper->Backup(dir);
|
|
if (!logging_file.empty()) {
|
|
is_logging_ = log_keeper_->AppendLogging(
|
|
base::FilePath(logging_file));
|
|
}
|
|
if (is_logging_) {
|
|
// set the flag before pop data cache queue to avoid race condition
|
|
{
|
|
std::lock_guard<std::mutex> lock(
|
|
boot_backup_helper->GetDataWrapperMutex());
|
|
boot_backup_helper->set_is_backup_in_flight(false);
|
|
}
|
|
boot_backup_helper->Flush(log_keeper_, btsnoop_packetizer_.get());
|
|
} else {
|
|
// error handling here
|
|
std::lock_guard<std::mutex> lock(
|
|
boot_backup_helper->GetDataWrapperMutex());
|
|
boot_backup_helper->set_is_backup_in_flight(false);
|
|
}
|
|
} else {
|
|
is_logging_ = log_keeper_->StartLogging(dir);
|
|
}
|
|
if (is_logging_) {
|
|
log_tree_mgr_->InitLogsFileTree(dir);
|
|
log_tree_mgr_->set_current_logging_dir(dir);
|
|
InitLoggedDataSize(log_stats_, log_tree_mgr_);
|
|
UpdateLoggingState(is_logging_);
|
|
#if defined(BT_NET_DEBUG) && (BT_NET_DEBUG == TRUE)
|
|
btsnoop_net_open();
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
void UpdateLoggingState(bool is_logging) {
|
|
if (osi_property_set(
|
|
kBTSnoopLoggingProperty.c_str(),
|
|
is_logging ? "1" : "0")) {
|
|
LOG_ERROR("%s unable to set %s to %s.",
|
|
__func__,
|
|
kBTSnoopLoggingProperty.c_str(),
|
|
is_logging ? "1" : "0");
|
|
}
|
|
}
|
|
|
|
void OnParcelsReady(const std::vector<BTSnoopParcel>& parcels,
|
|
base::WaitableEvent* event) {
|
|
base::Closure callback = base::Bind(&WriteTask,
|
|
this,
|
|
std::move(parcels),
|
|
[this](ssize_t ret) {
|
|
if (ret) {
|
|
log_stats_->UpdateCurrentLogedFileSize(ret + log_stats_->current_loged_file_size(), true);
|
|
log_stats_->UpdateTotalLogedSize(ret + log_stats_->total_loged_size(), true);
|
|
}
|
|
},
|
|
event);
|
|
std::lock_guard<std::mutex> lock(message_loop_mutex_);
|
|
if (message_loop_ != nullptr) {
|
|
message_loop_->task_runner()->PostTask(FROM_HERE, std::move(callback));
|
|
}
|
|
}
|
|
|
|
void OnPacketReady(const BTSnoopParcel& parcel) {
|
|
#if defined(BT_NET_DEBUG) && (BT_NET_DEBUG == TRUE)
|
|
btsnoop_net_write(&parcel.header(), sizeof(parcel.header()));
|
|
btsnoop_net_write(parcel.packet().data(), parcel.packet().size());
|
|
#endif
|
|
}
|
|
|
|
void FlushData() {
|
|
base::WaitableEvent event(base::WaitableEvent::ResetPolicy::AUTOMATIC,
|
|
base::WaitableEvent::InitialState::NOT_SIGNALED);
|
|
btsnoop_packetizer_->OnShutdown(&event);
|
|
// Wait for the left data are flashed into storage
|
|
event.Wait();
|
|
}
|
|
|
|
LogFileKeeper* log_keeper_;
|
|
LogStatistics* log_stats_;
|
|
LogFileTreeManager* log_tree_mgr_;
|
|
std::unique_ptr<SnoopPacketizer> btsnoop_packetizer_;
|
|
bool is_logging_;
|
|
|
|
thread_t* thread_;
|
|
base::MessageLoop* message_loop_;
|
|
base::RunLoop* run_loop_;
|
|
|
|
std::mutex message_loop_mutex_;
|
|
};
|
|
|
|
class BtSnoopImpl : public BtSnoopSubject {
|
|
public:
|
|
BtSnoopImpl()
|
|
: is_started_(false),
|
|
inbound_data_monitor_(nullptr),
|
|
log_tree_mgr_(nullptr),
|
|
log_keeper_(nullptr),
|
|
log_cleaner_(nullptr),
|
|
log_stats_(nullptr),
|
|
bt_snoop_helper_(nullptr) {}
|
|
~BtSnoopImpl() = default;
|
|
|
|
void StartUp(const BtSnoopStartUpArg& arg) override {
|
|
if (is_started_) {
|
|
LOG_INFO("%s: BtSnoopImpl ignore dup StartUp", __func__);
|
|
return;
|
|
}
|
|
std::lock_guard<std::mutex> lock(btsnoop_mutex_);
|
|
LOG_INFO("%s: BtSnoopImpl StartUp", __func__);
|
|
Setup(arg.base_dir_);
|
|
base::FilePath current_logging_dir(arg.is_need_append_tag_dir_ ?
|
|
log_tree_mgr_->logging_base_dir().Append(
|
|
std::string(kLogDirPrefix) +
|
|
LogTime::GetInstance()->GetLogTimeTag()) :
|
|
log_tree_mgr_->logging_base_dir());
|
|
BtSnoopStartUpHelperArg* helper_arg(new BtSnoopStartUpHelperArg(
|
|
current_logging_dir, arg.boot_log_backup_helper_, bt_snoop_helper_.get()));
|
|
bt_snoop_helper_->StartUp(helper_arg);
|
|
is_started_ = true;
|
|
}
|
|
|
|
void Capture(const BT_HDR* buffer, bool is_received) override {
|
|
uint64_t timestamp_us = LogTime::GetInstance()->GetEpochTime();
|
|
btsnoop_mem_capture(buffer, timestamp_us);
|
|
|
|
std::lock_guard<std::mutex> lock(btsnoop_mutex_);
|
|
if ((inbound_data_monitor_ != nullptr) && is_received) {
|
|
inbound_data_monitor_->InboundDataUpdate(buffer);
|
|
}
|
|
|
|
if (bt_snoop_helper_ && bt_snoop_helper_->is_logging()) {
|
|
bt_snoop_helper_->Capture(buffer, is_received);
|
|
}
|
|
}
|
|
|
|
void Stop() override {
|
|
if (!is_started_) {
|
|
LOG_INFO("%s: BtSnoopImpl ignore dup Stop", __func__);
|
|
return;
|
|
}
|
|
std::lock_guard<std::mutex> lock(btsnoop_mutex_);
|
|
LOG_INFO("%s: BtSnoopImpl Stop", __func__);
|
|
if (bt_snoop_helper_) {
|
|
bt_snoop_helper_->Stop();
|
|
}
|
|
Teardown();
|
|
is_started_ = false;
|
|
}
|
|
|
|
bool IsLogging() const override {
|
|
return (bt_snoop_helper_ && bt_snoop_helper_->is_logging());
|
|
}
|
|
|
|
private:
|
|
void Setup(const std::string& base_dir) {
|
|
log_tree_mgr_ = std::unique_ptr<LogFileTreeManager>(
|
|
new LogFileTreeManager(base_dir,
|
|
std::string(kLogFilePrefix),
|
|
std::string(kLogFileSuffix),
|
|
std::string(kLogFileCurrentSuffix)));
|
|
log_keeper_ = std::unique_ptr<LogFileKeeper>(
|
|
new LogFileKeeper(log_tree_mgr_.get()));
|
|
log_cleaner_ = std::unique_ptr<LogFileClearner>(
|
|
new LogFileClearner(log_tree_mgr_.get(),
|
|
SnoopLogConfig::GetInstance()->GetRecycleLogsSize()));
|
|
log_stats_ = std::unique_ptr<LogStatistics>(new LogStatistics());
|
|
log_stats_->AppendObserver(log_keeper_.get());
|
|
log_stats_->AppendObserver(log_cleaner_.get());
|
|
bt_snoop_helper_ = std::unique_ptr<BtSnoopHelper>(new BtSnoopHelper(
|
|
log_keeper_.get(), log_stats_.get(), log_tree_mgr_.get()));
|
|
|
|
// For low power concern, don't default turn on inbound data monitor
|
|
if (StackConfig::GetInstance()->IsInboundDataMonitorEnabled()) {
|
|
inbound_data_monitor_.reset(
|
|
new InboundDataMonitor(std::string(MTK_BTSNOOP_MODULE)));
|
|
if (inbound_data_monitor_ != nullptr) {
|
|
inbound_data_monitor_->InboundDataFilterUpdate(kInboundDataVSEFWLog, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Teardown() {
|
|
if (bt_snoop_helper_ != nullptr) {
|
|
bt_snoop_helper_.reset();
|
|
bt_snoop_helper_ = nullptr;
|
|
}
|
|
|
|
if (log_stats_ != nullptr) {
|
|
log_stats_.reset();
|
|
log_stats_ = nullptr;
|
|
}
|
|
|
|
if (log_keeper_ != nullptr) {
|
|
log_keeper_.reset();
|
|
log_keeper_ = nullptr;
|
|
}
|
|
|
|
if (log_cleaner_ != nullptr) {
|
|
log_cleaner_.reset();
|
|
log_cleaner_ = nullptr;
|
|
}
|
|
|
|
if (log_tree_mgr_ != nullptr) {
|
|
log_tree_mgr_.reset();
|
|
log_tree_mgr_ = nullptr;
|
|
}
|
|
|
|
if (inbound_data_monitor_ != nullptr) {
|
|
inbound_data_monitor_.reset();
|
|
inbound_data_monitor_ = nullptr;
|
|
}
|
|
}
|
|
|
|
bool is_started_;
|
|
std::unique_ptr<InboundDataMonitor> inbound_data_monitor_;
|
|
|
|
std::unique_ptr<LogFileTreeManager> log_tree_mgr_;
|
|
std::unique_ptr<LogFileKeeper> log_keeper_;
|
|
std::unique_ptr<LogFileClearner> log_cleaner_;
|
|
std::unique_ptr<LogStatistics> log_stats_;
|
|
std::unique_ptr<BtSnoopHelper> bt_snoop_helper_;
|
|
|
|
std::mutex btsnoop_mutex_;
|
|
};
|
|
|
|
static constexpr uint64_t kInitLogCancelTimeoutMs = 1000 * 60 * 5;
|
|
class BtSnoopProxy
|
|
: public BtSnoopSubject,
|
|
public BtLogStateObserver {
|
|
public:
|
|
BtSnoopProxy()
|
|
: is_boot_logging_(true),
|
|
is_need_backup_boot_logs_(false),
|
|
last_logging_request_(kBtHciLogUnknown),
|
|
btsnoop_(nullptr),
|
|
snoop_boot_backup_helper_(new BTSnoopBootBackupHelper()),
|
|
init_log_timer_(nullptr) {}
|
|
~BtSnoopProxy() = default;
|
|
|
|
// TODO: To re-factor it if gets time with state pattern
|
|
void OnBtLogStateChanged(const BluetoothLogSetting* setting) override {
|
|
CHECK(setting);
|
|
if (last_logging_request_ == setting->GetHciLogLevel()) {
|
|
LOG_WARN("%s ignore dup request current: %d vs last: %d.",
|
|
__func__, setting->GetHciLogLevel(), last_logging_request_);
|
|
return;
|
|
}
|
|
|
|
CancelInitLogTimer();
|
|
if (is_boot_logging_) {
|
|
bool is_need_boot_backup = (nullptr != btsnoop_) ? true : false;
|
|
if (nullptr == btsnoop_) {
|
|
SnoopLogConfig::GetInstance()->Refresh();
|
|
btsnoop_ = std::unique_ptr<BtSnoopSubject>(new BtSnoopImpl());
|
|
}
|
|
LOG_INFO("%s boot: new state: %s vs current state: %s.",
|
|
__func__,
|
|
setting->IsHciSnoopEnabled() ? "on" : "off",
|
|
btsnoop_->IsLogging() ? "on" : "off");
|
|
is_boot_logging_ = false;
|
|
DoLogSwitch(setting->IsHciSnoopEnabled(), is_need_boot_backup,
|
|
setting->GetHciLogPath());
|
|
} else {
|
|
LOG_INFO("%s normal: new state: %s vs current state: %s.",
|
|
__func__,
|
|
setting->IsHciSnoopEnabled() ? "on" : "off",
|
|
btsnoop_->IsLogging() ? "on" : "off");
|
|
if (btsnoop_->IsLogging() !=
|
|
setting->IsHciSnoopEnabled()) {
|
|
DoLogSwitch(setting->IsHciSnoopEnabled(), false,
|
|
setting->GetHciLogPath());
|
|
} else {
|
|
LOG_INFO("%s ignore for duplicated request %s.",
|
|
__func__,
|
|
setting->IsHciSnoopEnabled() ? "on" : "off");
|
|
}
|
|
}
|
|
last_logging_request_ = setting->GetHciLogLevel();
|
|
}
|
|
|
|
void StartUp(const BtSnoopStartUpArg& arg) override {
|
|
if (!is_boot_logging_) {
|
|
if (!is_need_backup_boot_logs_) {
|
|
LOG_INFO("%s backup already", __func__);
|
|
btsnoop_->StartUp(arg);
|
|
} else {
|
|
LOG_INFO("%s do backup", __func__);
|
|
btsnoop_->Stop();
|
|
{
|
|
std::lock_guard<std::mutex> lock(
|
|
snoop_boot_backup_helper_->GetDataWrapperMutex());
|
|
snoop_boot_backup_helper_->set_is_backup_in_flight(true);
|
|
}
|
|
btsnoop_->StartUp(arg);
|
|
}
|
|
} else {
|
|
LOG_INFO("%s boot", __func__);
|
|
if (BluetoothLogSetting::GetInstance()->IsAutoStartHciSnoop()) {
|
|
SnoopLogConfig::GetInstance()->Refresh();
|
|
btsnoop_ = std::unique_ptr<BtSnoopSubject>(new BtSnoopImpl());
|
|
snoop_boot_backup_helper_->RemoveLastTimeInitLogs(arg.base_dir_);
|
|
btsnoop_->StartUp(arg);
|
|
if (!StackConfig::GetInstance()->IsBTSnoopEnabled()) {
|
|
StartInitLogTimer();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Capture(const BT_HDR* buffer, bool is_received) override {
|
|
if (!is_boot_logging_) {
|
|
std::lock_guard<std::mutex> lock(
|
|
snoop_boot_backup_helper_->GetDataWrapperMutex());
|
|
if (snoop_boot_backup_helper_ &&
|
|
snoop_boot_backup_helper_->is_backup_in_flight()) {
|
|
// error handling
|
|
if (BtSnoopPacketizer::Validate(buffer, is_received)) {
|
|
snoop_boot_backup_helper_->Capture(buffer, is_received);
|
|
}
|
|
} else {
|
|
if (btsnoop_) {
|
|
btsnoop_->Capture(buffer, is_received);
|
|
}
|
|
}
|
|
} else {
|
|
if (btsnoop_) {
|
|
btsnoop_->Capture(buffer, is_received);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Stop() override {
|
|
// lock to prevent race condition between stop by init_log_timer_ and normal stop
|
|
std::lock_guard<std::mutex> lock(stop_mutex_);
|
|
CancelInitLogTimer();
|
|
if (btsnoop_) {
|
|
btsnoop_->Stop();
|
|
}
|
|
}
|
|
|
|
private:
|
|
void DoLogSwitch(bool is_on, bool is_need_boot_backup,
|
|
const std::string& root_path) {
|
|
if (is_on) {
|
|
BtSnoopStartUpArg arg{
|
|
GetBTSnoopLogPath(root_path), true,
|
|
is_need_boot_backup ? snoop_boot_backup_helper_.get() : nullptr
|
|
};
|
|
is_need_backup_boot_logs_ = is_need_boot_backup;
|
|
StartUp(arg);
|
|
} else {
|
|
Stop();
|
|
}
|
|
}
|
|
|
|
void StartInitLogTimer() {
|
|
init_log_timer_ = alarm_new("bt_log_init_snoop_timer");
|
|
if (init_log_timer_) {
|
|
alarm_set(init_log_timer_, kInitLogCancelTimeoutMs,
|
|
[](void* context) {
|
|
CHECK(NULL != context);
|
|
LOG_INFO("%s, no log command, to stop", __func__);
|
|
BtSnoopSubject* btsnoop =
|
|
static_cast<BtSnoopSubject*>(context);
|
|
btsnoop->Stop();
|
|
},
|
|
this);
|
|
}
|
|
}
|
|
|
|
void CancelInitLogTimer() {
|
|
if (init_log_timer_ != nullptr) {
|
|
LOG_INFO("%s", __func__);
|
|
alarm_free(init_log_timer_);
|
|
init_log_timer_ = nullptr;
|
|
}
|
|
}
|
|
|
|
bool is_boot_logging_;
|
|
bool is_need_backup_boot_logs_;
|
|
BtHciLogLevel last_logging_request_;
|
|
std::unique_ptr<BtSnoopSubject> btsnoop_;
|
|
std::unique_ptr<BTSnoopBootBackupHelper> snoop_boot_backup_helper_;
|
|
alarm_t* init_log_timer_;
|
|
std::mutex stop_mutex_;
|
|
};
|
|
|
|
// It is WRONG to default initialize tons of BT snoop classes and threads
|
|
// End user build should NOT suffer that if it is not for debugging on propose
|
|
BTSnoop::BTSnoop() : btsnoop_proxy_(new BtSnoopProxy()) {}
|
|
|
|
BTSnoop* BTSnoop::GetInstance() { return base::Singleton<BTSnoop>::get(); }
|
|
|
|
void BTSnoop::StartUp() {
|
|
BtSnoopStartUpArg arg{GetBTSnoopLogInitPath(), false, nullptr};
|
|
btsnoop_proxy_->StartUp(arg);
|
|
btlogtool::SetUp();
|
|
btlogtool::AddObserver(btsnoop_proxy_.get());
|
|
// Add observer for SnoopLogConfig
|
|
SnoopLogConfig::GetInstance()->Setup();
|
|
}
|
|
|
|
void BTSnoop::Capture(const BT_HDR* buffer, bool is_received) {
|
|
btsnoop_proxy_->Capture(buffer, is_received);
|
|
}
|
|
|
|
void BTSnoop::Stop() {
|
|
// Remove observer for SnoopLogConfig
|
|
SnoopLogConfig::GetInstance()->TearDown();
|
|
btlogtool::RemoveObserver(btsnoop_proxy_.get());
|
|
btlogtool::TearDown();
|
|
|
|
btsnoop_proxy_->Stop();
|
|
}
|
|
|
|
} // namespace stack
|
|
} // namespace bt
|
|
} // namespace mediatek
|
|
} // namespace vendor
|
|
|
|
#endif
|