ZLMediaKit/src/Common/config.cpp

595 lines
18 KiB
C++
Raw Normal View History

2017-10-09 22:11:01 +08:00
/*
2020-04-04 20:30:09 +08:00
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
2017-09-27 16:20:30 +08:00
*
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
2017-09-27 16:20:30 +08:00
*
2020-04-04 20:30:09 +08:00
* Use of this source code is governed by MIT license that can be found in the
* LICENSE file in the root of the source tree. All contributing project authors
* may be found in the AUTHORS file in the root of the source tree.
2017-09-27 16:20:30 +08:00
*/
2017-05-02 17:15:12 +08:00
#include "Common/config.h"
2022-06-15 00:20:53 +08:00
#include "Util/NoticeCenter.h"
2019-05-10 13:25:22 +08:00
#include "Util/logger.h"
2017-04-01 16:35:56 +08:00
#include "Util/onceToken.h"
2022-06-15 00:20:53 +08:00
#include "Util/util.h"
#include <assert.h>
#include <stdio.h>
2017-04-01 16:35:56 +08:00
using namespace std;
2018-10-24 17:17:55 +08:00
using namespace toolkit;
2017-04-01 16:35:56 +08:00
2018-10-24 17:17:55 +08:00
namespace mediakit {
2017-04-01 16:35:56 +08:00
2022-06-15 00:20:53 +08:00
bool loadIniConfig(const char *ini_path) {
string ini;
2022-06-15 00:20:53 +08:00
if (ini_path && ini_path[0] != '\0') {
ini = ini_path;
2022-06-15 00:20:53 +08:00
} else {
ini = exePath() + ".ini";
}
2022-06-15 00:20:53 +08:00
try {
mINI::Instance().parseFile(ini);
2018-02-09 11:42:55 +08:00
NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastReloadConfig);
return true;
2022-06-15 00:20:53 +08:00
} catch (std::exception &) {
2020-03-20 11:51:24 +08:00
InfoL << "dump ini file to:" << ini;
mINI::Instance().dumpFile(ini);
return false;
2020-03-20 11:51:24 +08:00
}
2017-05-05 18:03:23 +08:00
}
2017-04-01 16:35:56 +08:00
////////////广播名称///////////
namespace Broadcast {
2019-06-12 18:37:52 +08:00
const string kBroadcastMediaChanged = "kBroadcastMediaChanged";
const string kBroadcastRecordMP4 = "kBroadcastRecordMP4";
2022-05-25 15:11:26 +08:00
const string kBroadcastRecordTs = "kBroadcastRecordTs";
2019-06-12 18:37:52 +08:00
const string kBroadcastHttpRequest = "kBroadcastHttpRequest";
const string kBroadcastHttpAccess = "kBroadcastHttpAccess";
const string kBroadcastOnGetRtspRealm = "kBroadcastOnGetRtspRealm";
const string kBroadcastOnRtspAuth = "kBroadcastOnRtspAuth";
const string kBroadcastMediaPlayed = "kBroadcastMediaPlayed";
const string kBroadcastMediaPublish = "kBroadcastMediaPublish";
const string kBroadcastFlowReport = "kBroadcastFlowReport";
const string kBroadcastReloadConfig = "kBroadcastReloadConfig";
const string kBroadcastShellLogin = "kBroadcastShellLogin";
const string kBroadcastNotFoundStream = "kBroadcastNotFoundStream";
const string kBroadcastStreamNoneReader = "kBroadcastStreamNoneReader";
const string kBroadcastHttpBeforeAccess = "kBroadcastHttpBeforeAccess";
2022-08-27 10:53:47 +08:00
const string kBroadcastSendRtpStopped = "kBroadcastSendRtpStopped";
2022-06-15 00:20:53 +08:00
} // namespace Broadcast
2022-06-15 00:20:53 +08:00
// 通用配置项目
namespace General {
2019-05-28 17:14:36 +08:00
#define GENERAL_FIELD "general."
2022-06-15 00:20:53 +08:00
const string kMediaServerId = GENERAL_FIELD "mediaServerId";
const string kFlowThreshold = GENERAL_FIELD "flowThreshold";
const string kStreamNoneReaderDelayMS = GENERAL_FIELD "streamNoneReaderDelayMS";
const string kMaxStreamWaitTimeMS = GENERAL_FIELD "maxStreamWaitMS";
const string kEnableVhost = GENERAL_FIELD "enableVhost";
const string kAddMuteAudio = GENERAL_FIELD "addMuteAudio";
const string kResetWhenRePlay = GENERAL_FIELD "resetWhenRePlay";
const string kPublishToHls = GENERAL_FIELD "publishToHls";
const string kPublishToMP4 = GENERAL_FIELD "publishToMP4";
const string kMergeWriteMS = GENERAL_FIELD "mergeWriteMS";
const string kModifyStamp = GENERAL_FIELD "modifyStamp";
const string kHlsDemand = GENERAL_FIELD "hls_demand";
const string kRtspDemand = GENERAL_FIELD "rtsp_demand";
const string kRtmpDemand = GENERAL_FIELD "rtmp_demand";
const string kTSDemand = GENERAL_FIELD "ts_demand";
const string kFMP4Demand = GENERAL_FIELD "fmp4_demand";
const string kEnableAudio = GENERAL_FIELD "enable_audio";
const string kCheckNvidiaDev = GENERAL_FIELD "check_nvidia_dev";
const string kEnableFFmpegLog = GENERAL_FIELD "enable_ffmpeg_log";
const string kWaitTrackReadyMS = GENERAL_FIELD "wait_track_ready_ms";
const string kWaitAddTrackMS = GENERAL_FIELD "wait_add_track_ms";
const string kUnreadyFrameCache = GENERAL_FIELD "unready_frame_cache";
const string kContinuePushMS = GENERAL_FIELD "continue_push_ms";
static onceToken token([]() {
mINI::Instance()[kFlowThreshold] = 1024;
2019-12-29 11:52:02 +08:00
mINI::Instance()[kStreamNoneReaderDelayMS] = 20 * 1000;
mINI::Instance()[kMaxStreamWaitTimeMS] = 15 * 1000;
2019-12-26 12:23:38 +08:00
mINI::Instance()[kEnableVhost] = 0;
2020-03-20 11:51:24 +08:00
mINI::Instance()[kAddMuteAudio] = 1;
mINI::Instance()[kResetWhenRePlay] = 1;
mINI::Instance()[kPublishToHls] = 1;
mINI::Instance()[kPublishToMP4] = 0;
2020-04-29 11:08:43 +08:00
mINI::Instance()[kMergeWriteMS] = 0;
2020-05-25 12:07:38 +08:00
mINI::Instance()[kModifyStamp] = 0;
2020-10-01 14:37:52 +08:00
mINI::Instance()[kMediaServerId] = makeRandStr(16);
mINI::Instance()[kHlsDemand] = 0;
mINI::Instance()[kRtspDemand] = 0;
mINI::Instance()[kRtmpDemand] = 0;
mINI::Instance()[kTSDemand] = 0;
mINI::Instance()[kFMP4Demand] = 0;
2021-05-22 10:17:52 +08:00
mINI::Instance()[kEnableAudio] = 1;
2022-05-25 15:11:26 +08:00
mINI::Instance()[kCheckNvidiaDev] = 1;
mINI::Instance()[kEnableFFmpegLog] = 0;
mINI::Instance()[kWaitTrackReadyMS] = 10000;
mINI::Instance()[kWaitAddTrackMS] = 3000;
mINI::Instance()[kUnreadyFrameCache] = 100;
mINI::Instance()[kContinuePushMS] = 15 * 1000;
});
2019-05-28 17:14:36 +08:00
2022-06-15 00:20:53 +08:00
} // namespace General
2017-04-01 16:35:56 +08:00
////////////HTTP配置///////////
namespace Http {
#define HTTP_FIELD "http."
2022-06-15 00:20:53 +08:00
const string kSendBufSize = HTTP_FIELD "sendBufSize";
const string kMaxReqSize = HTTP_FIELD "maxReqSize";
const string kKeepAliveSecond = HTTP_FIELD "keepAliveSecond";
const string kCharSet = HTTP_FIELD "charSet";
const string kRootPath = HTTP_FIELD "rootPath";
const string kVirtualPath = HTTP_FIELD "virtualPath";
2022-06-15 00:20:53 +08:00
const string kNotFound = HTTP_FIELD "notFound";
const string kDirMenu = HTTP_FIELD "dirMenu";
const string kForbidCacheSuffix = HTTP_FIELD "forbidCacheSuffix";
const string kForwardedIpHeader = HTTP_FIELD "forwarded_ip_header";
2022-02-17 10:35:10 +08:00
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kSendBufSize] = 64 * 1024;
2021-06-28 20:12:35 +08:00
mINI::Instance()[kMaxReqSize] = 4 * 10240;
2020-03-20 11:51:24 +08:00
mINI::Instance()[kKeepAliveSecond] = 15;
mINI::Instance()[kDirMenu] = true;
mINI::Instance()[kVirtualPath] = "";
#if defined(_WIN32)
2020-03-20 11:51:24 +08:00
mINI::Instance()[kCharSet] = "gb2312";
#else
2022-06-15 00:20:53 +08:00
mINI::Instance()[kCharSet] = "utf-8";
#endif
2020-03-20 11:51:24 +08:00
mINI::Instance()[kRootPath] = "./www";
mINI::Instance()[kNotFound] = StrPrinter << "<html>"
"<head><title>404 Not Found</title></head>"
"<body bgcolor=\"white\">"
"<center><h1>您访问的资源不存在!</h1></center>"
"<hr><center>"
2021-10-16 16:58:03 +08:00
<< kServerName
<< "</center>"
"</body>"
"</html>"
<< endl;
2022-06-15 00:20:53 +08:00
mINI::Instance()[kForbidCacheSuffix] = "";
mINI::Instance()[kForwardedIpHeader] = "";
});
2017-04-01 16:35:56 +08:00
2022-06-15 00:20:53 +08:00
} // namespace Http
2017-04-01 16:35:56 +08:00
////////////SHELL配置///////////
namespace Shell {
#define SHELL_FIELD "shell."
2022-06-15 00:20:53 +08:00
const string kMaxReqSize = SHELL_FIELD "maxReqSize";
2017-04-01 16:35:56 +08:00
2022-06-15 00:20:53 +08:00
static onceToken token([]() { mINI::Instance()[kMaxReqSize] = 1024; });
} // namespace Shell
2017-04-01 16:35:56 +08:00
////////////RTSP服务器配置///////////
namespace Rtsp {
#define RTSP_FIELD "rtsp."
2022-06-15 00:20:53 +08:00
const string kAuthBasic = RTSP_FIELD "authBasic";
const string kHandshakeSecond = RTSP_FIELD "handshakeSecond";
const string kKeepAliveSecond = RTSP_FIELD "keepAliveSecond";
const string kDirectProxy = RTSP_FIELD "directProxy";
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
// 默认Md5方式认证
2020-03-20 11:51:24 +08:00
mINI::Instance()[kAuthBasic] = 0;
2019-05-29 18:24:35 +08:00
mINI::Instance()[kHandshakeSecond] = 15;
mINI::Instance()[kKeepAliveSecond] = 15;
2020-03-20 11:51:24 +08:00
mINI::Instance()[kDirectProxy] = 1;
});
2022-06-15 00:20:53 +08:00
} // namespace Rtsp
2017-04-01 16:35:56 +08:00
////////////RTMP服务器配置///////////
namespace Rtmp {
#define RTMP_FIELD "rtmp."
2022-06-15 00:20:53 +08:00
const string kModifyStamp = RTMP_FIELD "modifyStamp";
const string kHandshakeSecond = RTMP_FIELD "handshakeSecond";
const string kKeepAliveSecond = RTMP_FIELD "keepAliveSecond";
2017-04-01 16:35:56 +08:00
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kModifyStamp] = false;
2019-05-29 18:24:35 +08:00
mINI::Instance()[kHandshakeSecond] = 15;
mINI::Instance()[kKeepAliveSecond] = 15;
});
2022-06-15 00:20:53 +08:00
} // namespace Rtmp
2017-04-01 16:35:56 +08:00
////////////RTP配置///////////
namespace Rtp {
#define RTP_FIELD "rtp."
2022-06-15 00:20:53 +08:00
// RTP打包最大MTU,公网情况下更小
const string kVideoMtuSize = RTP_FIELD "videoMtuSize";
const string kAudioMtuSize = RTP_FIELD "audioMtuSize";
// rtp包最大长度限制单位是KB
const string kRtpMaxSize = RTP_FIELD "rtpMaxSize";
2017-04-01 16:35:56 +08:00
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kVideoMtuSize] = 1400;
mINI::Instance()[kAudioMtuSize] = 600;
2021-08-11 15:48:15 +08:00
mINI::Instance()[kRtpMaxSize] = 10;
});
2022-06-15 00:20:53 +08:00
} // namespace Rtp
2017-04-01 16:35:56 +08:00
////////////组播配置///////////
namespace MultiCast {
#define MULTI_FIELD "multicast."
2022-06-15 00:20:53 +08:00
// 组播分配起始地址
const string kAddrMin = MULTI_FIELD "addrMin";
// 组播分配截止地址
const string kAddrMax = MULTI_FIELD "addrMax";
// 组播TTL
const string kUdpTTL = MULTI_FIELD "udpTTL";
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kAddrMin] = "239.0.0.0";
mINI::Instance()[kAddrMax] = "239.255.255.255";
mINI::Instance()[kUdpTTL] = 64;
});
2022-06-15 00:20:53 +08:00
} // namespace MultiCast
2017-04-01 16:35:56 +08:00
////////////录像配置///////////
namespace Record {
#define RECORD_FIELD "record."
2022-06-15 00:20:53 +08:00
const string kAppName = RECORD_FIELD "appName";
const string kSampleMS = RECORD_FIELD "sampleMS";
const string kFileSecond = RECORD_FIELD "fileSecond";
const string kFilePath = RECORD_FIELD "filePath";
const string kFileBufSize = RECORD_FIELD "fileBufSize";
const string kFastStart = RECORD_FIELD "fastStart";
const string kFileRepeat = RECORD_FIELD "fileRepeat";
const string kMP4AsPlayer = RECORD_FIELD "mp4_as_player";
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kAppName] = "record";
mINI::Instance()[kSampleMS] = 500;
2022-06-15 00:20:53 +08:00
mINI::Instance()[kFileSecond] = 60 * 60;
2020-03-20 11:51:24 +08:00
mINI::Instance()[kFilePath] = "./www";
mINI::Instance()[kFileBufSize] = 64 * 1024;
mINI::Instance()[kFastStart] = false;
mINI::Instance()[kFileRepeat] = false;
mINI::Instance()[kMP4AsPlayer] = false;
});
2022-06-15 00:20:53 +08:00
} // namespace Record
2017-04-01 16:35:56 +08:00
////////////HLS相关配置///////////
namespace Hls {
#define HLS_FIELD "hls."
2022-06-15 00:20:53 +08:00
const string kSegmentDuration = HLS_FIELD "segDur";
const string kSegmentNum = HLS_FIELD "segNum";
const string kSegmentKeep = HLS_FIELD "segKeep";
const string kSegmentRetain = HLS_FIELD "segRetain";
const string kFileBufSize = HLS_FIELD "fileBufSize";
const string kFilePath = HLS_FIELD "filePath";
const string kBroadcastRecordTs = HLS_FIELD "broadcastRecordTs";
const string kDeleteDelaySec = HLS_FIELD "deleteDelaySec";
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kSegmentDuration] = 2;
mINI::Instance()[kSegmentNum] = 3;
mINI::Instance()[kSegmentKeep] = false;
2020-03-20 11:51:24 +08:00
mINI::Instance()[kSegmentRetain] = 5;
mINI::Instance()[kFileBufSize] = 64 * 1024;
mINI::Instance()[kFilePath] = "./www";
mINI::Instance()[kBroadcastRecordTs] = false;
mINI::Instance()[kDeleteDelaySec] = 0;
});
2022-06-15 00:20:53 +08:00
} // namespace Hls
2019-12-06 11:54:10 +08:00
////////////Rtp代理相关配置///////////
namespace RtpProxy {
#define RTP_PROXY_FIELD "rtp_proxy."
2022-06-15 00:20:53 +08:00
const string kDumpDir = RTP_PROXY_FIELD "dumpDir";
const string kTimeoutSec = RTP_PROXY_FIELD "timeoutSec";
const string kPortRange = RTP_PROXY_FIELD "port_range";
const string kH264PT = RTP_PROXY_FIELD "h264_pt";
const string kH265PT = RTP_PROXY_FIELD "h265_pt";
const string kPSPT = RTP_PROXY_FIELD "ps_pt";
const string kTSPT = RTP_PROXY_FIELD "ts_pt";
const string kOpusPT = RTP_PROXY_FIELD "opus_pt";
const string kG711UPT = RTP_PROXY_FIELD "g711u_pt";
const string kG711APT = RTP_PROXY_FIELD "g711a_pt";
2022-06-15 00:20:53 +08:00
static onceToken token([]() {
2020-03-20 11:51:24 +08:00
mINI::Instance()[kDumpDir] = "";
mINI::Instance()[kTimeoutSec] = 15;
mINI::Instance()[kPortRange] = "30000-35000";
mINI::Instance()[kH264PT] = 98;
mINI::Instance()[kH265PT] = 99;
mINI::Instance()[kPSPT] = 96;
mINI::Instance()[kTSPT] = 33;
mINI::Instance()[kOpusPT] = 100;
mINI::Instance()[kG711UPT] = 0;
mINI::Instance()[kG711APT] = 8;
});
2022-06-15 00:20:53 +08:00
} // namespace RtpProxy
2019-12-06 11:54:10 +08:00
2019-03-27 18:56:49 +08:00
namespace Client {
2019-06-12 18:37:52 +08:00
const string kNetAdapter = "net_adapter";
const string kRtpType = "rtp_type";
2022-06-15 00:20:53 +08:00
const string kRtspUser = "rtsp_user";
2019-06-12 18:37:52 +08:00
const string kRtspPwd = "rtsp_pwd";
const string kRtspPwdIsMD5 = "rtsp_pwd_md5";
const string kTimeoutMS = "protocol_timeout_ms";
const string kMediaTimeoutMS = "media_timeout_ms";
const string kBeatIntervalMS = "beat_interval_ms";
2020-04-08 11:16:09 +08:00
const string kBenchmarkMode = "benchmark_mode";
const string kWaitTrackReady = "wait_track_ready";
2022-06-15 00:20:53 +08:00
} // namespace Client
2019-03-27 18:56:49 +08:00
2022-06-15 00:20:53 +08:00
} // namespace mediakit
2017-04-01 16:35:56 +08:00
2021-02-21 21:28:17 +08:00
#ifdef ENABLE_MEM_DEBUG
2021-12-27 17:40:15 +08:00
extern "C" {
2022-06-15 00:20:53 +08:00
extern void *__real_malloc(size_t);
extern void __real_free(void *);
extern void *__real_realloc(void *ptr, size_t c);
void *__wrap_malloc(size_t c);
void __wrap_free(void *ptr);
void *__wrap_calloc(size_t __nmemb, size_t __size);
void *__wrap_realloc(void *ptr, size_t c);
2021-12-27 17:40:15 +08:00
}
#define BLOCK_TYPES 16
#define MIN_BLOCK_SIZE 128
static int get_mem_block_type(size_t c) {
int ret = 0;
while (c > MIN_BLOCK_SIZE && ret + 1 < BLOCK_TYPES) {
c >>= 1;
++ret;
}
return ret;
}
std::vector<size_t> getBlockTypeSize() {
std::vector<size_t> ret;
ret.resize(BLOCK_TYPES);
size_t block_size = MIN_BLOCK_SIZE;
for (auto i = 0; i < BLOCK_TYPES; ++i) {
ret[i] = block_size;
block_size <<= 1;
}
return ret;
}
class MemThreadInfo {
2021-12-27 17:40:15 +08:00
public:
using Ptr = std::shared_ptr<MemThreadInfo>;
2022-06-15 00:20:53 +08:00
atomic<uint64_t> mem_usage { 0 };
atomic<uint64_t> mem_block { 0 };
atomic<uint64_t> mem_block_map[BLOCK_TYPES];
static MemThreadInfo *Instance(bool is_thread_local) {
if (!is_thread_local) {
static auto instance = new MemThreadInfo(is_thread_local);
return instance;
}
static auto thread_local instance = new MemThreadInfo(is_thread_local);
return instance;
}
~MemThreadInfo() {
2022-06-15 00:20:53 +08:00
// printf("%s %d\r\n", __FUNCTION__, (int) _is_thread_local);
}
MemThreadInfo(bool is_thread_local) {
_is_thread_local = is_thread_local;
if (_is_thread_local) {
2022-06-15 00:20:53 +08:00
// 确保所有线程退出后才能释放全局内存统计器
total_mem = Instance(false);
}
2022-06-15 00:20:53 +08:00
// printf("%s %d\r\n", __FUNCTION__, (int) _is_thread_local);
}
2022-06-15 00:20:53 +08:00
void *operator new(size_t sz) { return __real_malloc(sz); }
2022-06-15 00:20:53 +08:00
void operator delete(void *ptr) { __real_free(ptr); }
void addBlock(size_t c) {
if (total_mem) {
total_mem->addBlock(c);
}
mem_usage += c;
++mem_block_map[get_mem_block_type(c)];
++mem_block;
}
void delBlock(size_t c) {
if (total_mem) {
total_mem->delBlock(c);
}
mem_usage -= c;
--mem_block_map[get_mem_block_type(c)];
if (0 == --mem_block) {
delete this;
}
}
private:
bool _is_thread_local;
MemThreadInfo *total_mem = nullptr;
2021-12-27 17:40:15 +08:00
};
class MemThreadInfoLocal {
public:
MemThreadInfoLocal() {
ptr = MemThreadInfo::Instance(true);
ptr->addBlock(1);
}
2022-06-15 00:20:53 +08:00
~MemThreadInfoLocal() { ptr->delBlock(1); }
2022-06-15 00:20:53 +08:00
MemThreadInfo *get() const { return ptr; }
private:
MemThreadInfo *ptr;
};
2021-12-27 17:40:15 +08:00
2022-06-15 00:20:53 +08:00
// 该变量主要确保线程退出后才能释放MemThreadInfo变量
static thread_local MemThreadInfoLocal s_thread_mem_info;
2021-02-21 21:28:17 +08:00
uint64_t getTotalMemUsage() {
return MemThreadInfo::Instance(false)->mem_usage.load();
2021-02-21 21:28:17 +08:00
}
uint64_t getTotalMemBlock() {
return MemThreadInfo::Instance(false)->mem_block.load();
}
uint64_t getTotalMemBlockByType(int type) {
assert(type < BLOCK_TYPES);
return MemThreadInfo::Instance(false)->mem_block_map[type].load();
}
2021-12-27 17:40:15 +08:00
uint64_t getThisThreadMemUsage() {
return MemThreadInfo::Instance(true)->mem_usage.load();
2021-12-27 17:40:15 +08:00
}
2021-02-21 21:28:17 +08:00
uint64_t getThisThreadMemBlock() {
return MemThreadInfo::Instance(true)->mem_block.load();
}
uint64_t getThisThreadMemBlockByType(int type) {
assert(type < BLOCK_TYPES);
return MemThreadInfo::Instance(true)->mem_block_map[type].load();
}
2021-12-27 17:40:15 +08:00
class MemCookie {
public:
static constexpr uint32_t kMagic = 0xFEFDFCFB;
uint32_t magic;
uint32_t size;
2022-06-15 00:20:53 +08:00
MemThreadInfo *alloc_info;
2021-12-27 17:40:15 +08:00
char ptr;
};
2021-12-27 17:40:15 +08:00
#define MEM_OFFSET offsetof(MemCookie, ptr)
#if (defined(__linux__) && !defined(ANDROID)) || defined(__MACH__)
#define MAX_STACK_FRAMES 128
#define MEM_WARING
2022-06-15 00:20:53 +08:00
#include <execinfo.h>
#include <limits.h>
#include <sys/resource.h>
#include <sys/wait.h>
static void print_mem_waring(size_t c) {
void *array[MAX_STACK_FRAMES];
int size = backtrace(array, MAX_STACK_FRAMES);
char **strings = backtrace_symbols(array, size);
printf("malloc big memory:%d, back trace:\r\n", (int)c);
for (int i = 0; i < size; ++i) {
printf("[%d]: %s\r\n", i, strings[i]);
}
__real_free(strings);
}
#endif
2021-02-21 21:28:17 +08:00
static void init_cookie(MemCookie *cookie, size_t c) {
2021-12-27 17:40:15 +08:00
cookie->magic = MemCookie::kMagic;
cookie->size = c;
cookie->alloc_info = s_thread_mem_info.get();
cookie->alloc_info->addBlock(c);
#if defined(MEM_WARING)
static auto env = getenv("MEM_WARN_SIZE");
static size_t s_mem_waring_size = atoll(env ? env : "0");
if (s_mem_waring_size > 1024 && c >= s_mem_waring_size) {
print_mem_waring(c);
}
#endif
}
static void un_init_cookie(MemCookie *cookie) {
cookie->alloc_info->delBlock(cookie->size);
2021-12-27 17:40:15 +08:00
}
2021-02-21 21:28:17 +08:00
void *__wrap_malloc(size_t c) {
2021-12-27 17:40:15 +08:00
c += MEM_OFFSET;
2022-06-15 00:20:53 +08:00
auto cookie = (MemCookie *)__real_malloc(c);
2021-12-27 17:40:15 +08:00
if (cookie) {
init_cookie(cookie, c);
return &cookie->ptr;
2021-02-21 21:28:17 +08:00
}
return nullptr;
}
void __wrap_free(void *ptr) {
if (!ptr) {
return;
}
2022-06-15 00:20:53 +08:00
auto cookie = (MemCookie *)((char *)ptr - MEM_OFFSET);
2021-12-27 17:40:15 +08:00
if (cookie->magic != MemCookie::kMagic) {
__real_free(ptr);
return;
2021-02-21 21:28:17 +08:00
}
un_init_cookie(cookie);
2021-12-27 17:40:15 +08:00
__real_free(cookie);
2021-02-21 21:28:17 +08:00
}
void *__wrap_calloc(size_t __nmemb, size_t __size) {
auto size = __nmemb * __size;
auto ret = malloc(size);
if (ret) {
memset(ret, 0, size);
}
return ret;
}
2021-12-27 17:40:15 +08:00
void *__wrap_realloc(void *ptr, size_t c) {
2021-02-21 21:28:17 +08:00
if (!ptr) {
return malloc(c);
}
2022-06-15 00:20:53 +08:00
auto cookie = (MemCookie *)((char *)ptr - MEM_OFFSET);
2021-12-27 17:40:15 +08:00
if (cookie->magic != MemCookie::kMagic) {
return __real_realloc(ptr, c);
2021-02-21 21:28:17 +08:00
}
2021-12-27 17:40:15 +08:00
un_init_cookie(cookie);
2021-12-27 17:40:15 +08:00
c += MEM_OFFSET;
2022-06-15 00:20:53 +08:00
cookie = (MemCookie *)__real_realloc(cookie, c);
2021-12-27 17:40:15 +08:00
if (cookie) {
init_cookie(cookie, c);
return &cookie->ptr;
2021-02-21 21:28:17 +08:00
}
2021-12-27 17:40:15 +08:00
return nullptr;
2021-02-21 21:28:17 +08:00
}
void *operator new(std::size_t size) {
auto ret = malloc(size);
if (ret) {
return ret;
}
throw std::bad_alloc();
}
void operator delete(void *ptr) noexcept {
2021-02-21 21:28:17 +08:00
free(ptr);
}
void operator delete(void *ptr, std::size_t) noexcept {
2021-06-08 17:48:52 +08:00
free(ptr);
}
2021-02-21 21:28:17 +08:00
void *operator new[](std::size_t size) {
auto ret = malloc(size);
if (ret) {
return ret;
}
throw std::bad_alloc();
}
void operator delete[](void *ptr) noexcept {
2021-02-21 21:28:17 +08:00
free(ptr);
}
2021-06-08 17:48:52 +08:00
void operator delete[](void *ptr, std::size_t) noexcept {
2021-06-08 17:48:52 +08:00
free(ptr);
}
2021-02-21 21:28:17 +08:00
#endif