diff --git a/src/rdsn/src/http/pprof_http_service.cpp b/src/rdsn/src/http/pprof_http_service.cpp index 68fc349e2c..d3b6410c8c 100644 --- a/src/rdsn/src/http/pprof_http_service.cpp +++ b/src/rdsn/src/http/pprof_http_service.cpp @@ -39,6 +39,23 @@ namespace dsn { +bool check_TCMALLOC_SAMPLE_PARAMETER() +{ + char *str = getenv("TCMALLOC_SAMPLE_PARAMETER"); + if (str == nullptr) { + return false; + } + char *endptr; + int val = strtol(str, &endptr, 10); + return (*endptr == '\0' && val > 0); +} + +bool has_TCMALLOC_SAMPLE_PARAMETER() +{ + static bool val = check_TCMALLOC_SAMPLE_PARAMETER(); + return val; +} + // // // == ip:port/pprof/symbol == // // // @@ -332,31 +349,53 @@ void pprof_http_service::heap_handler(const http_request &req, http_response &re resp.status_code = http_status_code::internal_server_error; return; } + auto cleanup = dsn::defer([this]() { _in_pprof_action.store(false); }); + + // If "seconds" parameter is specified with a valid value, use heap profiling, + // otherwise, use heap sampling. + bool use_heap_profile = false; + uint32_t seconds = 0; + const auto &iter = req.query_args.find("seconds"); + if (iter != req.query_args.end() && buf2uint32(iter->second, seconds)) { + // This is true between calls to HeapProfilerStart() and HeapProfilerStop(), and + // also if the program has been run with HEAPPROFILER, or some other + // way to turn on whole-program profiling. + if (IsHeapProfilerRunning()) { + dwarn_f("heap profiling is running, dump the full profile directly"); + char *profile = GetHeapProfile(); + resp.status_code = http_status_code::ok; + resp.body = profile; + free(profile); + return; + } - const std::string SECOND = "seconds"; - const uint32_t kDefaultSecond = 10; - - // get seconds from query params, default value is `kDefaultSecond` - uint32_t seconds = kDefaultSecond; - const auto iter = req.query_args.find(SECOND); - if (iter != req.query_args.end()) { - const auto seconds_str = iter->second; - dsn::internal::buf2unsigned(seconds_str, seconds); - } - - std::stringstream profile_name_prefix; - profile_name_prefix << "heap_profile." << getpid() << "." << dsn_now_ns(); + std::stringstream profile_name_prefix; + profile_name_prefix << "heap_profile." << getpid() << "." << dsn_now_ns(); - HeapProfilerStart(profile_name_prefix.str().c_str()); - sleep(seconds); - const char *profile = GetHeapProfile(); - HeapProfilerStop(); + HeapProfilerStart(profile_name_prefix.str().c_str()); + sleep(seconds); + char *profile = GetHeapProfile(); + HeapProfilerStop(); - resp.status_code = http_status_code::ok; - resp.body = profile; - delete profile; + resp.status_code = http_status_code::ok; + resp.body = profile; + free(profile); + } else { + if (!has_TCMALLOC_SAMPLE_PARAMETER()) { + static const std::string kNoEnvMsg = "The environment variable " + "TCMALLOC_SAMPLE_PARAMETER should set to a " + "positive value, such as 524288, before running."; + dwarn_f(kNoEnvMsg); + resp.status_code = http_status_code::internal_server_error; + resp.body = kNoEnvMsg; + return; + } - _in_pprof_action.store(false); + std::string buf; + MallocExtension::instance()->GetHeapSample(&buf); + resp.status_code = http_status_code::ok; + resp.body = std::move(buf); + } } // //