mirror of
				https://github.com/HappyTanuki/BumbleCee.git
				synced 2025-10-26 01:45:15 +00:00 
			
		
		
		
	아마도? 쉘 인젝션 취약점 픽스
This commit is contained in:
		
							
								
								
									
										4
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -102,6 +102,8 @@ | |||||||
|         "hash_set": "cpp", |         "hash_set": "cpp", | ||||||
|         "regex": "cpp", |         "regex": "cpp", | ||||||
|         "stack": "cpp", |         "stack": "cpp", | ||||||
|         "__memory": "cpp" |         "__memory": "cpp", | ||||||
|  |         "__verbose_abort": "cpp", | ||||||
|  |         "print": "cpp" | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -1,5 +1,7 @@ | |||||||
| cmake_minimum_required (VERSION 3.6) | cmake_minimum_required (VERSION 3.6) | ||||||
|  |  | ||||||
|  | INCLUDE_DIRECTORIES(include . ${Boost_INCLUDE_DIR}) | ||||||
|  |  | ||||||
| list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) | list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) | ||||||
| set(BOT_NAME "BumbleCee") | set(BOT_NAME "BumbleCee") | ||||||
|  |  | ||||||
| @@ -18,8 +20,11 @@ set(CMAKE_CXX_STANDARD 20) | |||||||
| set(CMAKE_CXX_STANDARD_REQUIRED ON) | set(CMAKE_CXX_STANDARD_REQUIRED ON) | ||||||
|  |  | ||||||
| set(THREADS_PREFER_PTHREAD_FLAG TRUE) | set(THREADS_PREFER_PTHREAD_FLAG TRUE) | ||||||
|  | add_definitions( -DBOOST_ALL_NO_LIB ) | ||||||
|  | set( Boost_USE_STATIC_LIBS ON ) | ||||||
| find_package(Threads REQUIRED) | find_package(Threads REQUIRED) | ||||||
| find_package(OpenSSL REQUIRED) | find_package(OpenSSL REQUIRED) | ||||||
|  | find_package(Boost REQUIRED) | ||||||
|  |  | ||||||
| target_include_directories(${BOT_NAME} PUBLIC | target_include_directories(${BOT_NAME} PUBLIC | ||||||
|     ${CMAKE_CURRENT_SOURCE_DIR}/include |     ${CMAKE_CURRENT_SOURCE_DIR}/include | ||||||
| @@ -35,6 +40,7 @@ target_link_libraries(${BOT_NAME} | |||||||
|     ${CMAKE_THREAD_LIBS_INIT} |     ${CMAKE_THREAD_LIBS_INIT} | ||||||
|     ${OPENSSL_CRYPTO_LIBRARY}  |     ${OPENSSL_CRYPTO_LIBRARY}  | ||||||
|     ${OPENSSL_SSL_LIBRARY} |     ${OPENSSL_SSL_LIBRARY} | ||||||
|  |     ${Boost_LIBRARIES} | ||||||
| ) | ) | ||||||
|  |  | ||||||
| link_directories(/usr/lib) | link_directories(/usr/lib) | ||||||
							
								
								
									
										1303
									
								
								flamegraph.pl
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1303
									
								
								flamegraph.pl
									
									
									
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										67
									
								
								generateStackFlame.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										67
									
								
								generateStackFlame.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | # 기본값 설정 | ||||||
|  | n_flag=20 | ||||||
|  | i_flag=0 | ||||||
|  | process_args=() | ||||||
|  |  | ||||||
|  | # 입력값 파싱 | ||||||
|  | while [[ $# -gt 0 ]]; do | ||||||
|  |     case "$1" in | ||||||
|  |         -n) | ||||||
|  |             shift | ||||||
|  |             if [[ ! "$1" =~ ^[0-9]+$ ]]; then | ||||||
|  |                 echo "오류: -n 옵션 뒤에는 숫자가 와야 합니다." >&2 | ||||||
|  |                 exit 1 | ||||||
|  |             fi | ||||||
|  |             n_flag="$1" | ||||||
|  |             ;; | ||||||
|  |         -i) | ||||||
|  |             shift | ||||||
|  |             if [[ ! "$1" =~ ^[0-9]+$ ]]; then | ||||||
|  |                 echo "오류: -i 옵션 뒤에는 숫자가 와야 합니다." >&2 | ||||||
|  |                 exit 1 | ||||||
|  |             fi | ||||||
|  |             i_flag="$1" | ||||||
|  |             ;; | ||||||
|  |         *) | ||||||
|  |             # 프로세스 명 또는 PID 검증 | ||||||
|  |             if [[ "$1" =~ ^[0-9]+$ ]]; then | ||||||
|  |                 # 숫자인 경우, PID 존재 여부 확인 | ||||||
|  |                 if ! ps -p "$1" > /dev/null 2>&1; then | ||||||
|  |                     echo "오류: PID $1 에 해당하는 프로세스가 존재하지 않습니다." >&2 | ||||||
|  |                     exit 1 | ||||||
|  |                 fi | ||||||
|  |             else | ||||||
|  |                 # 문자열인 경우, 프로세스 이름 존재 여부 확인 | ||||||
|  |                 if ! pgrep -x "$1" > /dev/null 2>&1; then | ||||||
|  |                     echo "오류: 프로세스 '$1' 가 실행 중이지 않습니다." >&2 | ||||||
|  |                     exit 1 | ||||||
|  |                 fi | ||||||
|  |             fi | ||||||
|  |             process_args+=("$1") | ||||||
|  |             ;; | ||||||
|  |     esac | ||||||
|  |     shift | ||||||
|  | done | ||||||
|  |  | ||||||
|  | # 프로세스 인자 검증 | ||||||
|  | if [[ ${#process_args[@]} -eq 0 ]]; then | ||||||
|  |     echo "오류: 최소한 하나 이상의 프로세스 이름 또는 PID를 입력해야 합니다." >&2 | ||||||
|  |     exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | pid=$(pidof $process_args) | ||||||
|  |  | ||||||
|  | for x in $(seq 1 $n_flag) | ||||||
|  |     do | ||||||
|  |         sudo gdb -ex "set pagination 0" \ | ||||||
|  |                 -ex "thread apply all bt" \ | ||||||
|  |                 -batch -p $pid >> out.gdb 2> /dev/null  | ||||||
|  |         sleep $i_flag | ||||||
|  |     done | ||||||
|  |  | ||||||
|  | ./stackcollapse-gdb.pl out.gdb > out.folded | ||||||
|  | rm out.gdb | ||||||
|  | ./flamegraph.pl out.folded > stackflame.svg | ||||||
|  | rm out.folded | ||||||
| @@ -2,6 +2,8 @@ | |||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <sstream> | #include <sstream> | ||||||
| #include <queue> | #include <queue> | ||||||
|  | #include <regex> | ||||||
|  | #include <boost/process.hpp> | ||||||
|  |  | ||||||
| namespace bumbleBee { | namespace bumbleBee { | ||||||
| class ConsoleUtils { | class ConsoleUtils { | ||||||
| @@ -9,28 +11,36 @@ public: | |||||||
|     /**  |     /**  | ||||||
|      * @brief 명령어를 쉘에서 실행하고 결과를 EOF 전까지 읽어 \n을 구분자로 토큰화하여 반환합니다 |      * @brief 명령어를 쉘에서 실행하고 결과를 EOF 전까지 읽어 \n을 구분자로 토큰화하여 반환합니다 | ||||||
|      * @param cmd 실행할 명령 |      * @param cmd 실행할 명령 | ||||||
|  |      * @param args 아규먼트 | ||||||
|      * @return std::queue<std::string> tokens |      * @return std::queue<std::string> tokens | ||||||
|      */ |      */ | ||||||
|     static std::queue<std::string> getResultFromCommand(std::string cmd) { |     static std::queue<std::string> safe_execute_command(const std::string& cmd, const std::vector<std::string>& args) { | ||||||
|         std::string result, token; |  | ||||||
|         std::queue<std::string> tokens; |         std::queue<std::string> tokens; | ||||||
|         FILE* stream; |         try { | ||||||
|         const int maxBuffer = 12; // 적당한 크기 |             boost::process::ipstream output;  // 명령의 출력을 받을 스트림 | ||||||
|         char buffer[maxBuffer]; |             boost::process::child c(cmd, boost::process::args(args), boost::process::std_out > output); | ||||||
|         cmd.append(" 2>&1"); // 표준에러를 표준출력으로 redirect |  | ||||||
|  |  | ||||||
|         stream = popen(cmd.c_str(), "r"); // 주어진 command를 shell로 실행하고 파이프 연결 (fd 반환) |             std::string line; | ||||||
|             if (stream) { |             while (!output.eof() && std::getline(output, line)) | ||||||
|                 while (fgets(buffer, maxBuffer, stream) != NULL) result.append(buffer); // fgets: fd (stream)를 길이 (maxBuffer)만큼 읽어 버퍼 (buffer)에 저장 |                 tokens.push(line); | ||||||
|                 pclose(stream); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         std::stringstream ss(result); |             c.wait();  // 프로세스가 종료될 때까지 대기 | ||||||
|         while (std::getline(ss, token, '\n')) { |             return tokens; | ||||||
|             tokens.push(token); |         } catch (const std::exception& e) { | ||||||
|  |             return tokens; | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |     /**  | ||||||
|  |      * @brief 명령어를 쉘에서 실행하고 결과를 파이프로 연결하여 반환합니다 | ||||||
|  |      * @param cmd 실행할 명령 | ||||||
|  |      * @param args 아규먼트 | ||||||
|  |      * @return FILE* fd | ||||||
|  |      */ | ||||||
|  |     static FILE* safe_open_pipe(const std::string& cmd, const std::vector<std::string>& args) { | ||||||
|  |         boost::process::pipe pipe; | ||||||
|  |         boost::process::child c(cmd, boost::process::args(args), boost::process::std_out > pipe); | ||||||
|  |  | ||||||
|         return tokens; |         return fdopen(pipe.native_source(), "r"); | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| } | } | ||||||
| @@ -7,7 +7,7 @@ namespace bumbleBee { | |||||||
| class VersionsCheckUtils { | class VersionsCheckUtils { | ||||||
| public: | public: | ||||||
|     static bool isThereCMD(std::shared_ptr<dpp::cluster> cluster, std::string cmd) { |     static bool isThereCMD(std::shared_ptr<dpp::cluster> cluster, std::string cmd) { | ||||||
|         if (ConsoleUtils::getResultFromCommand("which " + cmd).size() == 0) { |         if (ConsoleUtils::safe_execute_command("which", {cmd}).size() == 0) { | ||||||
|             cluster->log(dpp::ll_error, cmd + " is unavaliable. unresolable error please install " + cmd); |             cluster->log(dpp::ll_error, cmd + " is unavaliable. unresolable error please install " + cmd); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
| @@ -16,7 +16,7 @@ public: | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     static void validateFFMPEG(std::shared_ptr<dpp::cluster> cluster) { |     static void validateFFMPEG(std::shared_ptr<dpp::cluster> cluster) { | ||||||
|         std::queue<std::string> result = ConsoleUtils::getResultFromCommand(SettingsManager::getFFMPEG_CMD() + " -version"); |         std::queue<std::string> result = ConsoleUtils::safe_execute_command(SettingsManager::getFFMPEG_CMD(), {"-version"}); | ||||||
|         std::string front = result.front(); |         std::string front = result.front(); | ||||||
|         if (front[0] != 'f' || |         if (front[0] != 'f' || | ||||||
|             front[1] != 'f' || |             front[1] != 'f' || | ||||||
| @@ -43,7 +43,7 @@ public: | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     static void validateYTDLP(std::shared_ptr<dpp::cluster> cluster) { |     static void validateYTDLP(std::shared_ptr<dpp::cluster> cluster) { | ||||||
|         std::queue<std::string> result = ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + " --version"); |         std::queue<std::string> result = ConsoleUtils::safe_execute_command(SettingsManager::getYTDLP_CMD(), {"--version"}); | ||||||
|         std::string front = result.front(); |         std::string front = result.front(); | ||||||
|         if ((front[0]-'0' < 0 || front[0]-'0' > 9) || |         if ((front[0]-'0' < 0 || front[0]-'0' > 9) || | ||||||
|             (front[1]-'0' < 0 || front[1]-'0' > 9) || |             (front[1]-'0' < 0 || front[1]-'0' > 9) || | ||||||
| @@ -72,7 +72,7 @@ public: | |||||||
|  |  | ||||||
|     static void updateytdlp(std::shared_ptr<dpp::cluster> cluster) { |     static void updateytdlp(std::shared_ptr<dpp::cluster> cluster) { | ||||||
|         cluster->log(dpp::ll_info, "Checking if yt-dlp update is available..."); |         cluster->log(dpp::ll_info, "Checking if yt-dlp update is available..."); | ||||||
|         std::queue<std::string> result = ConsoleUtils::getResultFromCommand("./yt-dlp -U"); |         std::queue<std::string> result = ConsoleUtils::safe_execute_command("./yt-dlp", {"-U"}); | ||||||
|         while(!result.empty()) { |         while(!result.empty()) { | ||||||
|             std::string front = result.front(); |             std::string front = result.front(); | ||||||
|             result.pop(); |             result.pop(); | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
| #include <Settings/SettingsManager.hpp> | #include <Settings/SettingsManager.hpp> | ||||||
| #include <dpp/nlohmann/json.hpp> | #include <dpp/nlohmann/json.hpp> | ||||||
| #include <Utils/QueuedMusicListEmbedProvider.hpp> | #include <Utils/QueuedMusicListEmbedProvider.hpp> | ||||||
|  | #include <Utils/ConsoleUtils.hpp> | ||||||
| #include <variant> | #include <variant> | ||||||
|  |  | ||||||
| namespace bumbleBee::commands { | namespace bumbleBee::commands { | ||||||
| @@ -23,12 +24,20 @@ namespace bumbleBee::commands { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         std::string query = std::get<std::string>(event.get_parameter("query")); |         std::string query = std::get<std::string>(event.get_parameter("query")); | ||||||
|         query = "\"" + query + "\""; |         // query = "\"" + query + "\""; | ||||||
|  |  | ||||||
|         std::queue<std::string> ids = |         std::queue<std::string> ids = | ||||||
|             ConsoleUtils::getResultFromCommand( |             ConsoleUtils::safe_execute_command( | ||||||
|             SettingsManager::getYTDLP_CMD() + |             SettingsManager::getYTDLP_CMD(), { | ||||||
|             " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors --print id " + query); |             "--default-search", | ||||||
|  |             "ytsearch", | ||||||
|  |             "--flat-playlist", | ||||||
|  |             "--skip-download", | ||||||
|  |             "--quiet", | ||||||
|  |             "--ignore-errors", | ||||||
|  |             "--print", | ||||||
|  |             "id", | ||||||
|  |             query}); | ||||||
|  |  | ||||||
|         std::queue<std::shared_ptr<MusicQueueElement>> musics; |         std::queue<std::shared_ptr<MusicQueueElement>> musics; | ||||||
|  |  | ||||||
| @@ -46,19 +55,18 @@ namespace bumbleBee::commands { | |||||||
|                 ids.pop(); |                 ids.pop(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             FILE* file = popen((SettingsManager::getYTDLP_CMD() + |             std::string jsonData = ConsoleUtils::safe_execute_command(SettingsManager::getYTDLP_CMD(), { | ||||||
|                 " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors -J http://youtu.be/" + ids.front()).c_str(), "r"); |                 "--default-search", | ||||||
|  |                 "ytsearch", | ||||||
|  |                 "--flat-playlist", | ||||||
|  |                 "--skip-download", | ||||||
|  |                 "--quiet", | ||||||
|  |                 "--ignore-errors", | ||||||
|  |                 "-J", | ||||||
|  |                 "http://youtu.be/" + ids.front() | ||||||
|  |             }).front(); | ||||||
|  |  | ||||||
|             std::ostringstream oss; |             std::istringstream iss(jsonData); | ||||||
|             char buffer[1024]; |  | ||||||
|             size_t bytesRead; |  | ||||||
|              |  | ||||||
|             while ((bytesRead = fread(buffer, 1, sizeof(buffer), file)) > 0) { |  | ||||||
|                 oss.write(buffer, bytesRead); |  | ||||||
|             } |  | ||||||
|             pclose(file); |  | ||||||
|  |  | ||||||
|             std::istringstream iss(oss.str()); |  | ||||||
|             nlohmann::json videoDataJson; |             nlohmann::json videoDataJson; | ||||||
|             iss >> videoDataJson; |             iss >> videoDataJson; | ||||||
|  |  | ||||||
| @@ -108,19 +116,18 @@ namespace bumbleBee::commands { | |||||||
|                             continue; |                             continue; | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         FILE* file = popen((SettingsManager::getYTDLP_CMD() + |                         std::string jsonData = ConsoleUtils::safe_execute_command(SettingsManager::getYTDLP_CMD(), { | ||||||
|                             " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors -J http://youtu.be/" + ids.front()).c_str(), "r"); |                             "--default-search", | ||||||
|  |                             "ytsearch", | ||||||
|  |                             "--flat-playlist", | ||||||
|  |                             "--skip-download", | ||||||
|  |                             "--quiet", | ||||||
|  |                             "--ignore-errors", | ||||||
|  |                             "-J", | ||||||
|  |                             "http://youtu.be/" + ids.front() | ||||||
|  |                         }).front(); | ||||||
|              |              | ||||||
|                         std::ostringstream oss; |                         std::istringstream iss(jsonData); | ||||||
|                         char buffer[1024]; |  | ||||||
|                         size_t bytesRead; |  | ||||||
|                          |  | ||||||
|                         while ((bytesRead = fread(buffer, 1, sizeof(buffer), file)) > 0) { |  | ||||||
|                             oss.write(buffer, bytesRead); |  | ||||||
|                         } |  | ||||||
|                         pclose(file); |  | ||||||
|  |  | ||||||
|                         std::istringstream iss(oss.str()); |  | ||||||
|                         nlohmann::json videoDataJson; |                         nlohmann::json videoDataJson; | ||||||
|                         iss >> videoDataJson; |                         iss >> videoDataJson; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,13 +14,19 @@ bool SettingsManager::REGISTER_COMMAND = false; | |||||||
|  |  | ||||||
| bool SettingsManager::validateToken() { | bool SettingsManager::validateToken() { | ||||||
|     nlohmann::json response; |     nlohmann::json response; | ||||||
|     if (ConsoleUtils::getResultFromCommand("which curl").size() == 0) { |     std::string curl = ConsoleUtils::safe_execute_command("/usr/bin/which", {"curl"}).front(); | ||||||
|  |     if (curl == "") { | ||||||
|         std::cout << "curl is unavaliable. unresolable error please install curl." << std::endl; |         std::cout << "curl is unavaliable. unresolable error please install curl." << std::endl; | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     std::string stresult = ConsoleUtils::getResultFromCommand("curl -sX GET \"https://discord.com/api/v10/users/@me\" -H \"Authorization: Bot " + |     std::string stresult = ConsoleUtils::safe_execute_command(curl, { | ||||||
|         TOKEN + "\"").front(); |         "-sX", | ||||||
|  |         "GET", | ||||||
|  |         "https://discord.com/api/v10/users/@me", | ||||||
|  |         "-H", | ||||||
|  |         "Authorization: Bot " + TOKEN + "" | ||||||
|  |     }).front(); | ||||||
|     std::stringstream ss(stresult); |     std::stringstream ss(stresult); | ||||||
|     ss >> response; |     ss >> response; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -38,8 +38,17 @@ void AsyncDownloadManager::downloadWorker() { | |||||||
|         cluster->log(dpp::ll_info, "AsyncDownloadManager: " + query + " accepted."); |         cluster->log(dpp::ll_info, "AsyncDownloadManager: " + query + " accepted."); | ||||||
|  |  | ||||||
|         std::queue<std::string> ids = |         std::queue<std::string> ids = | ||||||
|             ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + |             ConsoleUtils::safe_execute_command( | ||||||
|             " --default-search ytsearch --flat-playlist --skip-download --quiet --ignore-errors --print id " + query); |             SettingsManager::getYTDLP_CMD(), { | ||||||
|  |             "--default-search", | ||||||
|  |             "ytsearch", | ||||||
|  |             "--flat-playlist", | ||||||
|  |             "--skip-download", | ||||||
|  |             "--quiet", | ||||||
|  |             "--ignore-errors", | ||||||
|  |             "--print", | ||||||
|  |             "id", | ||||||
|  |             query}); | ||||||
|  |  | ||||||
|         if (ids.size() >= 2) { |         if (ids.size() >= 2) { | ||||||
|             cluster->log(dpp::ll_info, query + " is playlist"); |             cluster->log(dpp::ll_info, query + " is playlist"); | ||||||
| @@ -56,8 +65,13 @@ void AsyncDownloadManager::downloadWorker() { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         std::queue<std::string> urls = |         std::queue<std::string> urls = | ||||||
|             ConsoleUtils::getResultFromCommand(SettingsManager::getYTDLP_CMD() + |             ConsoleUtils::safe_execute_command(SettingsManager::getYTDLP_CMD(), { | ||||||
|             " -f ba* --print urls https://youtu.be/" + ids.front()); |             "-f", | ||||||
|  |             "ba*", | ||||||
|  |             "--print", | ||||||
|  |             "urls", | ||||||
|  |             "https://youtu.be/" + ids.front() | ||||||
|  |         }); | ||||||
|  |  | ||||||
|         cluster->log(dpp::ll_debug, "url: " + urls.front()); |         cluster->log(dpp::ll_debug, "url: " + urls.front()); | ||||||
|  |  | ||||||
| @@ -75,8 +89,11 @@ void AsyncDownloadManager::downloadWorker() { | |||||||
|  |  | ||||||
|             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + ": " + downloadID + " accepted."); |             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + ": " + downloadID + " accepted."); | ||||||
|  |  | ||||||
|             std::string command = std::string("./streamOpus.sh " + SettingsManager::getYTDLP_CMD() + " " + downloadID + " " + SettingsManager::getFFMPEG_CMD()); |             stream = ConsoleUtils::safe_open_pipe("./streamOpus.sh", { | ||||||
|             stream = popen(command.c_str(), "r"); |                 SettingsManager::getYTDLP_CMD(), | ||||||
|  |                 downloadID, | ||||||
|  |                 SettingsManager::getFFMPEG_CMD() | ||||||
|  |             }); | ||||||
|  |  | ||||||
|             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + " Opened stream: " + downloadID); |             cluster->log(dpp::ll_info, "Thread id: " + tid.str() + " Opened stream: " + downloadID); | ||||||
|         }); |         }); | ||||||
|   | |||||||
							
								
								
									
										72
									
								
								stackcollapse-gdb.pl
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										72
									
								
								stackcollapse-gdb.pl
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,72 @@ | |||||||
|  | #!/usr/bin/perl -ws | ||||||
|  | # | ||||||
|  | # stackcollapse-gdb  Collapse GDB backtraces | ||||||
|  | # | ||||||
|  | # Parse a list of GDB backtraces as generated with the poor man's | ||||||
|  | # profiler [1]: | ||||||
|  | # | ||||||
|  | #   for x in $(seq 1 500); do | ||||||
|  | #      gdb -ex "set pagination 0" -ex "thread apply all bt" -batch -p $pid 2> /dev/null | ||||||
|  | #      sleep 0.01 | ||||||
|  | #    done | ||||||
|  | # | ||||||
|  | # [1] http://poormansprofiler.org/ | ||||||
|  | # | ||||||
|  | # Copyright 2014 Gabriel Corona. All rights reserved. | ||||||
|  | # | ||||||
|  | # CDDL HEADER START | ||||||
|  | # | ||||||
|  | # The contents of this file are subject to the terms of the | ||||||
|  | # Common Development and Distribution License (the "License"). | ||||||
|  | # You may not use this file except in compliance with the License. | ||||||
|  | # | ||||||
|  | # You can obtain a copy of the license at docs/cddl1.txt or | ||||||
|  | # http://opensource.org/licenses/CDDL-1.0. | ||||||
|  | # See the License for the specific language governing permissions | ||||||
|  | # and limitations under the License. | ||||||
|  | # | ||||||
|  | # When distributing Covered Code, include this CDDL HEADER in each | ||||||
|  | # file and include the License file at docs/cddl1.txt. | ||||||
|  | # If applicable, add the following below this CDDL HEADER, with the | ||||||
|  | # fields enclosed by brackets "[]" replaced with your own identifying | ||||||
|  | # information: Portions Copyright [yyyy] [name of copyright owner] | ||||||
|  | # | ||||||
|  | # CDDL HEADER END | ||||||
|  |  | ||||||
|  | use strict; | ||||||
|  |  | ||||||
|  | my $current = ""; | ||||||
|  | my $previous_function = ""; | ||||||
|  |  | ||||||
|  | my %stacks; | ||||||
|  |  | ||||||
|  | while(<>) { | ||||||
|  |   chomp; | ||||||
|  |   if (m/^Thread/) { | ||||||
|  |     $current="" | ||||||
|  |   } | ||||||
|  |   elsif(m/^#[0-9]* *([^ ]*) ([^ ]*) ([^ ]*) ([^ ]*)/) { | ||||||
|  |     my $function = $3; | ||||||
|  |     my $alt = $1; | ||||||
|  |     if(not($1 =~ /0x[a-zA-Z0-9]*/)) { | ||||||
|  |       $function = $alt; | ||||||
|  |     } | ||||||
|  |     if ($current eq "") { | ||||||
|  |       $current = $function; | ||||||
|  |     } else { | ||||||
|  |       $current = $function . ";" . $current; | ||||||
|  |     } | ||||||
|  |   } elsif(!($current eq "")) { | ||||||
|  |     $stacks{$current} += 1; | ||||||
|  |     $current = ""; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | if(!($current eq "")) { | ||||||
|  |   $stacks{$current} += 1; | ||||||
|  |   $current = ""; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | foreach my $k (sort { $a cmp $b } keys %stacks) { | ||||||
|  |   print "$k $stacks{$k}\n"; | ||||||
|  | } | ||||||
							
								
								
									
										435
									
								
								stackcollapse-perf.pl
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										435
									
								
								stackcollapse-perf.pl
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,435 @@ | |||||||
|  | #!/usr/bin/perl -w | ||||||
|  | # | ||||||
|  | # stackcollapse-perf.pl	collapse perf samples into single lines. | ||||||
|  | # | ||||||
|  | # Parses a list of multiline stacks generated by "perf script", and | ||||||
|  | # outputs a semicolon separated stack followed by a space and a count. | ||||||
|  | # If memory addresses (+0xd) are present, they are stripped, and resulting | ||||||
|  | # identical stacks are colased with their counts summed. | ||||||
|  | # | ||||||
|  | # USAGE: ./stackcollapse-perf.pl [options] infile > outfile | ||||||
|  | # | ||||||
|  | # Run "./stackcollapse-perf.pl -h" to list options. | ||||||
|  | # | ||||||
|  | # Example input: | ||||||
|  | # | ||||||
|  | #  swapper     0 [000] 158665.570607: cpu-clock: | ||||||
|  | #         ffffffff8103ce3b native_safe_halt ([kernel.kallsyms]) | ||||||
|  | #         ffffffff8101c6a3 default_idle ([kernel.kallsyms]) | ||||||
|  | #         ffffffff81013236 cpu_idle ([kernel.kallsyms]) | ||||||
|  | #         ffffffff815bf03e rest_init ([kernel.kallsyms]) | ||||||
|  | #         ffffffff81aebbfe start_kernel ([kernel.kallsyms].init.text) | ||||||
|  | #  [...] | ||||||
|  | # | ||||||
|  | # Example output: | ||||||
|  | # | ||||||
|  | #  swapper;start_kernel;rest_init;cpu_idle;default_idle;native_safe_halt 1 | ||||||
|  | # | ||||||
|  | # Input may be created and processed using: | ||||||
|  | # | ||||||
|  | #  perf record -a -g -F 997 sleep 60 | ||||||
|  | #  perf script | ./stackcollapse-perf.pl > out.stacks-folded | ||||||
|  | # | ||||||
|  | # The output of "perf script" should include stack traces. If these are missing | ||||||
|  | # for you, try manually selecting the perf script output; eg: | ||||||
|  | # | ||||||
|  | #  perf script -f comm,pid,tid,cpu,time,event,ip,sym,dso,trace | ... | ||||||
|  | # | ||||||
|  | # This is also required for the --pid or --tid options, so that the output has | ||||||
|  | # both the PID and TID. | ||||||
|  | # | ||||||
|  | # Copyright 2012 Joyent, Inc.  All rights reserved. | ||||||
|  | # Copyright 2012 Brendan Gregg.  All rights reserved. | ||||||
|  | # | ||||||
|  | # CDDL HEADER START | ||||||
|  | # | ||||||
|  | # The contents of this file are subject to the terms of the | ||||||
|  | # Common Development and Distribution License (the "License"). | ||||||
|  | # You may not use this file except in compliance with the License. | ||||||
|  | # | ||||||
|  | # You can obtain a copy of the license at docs/cddl1.txt or | ||||||
|  | # http://opensource.org/licenses/CDDL-1.0. | ||||||
|  | # See the License for the specific language governing permissions | ||||||
|  | # and limitations under the License. | ||||||
|  | # | ||||||
|  | # When distributing Covered Code, include this CDDL HEADER in each | ||||||
|  | # file and include the License file at docs/cddl1.txt. | ||||||
|  | # If applicable, add the following below this CDDL HEADER, with the | ||||||
|  | # fields enclosed by brackets "[]" replaced with your own identifying | ||||||
|  | # information: Portions Copyright [yyyy] [name of copyright owner] | ||||||
|  | # | ||||||
|  | # CDDL HEADER END | ||||||
|  | # | ||||||
|  | # 02-Mar-2012	Brendan Gregg	Created this. | ||||||
|  | # 02-Jul-2014	   "	  "	Added process name to stacks. | ||||||
|  |  | ||||||
|  | use strict; | ||||||
|  | use Getopt::Long; | ||||||
|  |  | ||||||
|  | my %collapsed; | ||||||
|  |  | ||||||
|  | sub remember_stack { | ||||||
|  | 	my ($stack, $count) = @_; | ||||||
|  | 	$collapsed{$stack} += $count; | ||||||
|  | } | ||||||
|  | my $annotate_kernel = 0; # put an annotation on kernel function | ||||||
|  | my $annotate_jit = 0;   # put an annotation on jit symbols | ||||||
|  | my $annotate_all = 0;   # enale all annotations | ||||||
|  | my $include_pname = 1;	# include process names in stacks | ||||||
|  | my $include_pid = 0;	# include process ID with process name | ||||||
|  | my $include_tid = 0;	# include process & thread ID with process name | ||||||
|  | my $include_addrs = 0;	# include raw address where a symbol can't be found | ||||||
|  | my $tidy_java = 1;	# condense Java signatures | ||||||
|  | my $tidy_generic = 1;	# clean up function names a little | ||||||
|  | my $target_pname;	# target process name from perf invocation | ||||||
|  | my $event_filter = "";    # event type filter, defaults to first encountered event | ||||||
|  | my $event_defaulted = 0;  # whether we defaulted to an event (none provided) | ||||||
|  | my $event_warning = 0;	  # if we printed a warning for the event | ||||||
|  |  | ||||||
|  | my $show_inline = 0; | ||||||
|  | my $show_context = 0; | ||||||
|  |  | ||||||
|  | my $srcline_in_input = 0; # if there are extra lines with source location (perf script -F+srcline) | ||||||
|  | GetOptions('inline' => \$show_inline, | ||||||
|  |            'context' => \$show_context, | ||||||
|  |            'srcline' => \$srcline_in_input, | ||||||
|  |            'pid' => \$include_pid, | ||||||
|  |            'kernel' => \$annotate_kernel, | ||||||
|  |            'jit' => \$annotate_jit, | ||||||
|  |            'all' => \$annotate_all, | ||||||
|  |            'tid' => \$include_tid, | ||||||
|  |            'addrs' => \$include_addrs, | ||||||
|  |            'event-filter=s' => \$event_filter) | ||||||
|  | or die <<USAGE_END; | ||||||
|  | USAGE: $0 [options] infile > outfile\n | ||||||
|  | 	--pid		# include PID with process names [1] | ||||||
|  | 	--tid		# include TID and PID with process names [1] | ||||||
|  | 	--inline	# un-inline using addr2line | ||||||
|  | 	--all		# all annotations (--kernel --jit) | ||||||
|  | 	--kernel	# annotate kernel functions with a _[k] | ||||||
|  | 	--jit		# annotate jit functions with a _[j] | ||||||
|  | 	--context	# adds source context to --inline | ||||||
|  | 	--srcline	# parses output of 'perf script -F+srcline' and adds source context | ||||||
|  | 	--addrs		# include raw addresses where symbols can't be found | ||||||
|  | 	--event-filter=EVENT	# event name filter\n | ||||||
|  | [1] perf script must emit both PID and TIDs for these to work; eg, Linux < 4.1: | ||||||
|  | 	perf script -f comm,pid,tid,cpu,time,event,ip,sym,dso,trace | ||||||
|  |     for Linux >= 4.1: | ||||||
|  | 	perf script -F comm,pid,tid,cpu,time,event,ip,sym,dso,trace | ||||||
|  |     If you save this output add --header on Linux >= 3.14 to include perf info. | ||||||
|  | USAGE_END | ||||||
|  |  | ||||||
|  | if ($annotate_all) { | ||||||
|  | 	$annotate_kernel = $annotate_jit = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | my %inlineCache; | ||||||
|  |  | ||||||
|  | my %nmCache; | ||||||
|  |  | ||||||
|  | sub inlineCacheAdd { | ||||||
|  | 	my ($pc, $mod, $result) = @_; | ||||||
|  |    if (defined($inlineCache{$pc})) { | ||||||
|  |       $inlineCache{$pc}{$mod} = $result; | ||||||
|  |    } else { | ||||||
|  |       $inlineCache{$pc} = {$mod => $result}; | ||||||
|  |    } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # for the --inline option | ||||||
|  | sub inline { | ||||||
|  | 	my ($pc, $rawfunc, $mod) = @_; | ||||||
|  |  | ||||||
|  | 	return $inlineCache{$pc}{$mod} if defined($inlineCache{$pc}{$mod}); | ||||||
|  |  | ||||||
|  | 	# capture addr2line output | ||||||
|  | 	my $a2l_output = `addr2line -a $pc -e $mod -i -f -s -C`; | ||||||
|  |  | ||||||
|  | 	# remove first line | ||||||
|  | 	$a2l_output =~ s/^(.*\n){1}//; | ||||||
|  |  | ||||||
|  | 	if ($a2l_output =~ /\?\?\n\?\?:0/) { | ||||||
|  | 		# if addr2line fails and rawfunc is func+offset, then fall back to it | ||||||
|  | 		if ($rawfunc =~ /^(.+)\+0x([0-9a-f]+)$/) { | ||||||
|  | 			my $func = $1; | ||||||
|  | 			my $addr = hex $2; | ||||||
|  |  | ||||||
|  | 			$nmCache{$mod}=`nm $mod` unless defined $nmCache{$mod}; | ||||||
|  |  | ||||||
|  | 			if ($nmCache{$mod} =~ /^([0-9a-f]+) . \Q$func\E$/m) { | ||||||
|  | 			   my $base = hex $1; | ||||||
|  | 				my $newPc = sprintf "0x%x", $base+$addr; | ||||||
|  | 				my $result = inline($newPc, '', $mod); | ||||||
|  | 				inlineCacheAdd($pc, $mod, $result); | ||||||
|  | 				return $result; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	my @fullfunc; | ||||||
|  | 	my $one_item = ""; | ||||||
|  | 	for (split /^/, $a2l_output) { | ||||||
|  | 		chomp $_; | ||||||
|  |  | ||||||
|  | 		# remove discriminator info if exists | ||||||
|  | 		$_ =~ s/ \(discriminator \S+\)//; | ||||||
|  |  | ||||||
|  | 		if ($one_item eq "") { | ||||||
|  | 			$one_item = $_; | ||||||
|  | 		} else { | ||||||
|  | 			if ($show_context == 1) { | ||||||
|  | 				unshift @fullfunc, $one_item . ":$_"; | ||||||
|  | 			} else { | ||||||
|  | 				unshift @fullfunc, $one_item; | ||||||
|  | 			} | ||||||
|  | 			$one_item = ""; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	my $result = join ";" , @fullfunc; | ||||||
|  |  | ||||||
|  | 	inlineCacheAdd($pc, $mod, $result); | ||||||
|  |  | ||||||
|  | 	return $result; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | my @stack; | ||||||
|  | my $pname; | ||||||
|  | my $m_pid; | ||||||
|  | my $m_tid; | ||||||
|  | my $m_period; | ||||||
|  |  | ||||||
|  | # | ||||||
|  | # Main loop | ||||||
|  | # | ||||||
|  | while (defined($_ = <>)) { | ||||||
|  |  | ||||||
|  | 	# find the name of the process launched by perf, by stepping backwards | ||||||
|  | 	# over the args to find the first non-option (no dash): | ||||||
|  | 	if (/^# cmdline/) { | ||||||
|  | 		my @args = split ' ', $_; | ||||||
|  | 		foreach my $arg (reverse @args) { | ||||||
|  | 			if ($arg !~ /^-/) { | ||||||
|  | 				$target_pname = $arg; | ||||||
|  | 				$target_pname =~ s:.*/::;  # strip pathname | ||||||
|  | 				last; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	# skip remaining comments | ||||||
|  | 	next if m/^#/; | ||||||
|  | 	chomp; | ||||||
|  |  | ||||||
|  | 	# end of stack. save cached data. | ||||||
|  | 	if (m/^$/) { | ||||||
|  | 		# ignore filtered samples | ||||||
|  | 		next if not $pname; | ||||||
|  |  | ||||||
|  | 		if ($include_pname) { | ||||||
|  | 			if (defined $pname) { | ||||||
|  | 				unshift @stack, $pname; | ||||||
|  | 			} else { | ||||||
|  | 				unshift @stack, ""; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		remember_stack(join(";", @stack), $m_period) if @stack; | ||||||
|  | 		undef @stack; | ||||||
|  | 		undef $pname; | ||||||
|  | 		next; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	# | ||||||
|  | 	# event record start | ||||||
|  | 	# | ||||||
|  | 	if (/^(\S.+?)\s+(\d+)\/*(\d+)*\s+/) { | ||||||
|  | 		# default "perf script" output has TID but not PID | ||||||
|  | 		# eg, "java 25607 4794564.109216: 1 cycles:" | ||||||
|  | 		# eg, "java 12688 [002] 6544038.708352: 235 cpu-clock:" | ||||||
|  | 		# eg, "V8 WorkerThread 25607 4794564.109216: 104345 cycles:" | ||||||
|  | 		# eg, "java 24636/25607 [000] 4794564.109216: 1 cycles:" | ||||||
|  | 		# eg, "java 12688/12764 6544038.708352: 10309278 cpu-clock:" | ||||||
|  | 		# eg, "V8 WorkerThread 24636/25607 [000] 94564.109216: 100 cycles:" | ||||||
|  | 		# other combinations possible | ||||||
|  | 		my ($comm, $pid, $tid, $period) = ($1, $2, $3, ""); | ||||||
|  | 		if (not $tid) { | ||||||
|  | 			$tid = $pid; | ||||||
|  | 			$pid = "?"; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (/:\s*(\d+)*\s+(\S+):\s*$/) { | ||||||
|  | 			$period = $1; | ||||||
|  | 			my $event = $2; | ||||||
|  |  | ||||||
|  | 			if ($event_filter eq "") { | ||||||
|  | 				# By default only show events of the first encountered | ||||||
|  | 				# event type. Merging together different types, such as | ||||||
|  | 				# instructions and cycles, produces misleading results. | ||||||
|  | 				$event_filter = $event; | ||||||
|  | 				$event_defaulted = 1; | ||||||
|  | 			} elsif ($event ne $event_filter) { | ||||||
|  | 				if ($event_defaulted and $event_warning == 0) { | ||||||
|  | 					# only print this warning if necessary: | ||||||
|  | 					# when we defaulted and there was | ||||||
|  | 					# multiple event types. | ||||||
|  | 					print STDERR "Filtering for events of type: $event\n"; | ||||||
|  | 					$event_warning = 1; | ||||||
|  | 				} | ||||||
|  | 				next; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (not $period) { | ||||||
|  | 			$period = 1 | ||||||
|  | 		} | ||||||
|  | 		($m_pid, $m_tid, $m_period) = ($pid, $tid, $period); | ||||||
|  |  | ||||||
|  | 		if ($include_tid) { | ||||||
|  | 			$pname = "$comm-$m_pid/$m_tid"; | ||||||
|  | 		} elsif ($include_pid) { | ||||||
|  | 			$pname = "$comm-$m_pid"; | ||||||
|  | 		} else { | ||||||
|  | 			$pname = "$comm"; | ||||||
|  | 		} | ||||||
|  | 		$pname =~ tr/ /_/; | ||||||
|  |  | ||||||
|  | 	# | ||||||
|  | 	# stack line | ||||||
|  | 	# | ||||||
|  | 	} elsif (/^\s*(\w+)\s*(.+) \((.*)\)/) { | ||||||
|  | 		# ignore filtered samples | ||||||
|  | 		next if not $pname; | ||||||
|  |  | ||||||
|  | 		my ($pc, $rawfunc, $mod) = ($1, $2, $3); | ||||||
|  |  | ||||||
|  | 		if ($show_inline == 1 && $mod !~ m/(perf-\d+.map|kernel\.|\[[^\]]+\])/) { | ||||||
|  | 			my $inlineRes = inline($pc, $rawfunc, $mod); | ||||||
|  | 			# - empty result this happens e.g., when $mod does not exist or is a path to a compressed kernel module | ||||||
|  | 			#   if this happens, the user will see error message from addr2line written to stderr | ||||||
|  | 			# - if addr2line results in "??" , then it's much more sane to fall back than produce a '??' in graph | ||||||
|  | 			if($inlineRes ne "" and $inlineRes ne "??" and $inlineRes ne "??:??:0" ) { | ||||||
|  | 				unshift @stack, $inlineRes; | ||||||
|  | 				next; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		# Linux 4.8 included symbol offsets in perf script output by default, eg: | ||||||
|  | 		# 7fffb84c9afc cpu_startup_entry+0x800047c022ec ([kernel.kallsyms]) | ||||||
|  | 		# strip these off: | ||||||
|  | 		$rawfunc =~ s/\+0x[\da-f]+$//; | ||||||
|  |  | ||||||
|  | 		next if $rawfunc =~ /^\(/;		# skip process names | ||||||
|  |  | ||||||
|  | 		my $is_unknown=0; | ||||||
|  | 		my @inline; | ||||||
|  | 		for (split /\->/, $rawfunc) { | ||||||
|  | 			my $func = $_; | ||||||
|  |  | ||||||
|  | 			if ($func eq "[unknown]") { | ||||||
|  | 				if ($mod ne "[unknown]") { # use module name instead, if known | ||||||
|  | 					$func = $mod; | ||||||
|  | 					$func =~ s/.*\///; | ||||||
|  | 				} else { | ||||||
|  | 					$func = "unknown"; | ||||||
|  | 					$is_unknown=1; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				if ($include_addrs) { | ||||||
|  | 					$func = "\[$func \<$pc\>\]"; | ||||||
|  | 				} else { | ||||||
|  | 					$func = "\[$func\]"; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if ($tidy_generic) { | ||||||
|  | 				$func =~ s/;/:/g; | ||||||
|  | 				if ($func !~ m/\.\(.*\)\./) { | ||||||
|  | 					# This doesn't look like a Go method name (such as | ||||||
|  | 					# "net/http.(*Client).Do"), so everything after the first open | ||||||
|  | 					# paren (that is not part of an "(anonymous namespace)") is | ||||||
|  | 					# just noise. | ||||||
|  | 					$func =~ s/\((?!anonymous namespace\)).*//; | ||||||
|  | 				} | ||||||
|  | 				# now tidy this horrible thing: | ||||||
|  | 				# 13a80b608e0a RegExp:[&<>\"\'] (/tmp/perf-7539.map) | ||||||
|  | 				$func =~ tr/"\'//d; | ||||||
|  | 				# fall through to $tidy_java | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if ($tidy_java and $pname =~ m/^java/) { | ||||||
|  | 				# along with $tidy_generic, converts the following: | ||||||
|  | 				#	Lorg/mozilla/javascript/ContextFactory;.call(Lorg/mozilla/javascript/ContextAction;)Ljava/lang/Object; | ||||||
|  | 				#	Lorg/mozilla/javascript/ContextFactory;.call(Lorg/mozilla/javascript/C | ||||||
|  | 				#	Lorg/mozilla/javascript/MemberBox;.<init>(Ljava/lang/reflect/Method;)V | ||||||
|  | 				# into: | ||||||
|  | 				#	org/mozilla/javascript/ContextFactory:.call | ||||||
|  | 				#	org/mozilla/javascript/ContextFactory:.call | ||||||
|  | 				#	org/mozilla/javascript/MemberBox:.init | ||||||
|  | 				$func =~ s/^L// if $func =~ m:/:; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			# | ||||||
|  | 			# Annotations | ||||||
|  | 			# | ||||||
|  | 			# detect inlined from the @inline array | ||||||
|  | 			# detect kernel from the module name; eg, frames to parse include: | ||||||
|  | 			#          ffffffff8103ce3b native_safe_halt ([kernel.kallsyms])  | ||||||
|  | 			#          8c3453 tcp_sendmsg (/lib/modules/4.3.0-rc1-virtual/build/vmlinux) | ||||||
|  | 			#          7d8 ipv4_conntrack_local+0x7f8f80b8 ([nf_conntrack_ipv4]) | ||||||
|  | 			# detect jit from the module name; eg: | ||||||
|  | 			#          7f722d142778 Ljava/io/PrintStream;::print (/tmp/perf-19982.map) | ||||||
|  | 			if (scalar(@inline) > 0) { | ||||||
|  | 				$func .= "_[i]" unless $func =~ m/\_\[i\]/;	# inlined | ||||||
|  | 			} elsif ($annotate_kernel == 1 && $mod =~ m/(^\[|vmlinux$)/ && $mod !~ /unknown/) { | ||||||
|  | 				$func .= "_[k]";	# kernel | ||||||
|  | 			} elsif ($annotate_jit == 1 && $mod =~ m:/tmp/perf-\d+\.map:) { | ||||||
|  | 				$func .= "_[j]" unless $func =~ m/\_\[j\]/;	# jitted | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			# | ||||||
|  | 			# Source lines | ||||||
|  | 			# | ||||||
|  | 			# | ||||||
|  | 			# Sample outputs: | ||||||
|  | 			#   | a.out 35081 252436.005167:     667783 cycles: | ||||||
|  | 			#   |                   408ebb some_method_name+0x8b (/full/path/to/a.out) | ||||||
|  | 			#   |   uniform_int_dist.h:300 | ||||||
|  | 			#   |                   4069f5 main+0x935 (/full/path/to/a.out) | ||||||
|  | 			#   |   file.cpp:137 | ||||||
|  | 			#   |             7f6d2148eb25 __libc_start_main+0xd5 (/lib64/libc-2.33.so) | ||||||
|  | 			#   |   libc-2.33.so[27b25] | ||||||
|  | 			# | ||||||
|  | 			#   | a.out 35081 252435.738165:     306459 cycles: | ||||||
|  | 			#   |             7f6d213c2750 [unknown] (/usr/lib64/libkmod.so.2.3.6) | ||||||
|  | 			#   |   libkmod.so.2.3.6[6750] | ||||||
|  | 			# | ||||||
|  | 			#   | a.out 35081 252435.738373:     315813 cycles: | ||||||
|  | 			#   |             7f6d215ca51b __strlen_avx2+0x4b (/lib64/libc-2.33.so) | ||||||
|  | 			#   |   libc-2.33.so[16351b] | ||||||
|  | 			#   |             7ffc71ee9580 [unknown] ([unknown])			 | ||||||
|  | 			#   | | ||||||
|  | 			# | ||||||
|  | 			#   | a.out 35081 252435.718940:     247984 cycles: | ||||||
|  | 			#   |         ffffffff814f9302 up_write+0x32 ([kernel.kallsyms]) | ||||||
|  | 			#   |   [kernel.kallsyms][ffffffff814f9302] | ||||||
|  | 			if($srcline_in_input and not $is_unknown){ | ||||||
|  | 				$_ = <>; | ||||||
|  | 				chomp; | ||||||
|  | 				s/\[.*?\]//g; | ||||||
|  | 				s/^\s*//g; | ||||||
|  | 				s/\s*$//g; | ||||||
|  | 				$func.=':'.$_ unless $_ eq ""; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			push @inline, $func; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		unshift @stack, @inline; | ||||||
|  | 	} else { | ||||||
|  | 		warn "Unrecognized line: $_"; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | foreach my $k (sort { $a cmp $b } keys %collapsed) { | ||||||
|  | 	print "$k $collapsed{$k}\n"; | ||||||
|  | } | ||||||
							
								
								
									
										862
									
								
								stackflame.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										862
									
								
								stackflame.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,862 @@ | |||||||
|  | <?xml version="1.0" standalone="no"?> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||||
|  | <svg version="1.1" width="1200" height="390" onload="init(evt)" viewBox="0 0 1200 390" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  | <!-- Flame graph stack visualization. See https://github.com/brendangregg/FlameGraph for latest version, and http://www.brendangregg.com/flamegraphs.html for examples. --> | ||||||
|  | <!-- NOTES:  --> | ||||||
|  | <defs> | ||||||
|  | 	<linearGradient id="background" y1="0" y2="1" x1="0" x2="0" > | ||||||
|  | 		<stop stop-color="#eeeeee" offset="5%" /> | ||||||
|  | 		<stop stop-color="#eeeeb0" offset="95%" /> | ||||||
|  | 	</linearGradient> | ||||||
|  | </defs> | ||||||
|  | <style type="text/css"> | ||||||
|  | 	text { font-family:Verdana; font-size:12px; fill:rgb(0,0,0); } | ||||||
|  | 	#search, #ignorecase { opacity:0.1; cursor:pointer; } | ||||||
|  | 	#search:hover, #search.show, #ignorecase:hover, #ignorecase.show { opacity:1; } | ||||||
|  | 	#subtitle { text-anchor:middle; font-color:rgb(160,160,160); } | ||||||
|  | 	#title { text-anchor:middle; font-size:17px} | ||||||
|  | 	#unzoom { cursor:pointer; } | ||||||
|  | 	#frames > *:hover { stroke:black; stroke-width:0.5; cursor:pointer; } | ||||||
|  | 	.hide { display:none; } | ||||||
|  | 	.parent { opacity:0.5; } | ||||||
|  | </style> | ||||||
|  | <script type="text/ecmascript"> | ||||||
|  | <![CDATA[ | ||||||
|  | 	"use strict"; | ||||||
|  | 	var details, searchbtn, unzoombtn, matchedtxt, svg, searching, currentSearchTerm, ignorecase, ignorecaseBtn; | ||||||
|  | 	function init(evt) { | ||||||
|  | 		details = document.getElementById("details").firstChild; | ||||||
|  | 		searchbtn = document.getElementById("search"); | ||||||
|  | 		ignorecaseBtn = document.getElementById("ignorecase"); | ||||||
|  | 		unzoombtn = document.getElementById("unzoom"); | ||||||
|  | 		matchedtxt = document.getElementById("matched"); | ||||||
|  | 		svg = document.getElementsByTagName("svg")[0]; | ||||||
|  | 		searching = 0; | ||||||
|  | 		currentSearchTerm = null; | ||||||
|  |  | ||||||
|  | 		// use GET parameters to restore a flamegraphs state. | ||||||
|  | 		var params = get_params(); | ||||||
|  | 		if (params.x && params.y) | ||||||
|  | 			zoom(find_group(document.querySelector('[x="' + params.x + '"][y="' + params.y + '"]'))); | ||||||
|  |                 if (params.s) search(params.s); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// event listeners | ||||||
|  | 	window.addEventListener("click", function(e) { | ||||||
|  | 		var target = find_group(e.target); | ||||||
|  | 		if (target) { | ||||||
|  | 			if (target.nodeName == "a") { | ||||||
|  | 				if (e.ctrlKey === false) return; | ||||||
|  | 				e.preventDefault(); | ||||||
|  | 			} | ||||||
|  | 			if (target.classList.contains("parent")) unzoom(true); | ||||||
|  | 			zoom(target); | ||||||
|  | 			if (!document.querySelector('.parent')) { | ||||||
|  | 				// we have basically done a clearzoom so clear the url | ||||||
|  | 				var params = get_params(); | ||||||
|  | 				if (params.x) delete params.x; | ||||||
|  | 				if (params.y) delete params.y; | ||||||
|  | 				history.replaceState(null, null, parse_params(params)); | ||||||
|  | 				unzoombtn.classList.add("hide"); | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			// set parameters for zoom state | ||||||
|  | 			var el = target.querySelector("rect"); | ||||||
|  | 			if (el && el.attributes && el.attributes.y && el.attributes._orig_x) { | ||||||
|  | 				var params = get_params() | ||||||
|  | 				params.x = el.attributes._orig_x.value; | ||||||
|  | 				params.y = el.attributes.y.value; | ||||||
|  | 				history.replaceState(null, null, parse_params(params)); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		else if (e.target.id == "unzoom") clearzoom(); | ||||||
|  | 		else if (e.target.id == "search") search_prompt(); | ||||||
|  | 		else if (e.target.id == "ignorecase") toggle_ignorecase(); | ||||||
|  | 	}, false) | ||||||
|  |  | ||||||
|  | 	// mouse-over for info | ||||||
|  | 	// show | ||||||
|  | 	window.addEventListener("mouseover", function(e) { | ||||||
|  | 		var target = find_group(e.target); | ||||||
|  | 		if (target) details.nodeValue = "Function: " + g_to_text(target); | ||||||
|  | 	}, false) | ||||||
|  |  | ||||||
|  | 	// clear | ||||||
|  | 	window.addEventListener("mouseout", function(e) { | ||||||
|  | 		var target = find_group(e.target); | ||||||
|  | 		if (target) details.nodeValue = ' '; | ||||||
|  | 	}, false) | ||||||
|  |  | ||||||
|  | 	// ctrl-F for search | ||||||
|  | 	// ctrl-I to toggle case-sensitive search | ||||||
|  | 	window.addEventListener("keydown",function (e) { | ||||||
|  | 		if (e.keyCode === 114 || (e.ctrlKey && e.keyCode === 70)) { | ||||||
|  | 			e.preventDefault(); | ||||||
|  | 			search_prompt(); | ||||||
|  | 		} | ||||||
|  | 		else if (e.ctrlKey && e.keyCode === 73) { | ||||||
|  | 			e.preventDefault(); | ||||||
|  | 			toggle_ignorecase(); | ||||||
|  | 		} | ||||||
|  | 	}, false) | ||||||
|  |  | ||||||
|  | 	// functions | ||||||
|  | 	function get_params() { | ||||||
|  | 		var params = {}; | ||||||
|  | 		var paramsarr = window.location.search.substr(1).split('&'); | ||||||
|  | 		for (var i = 0; i < paramsarr.length; ++i) { | ||||||
|  | 			var tmp = paramsarr[i].split("="); | ||||||
|  | 			if (!tmp[0] || !tmp[1]) continue; | ||||||
|  | 			params[tmp[0]]  = decodeURIComponent(tmp[1]); | ||||||
|  | 		} | ||||||
|  | 		return params; | ||||||
|  | 	} | ||||||
|  | 	function parse_params(params) { | ||||||
|  | 		var uri = "?"; | ||||||
|  | 		for (var key in params) { | ||||||
|  | 			uri += key + '=' + encodeURIComponent(params[key]) + '&'; | ||||||
|  | 		} | ||||||
|  | 		if (uri.slice(-1) == "&") | ||||||
|  | 			uri = uri.substring(0, uri.length - 1); | ||||||
|  | 		if (uri == '?') | ||||||
|  | 			uri = window.location.href.split('?')[0]; | ||||||
|  | 		return uri; | ||||||
|  | 	} | ||||||
|  | 	function find_child(node, selector) { | ||||||
|  | 		var children = node.querySelectorAll(selector); | ||||||
|  | 		if (children.length) return children[0]; | ||||||
|  | 	} | ||||||
|  | 	function find_group(node) { | ||||||
|  | 		var parent = node.parentElement; | ||||||
|  | 		if (!parent) return; | ||||||
|  | 		if (parent.id == "frames") return node; | ||||||
|  | 		return find_group(parent); | ||||||
|  | 	} | ||||||
|  | 	function orig_save(e, attr, val) { | ||||||
|  | 		if (e.attributes["_orig_" + attr] != undefined) return; | ||||||
|  | 		if (e.attributes[attr] == undefined) return; | ||||||
|  | 		if (val == undefined) val = e.attributes[attr].value; | ||||||
|  | 		e.setAttribute("_orig_" + attr, val); | ||||||
|  | 	} | ||||||
|  | 	function orig_load(e, attr) { | ||||||
|  | 		if (e.attributes["_orig_"+attr] == undefined) return; | ||||||
|  | 		e.attributes[attr].value = e.attributes["_orig_" + attr].value; | ||||||
|  | 		e.removeAttribute("_orig_"+attr); | ||||||
|  | 	} | ||||||
|  | 	function g_to_text(e) { | ||||||
|  | 		var text = find_child(e, "title").firstChild.nodeValue; | ||||||
|  | 		return (text) | ||||||
|  | 	} | ||||||
|  | 	function g_to_func(e) { | ||||||
|  | 		var func = g_to_text(e); | ||||||
|  | 		// if there's any manipulation we want to do to the function | ||||||
|  | 		// name before it's searched, do it here before returning. | ||||||
|  | 		return (func); | ||||||
|  | 	} | ||||||
|  | 	function update_text(e) { | ||||||
|  | 		var r = find_child(e, "rect"); | ||||||
|  | 		var t = find_child(e, "text"); | ||||||
|  | 		var w = parseFloat(r.attributes.width.value) -3; | ||||||
|  | 		var txt = find_child(e, "title").textContent.replace(/\([^(]*\)$/,""); | ||||||
|  | 		t.attributes.x.value = parseFloat(r.attributes.x.value) + 3; | ||||||
|  |  | ||||||
|  | 		// Smaller than this size won't fit anything | ||||||
|  | 		if (w < 2 * 12 * 0.59) { | ||||||
|  | 			t.textContent = ""; | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		t.textContent = txt; | ||||||
|  | 		var sl = t.getSubStringLength(0, txt.length); | ||||||
|  | 		// check if only whitespace or if we can fit the entire string into width w | ||||||
|  | 		if (/^ *$/.test(txt) || sl < w) | ||||||
|  | 			return; | ||||||
|  |  | ||||||
|  | 		// this isn't perfect, but gives a good starting point | ||||||
|  | 		// and avoids calling getSubStringLength too often | ||||||
|  | 		var start = Math.floor((w/sl) * txt.length); | ||||||
|  | 		for (var x = start; x > 0; x = x-2) { | ||||||
|  | 			if (t.getSubStringLength(0, x + 2) <= w) { | ||||||
|  | 				t.textContent = txt.substring(0, x) + ".."; | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		t.textContent = ""; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// zoom | ||||||
|  | 	function zoom_reset(e) { | ||||||
|  | 		if (e.attributes != undefined) { | ||||||
|  | 			orig_load(e, "x"); | ||||||
|  | 			orig_load(e, "width"); | ||||||
|  | 		} | ||||||
|  | 		if (e.childNodes == undefined) return; | ||||||
|  | 		for (var i = 0, c = e.childNodes; i < c.length; i++) { | ||||||
|  | 			zoom_reset(c[i]); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function zoom_child(e, x, ratio) { | ||||||
|  | 		if (e.attributes != undefined) { | ||||||
|  | 			if (e.attributes.x != undefined) { | ||||||
|  | 				orig_save(e, "x"); | ||||||
|  | 				e.attributes.x.value = (parseFloat(e.attributes.x.value) - x - 10) * ratio + 10; | ||||||
|  | 				if (e.tagName == "text") | ||||||
|  | 					e.attributes.x.value = find_child(e.parentNode, "rect[x]").attributes.x.value + 3; | ||||||
|  | 			} | ||||||
|  | 			if (e.attributes.width != undefined) { | ||||||
|  | 				orig_save(e, "width"); | ||||||
|  | 				e.attributes.width.value = parseFloat(e.attributes.width.value) * ratio; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (e.childNodes == undefined) return; | ||||||
|  | 		for (var i = 0, c = e.childNodes; i < c.length; i++) { | ||||||
|  | 			zoom_child(c[i], x - 10, ratio); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function zoom_parent(e) { | ||||||
|  | 		if (e.attributes) { | ||||||
|  | 			if (e.attributes.x != undefined) { | ||||||
|  | 				orig_save(e, "x"); | ||||||
|  | 				e.attributes.x.value = 10; | ||||||
|  | 			} | ||||||
|  | 			if (e.attributes.width != undefined) { | ||||||
|  | 				orig_save(e, "width"); | ||||||
|  | 				e.attributes.width.value = parseInt(svg.width.baseVal.value) - (10 * 2); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (e.childNodes == undefined) return; | ||||||
|  | 		for (var i = 0, c = e.childNodes; i < c.length; i++) { | ||||||
|  | 			zoom_parent(c[i]); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function zoom(node) { | ||||||
|  | 		var attr = find_child(node, "rect").attributes; | ||||||
|  | 		var width = parseFloat(attr.width.value); | ||||||
|  | 		var xmin = parseFloat(attr.x.value); | ||||||
|  | 		var xmax = parseFloat(xmin + width); | ||||||
|  | 		var ymin = parseFloat(attr.y.value); | ||||||
|  | 		var ratio = (svg.width.baseVal.value - 2 * 10) / width; | ||||||
|  |  | ||||||
|  | 		// XXX: Workaround for JavaScript float issues (fix me) | ||||||
|  | 		var fudge = 0.0001; | ||||||
|  |  | ||||||
|  | 		unzoombtn.classList.remove("hide"); | ||||||
|  |  | ||||||
|  | 		var el = document.getElementById("frames").children; | ||||||
|  | 		for (var i = 0; i < el.length; i++) { | ||||||
|  | 			var e = el[i]; | ||||||
|  | 			var a = find_child(e, "rect").attributes; | ||||||
|  | 			var ex = parseFloat(a.x.value); | ||||||
|  | 			var ew = parseFloat(a.width.value); | ||||||
|  | 			var upstack; | ||||||
|  | 			// Is it an ancestor | ||||||
|  | 			if (0 == 0) { | ||||||
|  | 				upstack = parseFloat(a.y.value) > ymin; | ||||||
|  | 			} else { | ||||||
|  | 				upstack = parseFloat(a.y.value) < ymin; | ||||||
|  | 			} | ||||||
|  | 			if (upstack) { | ||||||
|  | 				// Direct ancestor | ||||||
|  | 				if (ex <= xmin && (ex+ew+fudge) >= xmax) { | ||||||
|  | 					e.classList.add("parent"); | ||||||
|  | 					zoom_parent(e); | ||||||
|  | 					update_text(e); | ||||||
|  | 				} | ||||||
|  | 				// not in current path | ||||||
|  | 				else | ||||||
|  | 					e.classList.add("hide"); | ||||||
|  | 			} | ||||||
|  | 			// Children maybe | ||||||
|  | 			else { | ||||||
|  | 				// no common path | ||||||
|  | 				if (ex < xmin || ex + fudge >= xmax) { | ||||||
|  | 					e.classList.add("hide"); | ||||||
|  | 				} | ||||||
|  | 				else { | ||||||
|  | 					zoom_child(e, xmin, ratio); | ||||||
|  | 					update_text(e); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		search(); | ||||||
|  | 	} | ||||||
|  | 	function unzoom(dont_update_text) { | ||||||
|  | 		unzoombtn.classList.add("hide"); | ||||||
|  | 		var el = document.getElementById("frames").children; | ||||||
|  | 		for(var i = 0; i < el.length; i++) { | ||||||
|  | 			el[i].classList.remove("parent"); | ||||||
|  | 			el[i].classList.remove("hide"); | ||||||
|  | 			zoom_reset(el[i]); | ||||||
|  | 			if(!dont_update_text) update_text(el[i]); | ||||||
|  | 		} | ||||||
|  | 		search(); | ||||||
|  | 	} | ||||||
|  | 	function clearzoom() { | ||||||
|  | 		unzoom(); | ||||||
|  |  | ||||||
|  | 		// remove zoom state | ||||||
|  | 		var params = get_params(); | ||||||
|  | 		if (params.x) delete params.x; | ||||||
|  | 		if (params.y) delete params.y; | ||||||
|  | 		history.replaceState(null, null, parse_params(params)); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// search | ||||||
|  | 	function toggle_ignorecase() { | ||||||
|  | 		ignorecase = !ignorecase; | ||||||
|  | 		if (ignorecase) { | ||||||
|  | 			ignorecaseBtn.classList.add("show"); | ||||||
|  | 		} else { | ||||||
|  | 			ignorecaseBtn.classList.remove("show"); | ||||||
|  | 		} | ||||||
|  | 		reset_search(); | ||||||
|  | 		search(); | ||||||
|  | 	} | ||||||
|  | 	function reset_search() { | ||||||
|  | 		var el = document.querySelectorAll("#frames rect"); | ||||||
|  | 		for (var i = 0; i < el.length; i++) { | ||||||
|  | 			orig_load(el[i], "fill") | ||||||
|  | 		} | ||||||
|  | 		var params = get_params(); | ||||||
|  | 		delete params.s; | ||||||
|  | 		history.replaceState(null, null, parse_params(params)); | ||||||
|  | 	} | ||||||
|  | 	function search_prompt() { | ||||||
|  | 		if (!searching) { | ||||||
|  | 			var term = prompt("Enter a search term (regexp " + | ||||||
|  | 			    "allowed, eg: ^ext4_)" | ||||||
|  | 			    + (ignorecase ? ", ignoring case" : "") | ||||||
|  | 			    + "\nPress Ctrl-i to toggle case sensitivity", ""); | ||||||
|  | 			if (term != null) search(term); | ||||||
|  | 		} else { | ||||||
|  | 			reset_search(); | ||||||
|  | 			searching = 0; | ||||||
|  | 			currentSearchTerm = null; | ||||||
|  | 			searchbtn.classList.remove("show"); | ||||||
|  | 			searchbtn.firstChild.nodeValue = "Search" | ||||||
|  | 			matchedtxt.classList.add("hide"); | ||||||
|  | 			matchedtxt.firstChild.nodeValue = "" | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	function search(term) { | ||||||
|  | 		if (term) currentSearchTerm = term; | ||||||
|  | 		if (currentSearchTerm === null) return; | ||||||
|  |  | ||||||
|  | 		var re = new RegExp(currentSearchTerm, ignorecase ? 'i' : ''); | ||||||
|  | 		var el = document.getElementById("frames").children; | ||||||
|  | 		var matches = new Object(); | ||||||
|  | 		var maxwidth = 0; | ||||||
|  | 		for (var i = 0; i < el.length; i++) { | ||||||
|  | 			var e = el[i]; | ||||||
|  | 			var func = g_to_func(e); | ||||||
|  | 			var rect = find_child(e, "rect"); | ||||||
|  | 			if (func == null || rect == null) | ||||||
|  | 				continue; | ||||||
|  |  | ||||||
|  | 			// Save max width. Only works as we have a root frame | ||||||
|  | 			var w = parseFloat(rect.attributes.width.value); | ||||||
|  | 			if (w > maxwidth) | ||||||
|  | 				maxwidth = w; | ||||||
|  |  | ||||||
|  | 			if (func.match(re)) { | ||||||
|  | 				// highlight | ||||||
|  | 				var x = parseFloat(rect.attributes.x.value); | ||||||
|  | 				orig_save(rect, "fill"); | ||||||
|  | 				rect.attributes.fill.value = "rgb(230,0,230)"; | ||||||
|  |  | ||||||
|  | 				// remember matches | ||||||
|  | 				if (matches[x] == undefined) { | ||||||
|  | 					matches[x] = w; | ||||||
|  | 				} else { | ||||||
|  | 					if (w > matches[x]) { | ||||||
|  | 						// overwrite with parent | ||||||
|  | 						matches[x] = w; | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				searching = 1; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (!searching) | ||||||
|  | 			return; | ||||||
|  | 		var params = get_params(); | ||||||
|  | 		params.s = currentSearchTerm; | ||||||
|  | 		history.replaceState(null, null, parse_params(params)); | ||||||
|  |  | ||||||
|  | 		searchbtn.classList.add("show"); | ||||||
|  | 		searchbtn.firstChild.nodeValue = "Reset Search"; | ||||||
|  |  | ||||||
|  | 		// calculate percent matched, excluding vertical overlap | ||||||
|  | 		var count = 0; | ||||||
|  | 		var lastx = -1; | ||||||
|  | 		var lastw = 0; | ||||||
|  | 		var keys = Array(); | ||||||
|  | 		for (k in matches) { | ||||||
|  | 			if (matches.hasOwnProperty(k)) | ||||||
|  | 				keys.push(k); | ||||||
|  | 		} | ||||||
|  | 		// sort the matched frames by their x location | ||||||
|  | 		// ascending, then width descending | ||||||
|  | 		keys.sort(function(a, b){ | ||||||
|  | 			return a - b; | ||||||
|  | 		}); | ||||||
|  | 		// Step through frames saving only the biggest bottom-up frames | ||||||
|  | 		// thanks to the sort order. This relies on the tree property | ||||||
|  | 		// where children are always smaller than their parents. | ||||||
|  | 		var fudge = 0.0001;	// JavaScript floating point | ||||||
|  | 		for (var k in keys) { | ||||||
|  | 			var x = parseFloat(keys[k]); | ||||||
|  | 			var w = matches[keys[k]]; | ||||||
|  | 			if (x >= lastx + lastw - fudge) { | ||||||
|  | 				count += w; | ||||||
|  | 				lastx = x; | ||||||
|  | 				lastw = w; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		// display matched percent | ||||||
|  | 		matchedtxt.classList.remove("hide"); | ||||||
|  | 		var pct = 100 * count / maxwidth; | ||||||
|  | 		if (pct != 100) pct = pct.toFixed(1) | ||||||
|  | 		matchedtxt.firstChild.nodeValue = "Matched: " + pct + "%"; | ||||||
|  | 	} | ||||||
|  | ]]> | ||||||
|  | </script> | ||||||
|  | <rect x="0.0" y="0" width="1200.0" height="390.0" fill="url(#background)"  /> | ||||||
|  | <text id="title" x="600.00" y="24" >Flame Graph</text> | ||||||
|  | <text id="details" x="10.00" y="373" > </text> | ||||||
|  | <text id="unzoom" x="10.00" y="24" class="hide">Reset Zoom</text> | ||||||
|  | <text id="search" x="1090.00" y="24" >Search</text> | ||||||
|  | <text id="ignorecase" x="1174.00" y="24" >ic</text> | ||||||
|  | <text id="matched" x="1090.00" y="373" > </text> | ||||||
|  | <g id="frames"> | ||||||
|  | <g > | ||||||
|  | <title>_IO_new_file_underflow (2 samples, 0.52%)</title><rect x="1122.4" y="101" width="6.1" height="15.0" fill="rgb(214,42,10)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="111.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::ssl_client::read_loop() (4 samples, 1.04%)</title><rect x="919.6" y="213" width="12.3" height="15.0" fill="rgb(243,175,41)" rx="2" ry="2" /> | ||||||
|  | <text  x="922.58" y="223.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>lll_mutex_lock_optimized (1 samples, 0.26%)</title><rect x="117.6" y="213" width="3.0" height="15.0" fill="rgb(209,22,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="120.55" y="223.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::https_client::connect() (4 samples, 1.04%)</title><rect x="919.6" y="229" width="12.3" height="15.0" fill="rgb(251,211,50)" rx="2" ry="2" /> | ||||||
|  | <text  x="922.58" y="239.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_wait (20 samples, 5.21%)</title><rect x="1128.5" y="277" width="61.5" height="15.0" fill="rgb(252,219,52)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="287.5" >___pth..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_mutex_lock (1 samples, 0.26%)</title><rect x="117.6" y="229" width="3.0" height="15.0" fill="rgb(206,7,1)" rx="2" ry="2" /> | ||||||
|  | <text  x="120.55" y="239.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>all (384 samples, 100%)</title><rect x="10.0" y="341" width="1180.0" height="15.0" fill="rgb(213,39,9)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="351.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___underflow (2 samples, 0.52%)</title><rect x="1122.4" y="117" width="6.1" height="15.0" fill="rgb(221,74,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="127.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::parser<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="149" width="3.0" height="15.0" fill="rgb(217,59,14)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="159.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common (256 samples, 66.67%)</title><rect x="132.9" y="197" width="786.7" height="15.0" fill="rgb(205,0,0)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="207.5" >__futex_abstimed_wait_common</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_clockwait64 (40 samples, 10.42%)</title><rect x="931.9" y="261" width="122.9" height="15.0" fill="rgb(233,129,30)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="271.5" >___pthread_cond..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>?? (2 samples, 0.52%)</title><rect x="1122.4" y="165" width="6.1" height="15.0" fill="rgb(249,205,49)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="175.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_free (1 samples, 0.26%)</title><rect x="1054.8" y="133" width="3.1" height="15.0" fill="rgb(252,218,52)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="143.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>_IO_new_file_underflow (3 samples, 0.78%)</title><rect x="1057.9" y="117" width="9.2" height="15.0" fill="rgb(214,42,10)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="127.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::play(dpp::discord_voice_client*)::<lambda(dpp::discord_voice_client*)>, (13 samples, 3.39%)</title><rect x="1082.4" y="261" width="40.0" height="15.0" fill="rgb(208,13,3)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="271.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_fread (3 samples, 0.78%)</title><rect x="1057.9" y="165" width="9.2" height="15.0" fill="rgb(212,34,8)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="175.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>bumbleBee::MusicPlayManager::remove (1 samples, 0.26%)</title><rect x="1054.8" y="165" width="3.1" height="15.0" fill="rgb(230,117,28)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="175.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_default_xsgetn (3 samples, 0.78%)</title><rect x="1057.9" y="149" width="9.2" height="15.0" fill="rgb(236,145,34)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="159.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_clockwait64 (40 samples, 10.42%)</title><rect x="931.9" y="245" width="122.9" height="15.0" fill="rgb(233,129,30)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="255.5" >___pthread_cond..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_getline (4 samples, 1.04%)</title><rect x="1067.1" y="133" width="12.3" height="15.0" fill="rgb(226,100,23)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="143.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>_int_free (1 samples, 0.26%)</title><rect x="1054.8" y="117" width="3.1" height="15.0" fill="rgb(247,196,46)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="127.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (2 samples, 0.52%)</title><rect x="1122.4" y="69" width="6.1" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="79.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::in_thread::in_loop(unsigned (260 samples, 67.71%)</title><rect x="132.9" y="277" width="799.0" height="15.0" fill="rgb(208,15,3)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="287.5" >dpp::in_thread::in_loop(unsigned</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>bumbleBee::BumbleBee::start (20 samples, 5.21%)</title><rect x="1128.5" y="309" width="61.5" height="15.0" fill="rgb(252,216,51)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="319.5" >bumble..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common64 (13 samples, 3.39%)</title><rect x="1082.4" y="101" width="40.0" height="15.0" fill="rgb(220,71,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="111.5" >__f..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::discord_voice_client::write_ready() (4 samples, 1.04%)</title><rect x="120.6" y="245" width="12.3" height="15.0" fill="rgb(245,186,44)" rx="2" ry="2" /> | ||||||
|  | <text  x="123.62" y="255.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>operator() (2 samples, 0.52%)</title><rect x="1122.4" y="197" width="6.1" height="15.0" fill="rgb(231,121,29)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="207.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__pthread_cond_wait_common (40 samples, 10.42%)</title><rect x="931.9" y="229" width="122.9" height="15.0" fill="rgb(210,24,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="239.5" >__pthread_cond_..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_default_xsgetn (2 samples, 0.52%)</title><rect x="1122.4" y="133" width="6.1" height="15.0" fill="rgb(236,145,34)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="143.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::operator>> (1 samples, 0.26%)</title><rect x="1079.4" y="165" width="3.0" height="15.0" fill="rgb(210,23,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="175.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common (40 samples, 10.42%)</title><rect x="931.9" y="197" width="122.9" height="15.0" fill="rgb(205,0,0)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="207.5" >__futex_abstime..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_wait (13 samples, 3.39%)</title><rect x="1082.4" y="165" width="40.0" height="15.0" fill="rgb(252,219,52)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="175.5" >___..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke_impl<void, (13 samples, 3.39%)</title><rect x="1082.4" y="213" width="40.0" height="15.0" fill="rgb(244,179,42)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="223.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>main (20 samples, 5.21%)</title><rect x="1128.5" y="325" width="61.5" height="15.0" fill="rgb(243,179,42)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="335.5" >main</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::BumbleBee::on_slashcommand(const (9 samples, 2.34%)</title><rect x="1054.8" y="245" width="27.6" height="15.0" fill="rgb(251,213,50)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="255.5" >s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke<bumbleBee::MusicPlayManager::send_audio_to_voice(std::shared_ptr<bumbleBee::MusicQueueElement>, (2 samples, 0.52%)</title><rect x="1122.4" y="229" width="6.1" height="15.0" fill="rgb(217,59,14)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="239.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::request_queue::out_loop() (40 samples, 10.42%)</title><rect x="931.9" y="277" width="122.9" height="15.0" fill="rgb(227,105,25)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="287.5" >dpp::request_qu..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::send_audio_to_voice(std::shared_ptr<bumbleBee::MusicQueueElement>, (2 samples, 0.52%)</title><rect x="1122.4" y="245" width="6.1" height="15.0" fill="rgb(216,50,12)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="255.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke_impl<void, (9 samples, 2.34%)</title><rect x="1054.8" y="213" width="27.6" height="15.0" fill="rgb(244,179,42)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="223.5" >s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::BumbleBee::on_slashcommand(const (9 samples, 2.34%)</title><rect x="1054.8" y="261" width="27.6" height="15.0" fill="rgb(251,213,50)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="271.5" >s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_State_impl<std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::play(dpp::discord_voice_client*)::<lambda(dpp::discord_voice_client*)>, (13 samples, 3.39%)</title><rect x="1082.4" y="277" width="40.0" height="15.0" fill="rgb(205,4,1)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="287.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>bumbleBee::commands::Play::execute (8 samples, 2.08%)</title><rect x="1057.9" y="181" width="24.5" height="15.0" fill="rgb(217,56,13)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="191.5" >b..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___poll (20 samples, 5.21%)</title><rect x="10.0" y="245" width="61.5" height="15.0" fill="rgb(228,110,26)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="255.5" >__GI__..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>_IO_fgets (4 samples, 1.04%)</title><rect x="1067.1" y="149" width="12.3" height="15.0" fill="rgb(235,139,33)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="159.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke<bumbleBee::MusicPlayManager::play(dpp::discord_voice_client*)::<lambda(dpp::discord_voice_client*)>, (13 samples, 3.39%)</title><rect x="1082.4" y="229" width="40.0" height="15.0" fill="rgb(209,22,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="239.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_State_impl<std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::send_audio_to_voice(std::shared_ptr<bumbleBee::MusicQueueElement>, (2 samples, 0.52%)</title><rect x="1122.4" y="277" width="6.1" height="15.0" fill="rgb(213,41,9)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="287.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (2 samples, 0.52%)</title><rect x="1122.4" y="85" width="6.1" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="95.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_clockwait64 (256 samples, 66.67%)</title><rect x="132.9" y="245" width="786.7" height="15.0" fill="rgb(233,129,30)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="255.5" >___pthread_cond_clockwait64</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>operator() (9 samples, 2.34%)</title><rect x="1054.8" y="197" width="27.6" height="15.0" fill="rgb(231,121,29)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="207.5" >o..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___clock_nanosleep (1 samples, 0.26%)</title><rect x="120.6" y="213" width="3.1" height="15.0" fill="rgb(232,126,30)" rx="2" ry="2" /> | ||||||
|  | <text  x="123.62" y="223.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common (13 samples, 3.39%)</title><rect x="1082.4" y="117" width="40.0" height="15.0" fill="rgb(205,0,0)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="127.5" >__f..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___poll (4 samples, 1.04%)</title><rect x="919.6" y="197" width="12.3" height="15.0" fill="rgb(228,110,26)" rx="2" ry="2" /> | ||||||
|  | <text  x="922.58" y="207.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__pthread_cond_wait_common (20 samples, 5.21%)</title><rect x="1128.5" y="261" width="61.5" height="15.0" fill="rgb(210,24,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="271.5" >__pthr..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::lexer<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="85" width="3.0" height="15.0" fill="rgb(222,78,18)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="95.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::parser<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="133" width="3.0" height="15.0" fill="rgb(217,59,14)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="143.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::discord_voice_client::thread_run() (20 samples, 5.21%)</title><rect x="71.5" y="277" width="61.4" height="15.0" fill="rgb(238,153,36)" rx="2" ry="2" /> | ||||||
|  | <text  x="74.46" y="287.5" >dpp::d..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::discord_client::thread_run() (20 samples, 5.21%)</title><rect x="10.0" y="277" width="61.5" height="15.0" fill="rgb(251,214,51)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="287.5" >dpp::d..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::lexer<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="101" width="3.0" height="15.0" fill="rgb(222,78,18)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="111.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___poll (15 samples, 3.91%)</title><rect x="71.5" y="245" width="46.1" height="15.0" fill="rgb(228,110,26)" rx="2" ry="2" /> | ||||||
|  | <text  x="74.46" y="255.5" >__GI..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___underflow (3 samples, 0.78%)</title><rect x="1057.9" y="133" width="9.2" height="15.0" fill="rgb(221,74,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="143.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_State_impl<std::thread::_Invoker<std::tuple<bumbleBee::BumbleBee::on_slashcommand(const (9 samples, 2.34%)</title><rect x="1054.8" y="277" width="27.6" height="15.0" fill="rgb(249,203,48)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="287.5" >s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___futex_abstimed_wait_cancelable64 (13 samples, 3.39%)</title><rect x="1082.4" y="133" width="40.0" height="15.0" fill="rgb(215,48,11)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="143.5" >__G..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>___pthread_cond_clockwait64 (256 samples, 66.67%)</title><rect x="132.9" y="261" width="786.7" height="15.0" fill="rgb(233,129,30)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="271.5" >___pthread_cond_clockwait64</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::https_client::https_client(std::__cxx11::basic_string<char, (4 samples, 1.04%)</title><rect x="919.6" y="245" width="12.3" height="15.0" fill="rgb(206,4,1)" rx="2" ry="2" /> | ||||||
|  | <text  x="922.58" y="255.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::lexer<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="69" width="3.0" height="15.0" fill="rgb(222,78,18)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="79.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::ssl_client::read_loop() (20 samples, 5.21%)</title><rect x="71.5" y="261" width="61.4" height="15.0" fill="rgb(243,175,41)" rx="2" ry="2" /> | ||||||
|  | <text  x="74.46" y="271.5" >dpp::s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___futex_abstimed_wait_cancelable64 (40 samples, 10.42%)</title><rect x="931.9" y="213" width="122.9" height="15.0" fill="rgb(215,48,11)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="223.5" >__GI___futex_ab..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common64 (40 samples, 10.42%)</title><rect x="931.9" y="181" width="122.9" height="15.0" fill="rgb(220,71,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="934.88" y="191.5" >__futex_abstime..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>futex_wait (1 samples, 0.26%)</title><rect x="117.6" y="181" width="3.0" height="15.0" fill="rgb(235,138,33)" rx="2" ry="2" /> | ||||||
|  | <text  x="120.55" y="191.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__pthread_cond_wait_common (256 samples, 66.67%)</title><rect x="132.9" y="229" width="786.7" height="15.0" fill="rgb(210,24,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="239.5" >__pthread_cond_wait_common</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::vector<char, (1 samples, 0.26%)</title><rect x="1079.4" y="53" width="3.0" height="15.0" fill="rgb(222,81,19)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="63.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::condition_variable::wait<bumbleBee::MusicPlayManager::play(dpp::discord_voice_client*)::<lambda(dpp::discord_voice_client*)>::<lambda()> (13 samples, 3.39%)</title><rect x="1082.4" y="181" width="40.0" height="15.0" fill="rgb(242,172,41)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="191.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (3 samples, 0.78%)</title><rect x="1057.9" y="85" width="9.2" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="95.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_fread (2 samples, 0.52%)</title><rect x="1122.4" y="149" width="6.1" height="15.0" fill="rgb(212,34,8)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="159.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>?? (364 samples, 94.79%)</title><rect x="10.0" y="293" width="1118.5" height="15.0" fill="rgb(249,205,49)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="303.5" >??</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common (20 samples, 5.21%)</title><rect x="1128.5" y="229" width="61.5" height="15.0" fill="rgb(205,0,0)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="239.5" >__fute..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>start_thread (364 samples, 94.79%)</title><rect x="10.0" y="309" width="1118.5" height="15.0" fill="rgb(212,34,8)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="319.5" >start_thread</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___lll_lock_wait (1 samples, 0.26%)</title><rect x="117.6" y="197" width="3.0" height="15.0" fill="rgb(226,96,23)" rx="2" ry="2" /> | ||||||
|  | <text  x="120.55" y="207.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>clone3 (364 samples, 94.79%)</title><rect x="10.0" y="325" width="1118.5" height="15.0" fill="rgb(216,54,12)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="335.5" >clone3</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::http_request::run(dpp::cluster*) (4 samples, 1.04%)</title><rect x="919.6" y="261" width="12.3" height="15.0" fill="rgb(230,119,28)" rx="2" ry="2" /> | ||||||
|  | <text  x="922.58" y="271.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_default_uflow (4 samples, 1.04%)</title><rect x="1067.1" y="101" width="12.3" height="15.0" fill="rgb(234,135,32)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="111.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::play(dpp::discord_voice_client*)::<lambda(dpp::discord_voice_client*)>, (13 samples, 3.39%)</title><rect x="1082.4" y="245" width="40.0" height="15.0" fill="rgb(208,13,3)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="255.5" >std..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (4 samples, 1.04%)</title><rect x="1067.1" y="69" width="12.3" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="79.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (3 samples, 0.78%)</title><rect x="1057.9" y="101" width="9.2" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1060.86" y="111.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>_IO_new_file_underflow (4 samples, 1.04%)</title><rect x="1067.1" y="85" width="12.3" height="15.0" fill="rgb(214,42,10)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="95.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___libc_read (4 samples, 1.04%)</title><rect x="1067.1" y="53" width="12.3" height="15.0" fill="rgb(241,166,39)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="63.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::discord_voice_client::stop_audio() (1 samples, 0.26%)</title><rect x="1054.8" y="149" width="3.1" height="15.0" fill="rgb(217,57,13)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="159.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke<bumbleBee::BumbleBee::on_slashcommand(const (9 samples, 2.34%)</title><rect x="1054.8" y="229" width="27.6" height="15.0" fill="rgb(253,222,53)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="239.5" >s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::ssl_client::read_loop() (20 samples, 5.21%)</title><rect x="10.0" y="261" width="61.5" height="15.0" fill="rgb(243,175,41)" rx="2" ry="2" /> | ||||||
|  | <text  x="13.00" y="271.5" >dpp::s..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI__IO_getline_info (4 samples, 1.04%)</title><rect x="1067.1" y="117" width="12.3" height="15.0" fill="rgb(248,199,47)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="127.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>bumbleBee::commands::Delete::execute (1 samples, 0.26%)</title><rect x="1054.8" y="181" width="3.1" height="15.0" fill="rgb(246,190,45)" rx="2" ry="2" /> | ||||||
|  | <text  x="1057.79" y="191.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::__invoke_impl<void, (2 samples, 0.52%)</title><rect x="1122.4" y="213" width="6.1" height="15.0" fill="rgb(244,179,42)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="223.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::cluster::start(bool) (20 samples, 5.21%)</title><rect x="1128.5" y="293" width="61.5" height="15.0" fill="rgb(205,0,0)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="303.5" >dpp::c..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::vector<dpp::voice_out_packet, (3 samples, 0.78%)</title><rect x="123.7" y="229" width="9.2" height="15.0" fill="rgb(222,82,19)" rx="2" ry="2" /> | ||||||
|  | <text  x="126.70" y="239.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>dpp::discord_voice_client::want_write() (1 samples, 0.26%)</title><rect x="117.6" y="245" width="3.0" height="15.0" fill="rgb(233,130,31)" rx="2" ry="2" /> | ||||||
|  | <text  x="120.55" y="255.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__pthread_cond_wait_common (13 samples, 3.39%)</title><rect x="1082.4" y="149" width="40.0" height="15.0" fill="rgb(210,24,5)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="159.5" >__p..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common64 (256 samples, 66.67%)</title><rect x="132.9" y="181" width="786.7" height="15.0" fill="rgb(220,71,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="191.5" >__futex_abstimed_wait_common64</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___nanosleep (1 samples, 0.26%)</title><rect x="120.6" y="229" width="3.1" height="15.0" fill="rgb(231,119,28)" rx="2" ry="2" /> | ||||||
|  | <text  x="123.62" y="239.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>oggz_read (2 samples, 0.52%)</title><rect x="1122.4" y="181" width="6.1" height="15.0" fill="rgb(252,218,52)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="191.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>operator() (13 samples, 3.39%)</title><rect x="1082.4" y="197" width="40.0" height="15.0" fill="rgb(231,121,29)" rx="2" ry="2" /> | ||||||
|  | <text  x="1085.45" y="207.5" >ope..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::thread::_Invoker<std::tuple<bumbleBee::MusicPlayManager::send_audio_to_voice(std::shared_ptr<bumbleBee::MusicQueueElement>, (2 samples, 0.52%)</title><rect x="1122.4" y="261" width="6.1" height="15.0" fill="rgb(216,50,12)" rx="2" ry="2" /> | ||||||
|  | <text  x="1125.40" y="271.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___futex_abstimed_wait_cancelable64 (256 samples, 66.67%)</title><rect x="132.9" y="213" width="786.7" height="15.0" fill="rgb(215,48,11)" rx="2" ry="2" /> | ||||||
|  | <text  x="135.92" y="223.5" >__GI___futex_abstimed_wait_cancelable64</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>nlohmann::json_abi_v3_11_2::detail::parser<nlohmann::json_abi_v3_11_2::basic_json<std::map, (1 samples, 0.26%)</title><rect x="1079.4" y="117" width="3.0" height="15.0" fill="rgb(217,59,14)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="127.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>std::move<char&> (1 samples, 0.26%)</title><rect x="1079.4" y="37" width="3.0" height="15.0" fill="rgb(232,124,29)" rx="2" ry="2" /> | ||||||
|  | <text  x="1082.38" y="47.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>bumbleBee::ConsoleUtils::getResultFromCommand (4 samples, 1.04%)</title><rect x="1067.1" y="165" width="12.3" height="15.0" fill="rgb(223,85,20)" rx="2" ry="2" /> | ||||||
|  | <text  x="1070.08" y="175.5" ></text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__futex_abstimed_wait_common64 (20 samples, 5.21%)</title><rect x="1128.5" y="213" width="61.5" height="15.0" fill="rgb(220,71,17)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="223.5" >__fute..</text> | ||||||
|  | </g> | ||||||
|  | <g > | ||||||
|  | <title>__GI___futex_abstimed_wait_cancelable64 (20 samples, 5.21%)</title><rect x="1128.5" y="245" width="61.5" height="15.0" fill="rgb(215,48,11)" rx="2" ry="2" /> | ||||||
|  | <text  x="1131.54" y="255.5" >__GI__..</text> | ||||||
|  | </g> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 36 KiB | 
							
								
								
									
										4
									
								
								systemStackCapture.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										4
									
								
								systemStackCapture.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | sudo echo recording... | ||||||
|  | sudo perf record -F 99 -a -g -- sleep 60 | ||||||
|  | sudo perf script | ./stackcollapse-perf.pl | ./flamegraph.pl > stackflame.svg | ||||||
|  | sudo rm perf.data | ||||||
		Reference in New Issue
	
	Block a user