/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "Util.hpp" #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Common.hpp" #include "Log.hpp" #include "Util.hpp" namespace Util { namespace rng { static std::random_device _rd; static std::mutex _rngMutex; static Poco::RandomBuf _randBuf; // Create the prng with a random-device for seed. // If we don't have a hardware random-device, we will get the same seed. // In that case we are better off with an arbitrary, but changing, seed. static std::mt19937_64 _rng = std::mt19937_64(_rd.entropy() ? _rd() : (clock() + getpid())); // A new seed is used to shuffle the sequence. // N.B. Always reseed after getting forked! void reseed() { _rng.seed(_rd.entropy() ? _rd() : (clock() + getpid())); } // Returns a new random number. unsigned getNext() { std::unique_lock lock(_rngMutex); return _rng(); } std::vector getBytes(const size_t length) { std::vector v(length); _randBuf.readFromDevice(v.data(), v.size()); return v; } /// Generates a random string in Base64. /// Note: May contain '/' characters. std::string getB64String(const size_t length) { std::stringstream ss; Poco::Base64Encoder b64(ss); b64.write(getBytes(length).data(), length); return ss.str().substr(0, length); } std::string getFilename(const size_t length) { std::string s = getB64String(length); std::replace(s.begin(), s.end(), '/', '_'); return s.substr(0, length); } } std::string encodeId(const unsigned number, const int padding) { std::ostringstream oss; oss << std::hex << std::setw(padding) << std::setfill('0') << number; return oss.str(); } unsigned decodeId(const std::string& str) { unsigned id = 0; std::stringstream ss; ss << std::hex << str; ss >> id; return id; } bool windowingAvailable() { return std::getenv("DISPLAY") != nullptr; } static const char *startsWith(const char *line, const char *tag) { int len = strlen(tag); if (!strncmp(line, tag, len)) { while (!isdigit(line[len]) && line[len] != '\0') ++len; const auto str = std::string(line + len, strlen(line + len) - 1); return line + len; } return nullptr; } std::pair getPssAndDirtyFromSMaps(FILE* file) { size_t numPSSKb = 0; size_t numDirtyKb = 0; if (file) { rewind(file); char line[4096] = { 0 }; while (fgets(line, sizeof (line), file)) { const char *value; if ((value = startsWith(line, "Private_Dirty:")) || (value = startsWith(line, "Shared_Dirty:"))) { numDirtyKb += atoi(value); } else if ((value = startsWith(line, "Pss:"))) { numPSSKb += atoi(value); } } } return std::make_pair(numPSSKb, numDirtyKb); } std::string getMemoryStats(FILE* file) { const auto pssAndDirtyKb = getPssAndDirtyFromSMaps(file); std::ostringstream oss; oss << "procmemstats: pid=" << getpid() << " pss=" << pssAndDirtyKb.first << " dirty=" << pssAndDirtyKb.second; LOG_TRC("Collected " << oss.str()); return oss.str(); } size_t getMemoryUsagePSS(const Poco::Process::PID pid) { if (pid > 0) { const auto cmd = "/proc/" + std::to_string(pid) + "/smaps"; FILE* fp = fopen(cmd.c_str(), "r"); if (fp != nullptr) { const auto pss = getPssAndDirtyFromSMaps(fp).first; fclose(fp); return pss; } } return 0; } size_t getMemoryUsageRSS(const Poco::Process::PID pid) { if (pid == -1) { return 0; } try { const auto cmd = "ps o rss= -p " + std::to_string(pid); FILE* fp = popen(cmd.c_str(), "r"); if (fp == nullptr) { return 0; } std::string sResponse; char cmdBuffer[1024]; while (fgets(cmdBuffer, sizeof(cmdBuffer) - 1, fp) != nullptr) { sResponse += cmdBuffer; } pclose(fp); return std::stoi(sResponse); } catch (const std::exception&) { LOG_WRN("Trying to find memory of invalid/dead PID " << pid); } return 0; } std::string replace(const std::string& s, const std::string& a, const std::string& b) { std::string result = s; std::string::size_type pos; while ((pos = result.find(a)) != std::string::npos) { result = result.replace(pos, a.size(), b); } return result; } std::string formatLinesForLog(const std::string& s) { std::string r; std::string::size_type n = s.size(); if (n > 0 && s.back() == '\n') r = s.substr(0, n-1); else r = s; return replace(r, "\n", " / "); } void setThreadName(const std::string& s) { if (prctl(PR_SET_NAME, reinterpret_cast(s.c_str()), 0, 0, 0) != 0) { LOG_SYS("Cannot set thread name to " << s << "."); } } void getVersionInfo(std::string& version, std::string& hash) { version = std::string(LOOLWSD_VERSION); hash = std::string(LOOLWSD_VERSION_HASH); hash.resize(std::min(8, (int)hash.length())); } std::string UniqueId() { static std::atomic_int counter(0); return std::to_string(Poco::Process::id()) + "/" + std::to_string(counter++); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */