libreoffice-online/loolwsd/Util.cpp

622 lines
17 KiB
C++
Raw Normal View History

/* -*- 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/.
*/
2016-04-08 07:22:22 -05:00
#include <execinfo.h>
#include <sys/poll.h>
#include <sys/prctl.h>
2016-04-08 07:22:22 -05:00
#include <sys/uio.h>
#include <cassert>
#include <cstdlib>
2015-04-09 17:25:48 -05:00
#include <cstring>
#include <iomanip>
#include <mutex>
#include <random>
#include <sstream>
#include <string>
2016-04-07 11:43:26 -05:00
#include <unistd.h>
#include <png.h>
2015-04-24 04:49:19 -05:00
#include <signal.h>
#include <Poco/ConsoleChannel.h>
#include <Poco/Exception.h>
#include <Poco/Format.h>
#include <Poco/Net/WebSocket.h>
#include <Poco/Process.h>
#include <Poco/Thread.h>
#include <Poco/Timestamp.h>
#include <Poco/Util/Application.h>
#include "Common.hpp"
#include "Util.hpp"
#include "Png.hpp"
// Callback functions for libpng
extern "C"
{
static void user_write_status_fn(png_structp, png_uint_32, int)
{
}
static void user_write_fn(png_structp png_ptr, png_bytep data, png_size_t length)
{
std::vector<char> *outputp = (std::vector<char> *) png_get_io_ptr(png_ptr);
const size_t oldsize = outputp->size();
outputp->resize(oldsize + length);
2016-03-01 07:27:36 -06:00
std::memcpy(outputp->data() + oldsize, data, length);
}
static void user_flush_fn(png_structp)
{
}
}
volatile bool TerminationFlag = false;
namespace Util
{
namespace rng
{
static std::random_device _rd;
static std::mutex _rngMutex;
// 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<std::mutex> lock(_rngMutex);
return _rng();
}
}
}
2016-04-07 11:43:26 -05:00
static char LogPrefix[256] = { '\0' };
namespace Log
{
static const Poco::Int64 epochStart = Poco::Timestamp().epochMicroseconds();
// help avoid destruction ordering issues.
struct StaticNames {
bool inited;
std::string name;
std::string id;
StaticNames() :
inited(true)
{
}
~StaticNames()
{
inited = false;
}
};
static StaticNames Source;
std::string logPrefix()
{
Poco::Int64 usec = Poco::Timestamp().epochMicroseconds() - epochStart;
const Poco::Int64 one_s = 1000000;
const Poco::Int64 hours = usec / (one_s*60*60);
usec %= (one_s*60*60);
const Poco::Int64 minutes = usec / (one_s*60);
usec %= (one_s*60);
const Poco::Int64 seconds = usec / (one_s);
usec %= (one_s);
std::ostringstream stream;
stream << (Source.inited ? Source.id : std::string())
<< '-' << std::setw(2) << std::setfill('0')
<< (Poco::Thread::current() ? Poco::Thread::current()->id() : 0) << ' '
<< std::setw(2) << hours << ':' << std::setw(2) << minutes << ':'
<< std::setw(2) << seconds << "." << std::setw(6) << usec
<< ' ';
char buf[32]; // we really need only 16
if (prctl(PR_GET_NAME, reinterpret_cast<unsigned long>(buf), 0, 0, 0) == 0)
stream << '[' << std::setw(15) << std::setfill(' ') << std::left << buf << "] ";
return stream.str();
}
void initialize(const std::string& name)
{
Source.name = name;
std::ostringstream oss;
oss << Source.name << '-'
<< std::setw(5) << std::setfill('0') << Poco::Process::id();
Source.id = oss.str();
2016-04-07 11:43:26 -05:00
assert (sizeof (LogPrefix) > strlen(oss.str().c_str()) + 1);
strcpy(LogPrefix, oss.str().c_str());
auto channel = (isatty(fileno(stdout)) || std::getenv("LOOL_LOGCOLOR")
? static_cast<Poco::Channel*>(new Poco::ColorConsoleChannel())
: static_cast<Poco::Channel*>(new Poco::ConsoleChannel()));
auto& logger = Poco::Logger::create(Source.name, channel, Poco::Message::PRIO_TRACE);
channel->release();
// Configure the logger.
// TODO: This should come from a file.
// See Poco::Logger::setLevel docs for values.
// Try: error, information, debug
char *loglevel = std::getenv("LOOL_LOGLEVEL");
if (loglevel)
logger.setLevel(std::string(loglevel));
info("Initializing " + name);
info("Log level is [" + std::to_string(logger.getLevel()) + "].");
}
Poco::Logger& logger()
{
return Poco::Logger::get(Source.inited ? Source.name : std::string());
}
void trace(const std::string& msg)
{
logger().trace(logPrefix() + msg);
}
void debug(const std::string& msg)
{
logger().debug(logPrefix() + msg);
}
void info(const std::string& msg)
{
logger().information(logPrefix() + msg);
}
void warn(const std::string& msg)
{
logger().warning(logPrefix() + msg);
}
void error(const std::string& msg)
{
logger().error(logPrefix() + msg);
}
void syserror(const std::string& msg)
{
logger().error(logPrefix() + msg + " (errno: " + std::string(std::strerror(errno)) + ")");
}
}
namespace Util
{
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;
}
std::string createRandomDir(const std::string& path)
{
Poco::File(path).createDirectories();
for (;;)
{
const auto name = Util::encodeId(rng::getNext());
Poco::File dir(Poco::Path(path, name));
if (dir.createDirectory())
{
return name;
}
}
}
std::string createRandomFile(const std::string& path)
{
Poco::File(path).createDirectories();
for (;;)
{
const auto name = Util::encodeId(rng::getNext());
Poco::File file(Poco::Path(path, name));
if (file.createFile())
{
return name;
}
}
}
bool windowingAvailable()
{
return std::getenv("DISPLAY") != nullptr;
}
bool encodeBufferToPNG(unsigned char *pixmap, int width, int height, std::vector<char>& output, LibreOfficeKitTileMode mode)
{
return encodeSubBufferToPNG(pixmap, 0, 0, width, height, width, height, output, mode);
}
bool encodeSubBufferToPNG(unsigned char *pixmap, int startX, int startY, int width, int height,
int bufferWidth, int bufferHeight, std::vector<char>& output, LibreOfficeKitTileMode mode)
{
if (bufferWidth < width || bufferHeight < height)
return false;
png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
png_infop info_ptr = png_create_info_struct(png_ptr);
if (setjmp(png_jmpbuf(png_ptr)))
{
png_destroy_write_struct(&png_ptr, nullptr);
return false;
}
png_set_IHDR(png_ptr, info_ptr, width, height, 8, PNG_COLOR_TYPE_RGB_ALPHA, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT);
png_set_write_fn(png_ptr, &output, user_write_fn, user_flush_fn);
png_set_write_status_fn(png_ptr, user_write_status_fn);
png_write_info(png_ptr, info_ptr);
if (mode == LOK_TILEMODE_BGRA)
{
png_set_write_user_transform_fn (png_ptr, unpremultiply_data);
}
for (int y = 0; y < height; ++y)
{
size_t position = ((startY + y) * bufferWidth * 4) + (startX * 4);
png_write_row(png_ptr, pixmap + position);
}
png_write_end(png_ptr, info_ptr);
png_destroy_write_struct(&png_ptr, &info_ptr);
return true;
}
const char *signalName(const int signo)
2015-04-24 04:49:19 -05:00
{
switch (signo)
{
#define CASE(x) case SIG##x: return "SIG" #x
2015-04-24 04:49:19 -05:00
CASE(HUP);
CASE(INT);
CASE(QUIT);
CASE(ILL);
CASE(ABRT);
CASE(FPE);
CASE(KILL);
CASE(SEGV);
CASE(PIPE);
2015-04-24 04:49:19 -05:00
CASE(ALRM);
CASE(TERM);
CASE(USR1);
CASE(USR2);
CASE(CHLD);
CASE(CONT);
CASE(STOP);
CASE(TSTP);
CASE(TTIN);
CASE(TTOU);
CASE(BUS);
#ifdef SIGPOLL
2015-04-24 04:49:19 -05:00
CASE(POLL);
#endif
2015-04-24 04:49:19 -05:00
CASE(PROF);
CASE(SYS);
CASE(TRAP);
CASE(URG);
CASE(VTALRM);
CASE(XCPU);
CASE(XFSZ);
#ifdef SIGEMT
CASE(EMT);
#endif
#ifdef SIGSTKFLT
2015-04-24 04:49:19 -05:00
CASE(STKFLT);
#endif
2015-04-24 04:49:19 -05:00
#if defined(SIGIO) && SIGIO != SIGPOLL
CASE(IO);
#endif
#ifdef SIGPWR
CASE(PWR);
#endif
#ifdef SIGLOST
CASE(LOST);
#endif
CASE(WINCH);
#if defined(SIGINFO) && SIGINFO != SIGPWR
CASE(INFO);
#endif
2015-04-24 04:49:19 -05:00
#undef CASE
default:
return "unknown";
2015-04-24 04:49:19 -05:00
}
}
2016-04-07 11:43:26 -05:00
// We need a signal safe means of writing messages
// $ man 7 signal
static
void log_signal(const char *message)
{
while (true) {
int length = strlen(message);
int written = write (STDERR_FILENO, message, length);
if (written < 0)
{
if (errno == EINTR)
continue; // ignore.
else
break;
}
message += written;
if (message[0] == '\0')
break;
}
}
static
void handleTerminationSignal(const int signal)
{
if (!TerminationFlag)
{
TerminationFlag = true;
2016-04-07 11:43:26 -05:00
log_signal(LogPrefix);
log_signal(" Termination signal received: ");
log_signal(signalName(signal));
2016-04-07 11:43:26 -05:00
log_signal("\n");
}
}
2016-02-01 19:26:19 -06:00
void setTerminationSignals()
{
struct sigaction action;
sigemptyset(&action.sa_mask);
action.sa_flags = 0;
action.sa_handler = handleTerminationSignal;
sigaction(SIGTERM, &action, nullptr);
sigaction(SIGINT, &action, nullptr);
sigaction(SIGQUIT, &action, nullptr);
sigaction(SIGHUP, &action, nullptr);
2016-02-01 19:26:19 -06:00
}
2016-04-07 11:43:26 -05:00
static char FatalGdbString[256] = { '\0' };
2016-02-01 19:26:19 -06:00
static
void handleFatalSignal(const int signal)
2016-02-01 19:26:19 -06:00
{
2016-04-07 11:43:26 -05:00
log_signal(LogPrefix);
log_signal(" Fatal signal received: ");
log_signal(signalName(signal));
2016-04-07 11:43:26 -05:00
log_signal("\n");
2016-03-01 07:27:36 -06:00
if (std::getenv("LOOL_DEBUG"))
{
2016-04-07 11:43:26 -05:00
log_signal(FatalGdbString);
2016-03-07 09:44:51 -06:00
sleep(30);
}
2016-02-01 19:26:19 -06:00
struct sigaction action;
2016-02-01 19:26:19 -06:00
sigemptyset(&action.sa_mask);
action.sa_flags = 0;
action.sa_handler = SIG_DFL;
2016-02-01 19:26:19 -06:00
sigaction(signal, &action, NULL);
2016-04-08 07:22:22 -05:00
const int maxSlots = 50;
void *backtraceBuffer[maxSlots];
int numSlots = backtrace(backtraceBuffer, maxSlots);
if (numSlots > 0)
{
char **symbols = backtrace_symbols(backtraceBuffer, numSlots);
if (symbols != NULL)
{
struct iovec ioVector[maxSlots*2+1];
ioVector[0].iov_base = (void*)"Backtrace:\n";
ioVector[0].iov_len = std::strlen((const char*)ioVector[0].iov_base);
for (int i = 0; i < numSlots; i++)
{
ioVector[1+i*2+0].iov_base = symbols[i];
ioVector[1+i*2+0].iov_len = std::strlen((const char *)ioVector[1+i*2+0].iov_base);
ioVector[1+i*2+1].iov_base = (void*)"\n";
ioVector[1+i*2+1].iov_len = 1;
}
if (writev(STDERR_FILENO, ioVector, numSlots*2+1) == -1)
{
Log::syserror("Failed to dump backtrace to stderr.");
}
2016-04-08 07:22:22 -05:00
}
}
2016-02-01 19:26:19 -06:00
// let default handler process the signal
kill(Poco::Process::id(), signal);
2016-02-01 19:26:19 -06:00
}
void setFatalSignals()
{
struct sigaction action;
2016-02-01 19:26:19 -06:00
sigemptyset(&action.sa_mask);
action.sa_flags = 0;
action.sa_handler = handleFatalSignal;
2016-02-01 19:26:19 -06:00
sigaction(SIGSEGV, &action, NULL);
sigaction(SIGBUS, &action, NULL);
sigaction(SIGABRT, &action, NULL);
sigaction(SIGILL, &action, NULL);
sigaction(SIGFPE, &action, NULL);
2016-04-07 11:43:26 -05:00
// prepare this in advance just in case.
std::ostringstream stream;
stream << "\nFatal signal! Attach debugger with:\n"
<< "sudo gdb --pid=" << Poco::Process::id() << "\n or \n"
<< "sudo gdb --q --n --ex 'thread apply all backtrace full' --batch --pid="
<< Poco::Process::id() << "\n";
std::string streamStr = stream.str();
assert (sizeof (FatalGdbString) > strlen(streamStr.c_str()) + 1);
strcpy(FatalGdbString, streamStr.c_str());
}
2016-02-01 20:23:58 -06:00
int getChildStatus(const int code)
2016-02-01 20:23:58 -06:00
{
int retVal;
2016-02-01 20:23:58 -06:00
switch (static_cast<const LOOLExitCode>(code))
2016-02-01 20:23:58 -06:00
{
case LOOLExitCode::LOOL_SECOND_OFFICE:
case LOOLExitCode::LOOL_FATAL_ERROR:
case LOOLExitCode::LOOL_CRASH_WITH_RESTART:
case LOOLExitCode::LOOL_NORMAL_RESTART:
case LOOLExitCode::LOOL_EXIT_SOFTWARE:
retVal = EXIT_FAILURE;
2016-02-01 20:23:58 -06:00
break;
case LOOLExitCode::LOOL_NO_ERROR:
retVal = EXIT_SUCCESS;
2016-02-01 20:23:58 -06:00
break;
// Why are other non-zero exit codes interpreted as success?
2016-02-01 20:23:58 -06:00
default:
retVal = EXIT_SUCCESS;
2016-02-01 20:23:58 -06:00
break;
}
return retVal;
2016-02-01 20:23:58 -06:00
}
int getSignalStatus(const int code)
2016-02-01 20:23:58 -06:00
{
int retVal;
2016-02-01 20:23:58 -06:00
switch (code)
2016-02-01 20:23:58 -06:00
{
case SIGSEGV:
case SIGBUS:
case SIGABRT:
case SIGILL:
case SIGFPE:
case SIGTERM:
case SIGINT:
case SIGQUIT:
case SIGHUP:
retVal = EXIT_FAILURE;
2016-02-01 20:23:58 -06:00
break;
// Why are other signals treated as success? Will this function ever be called when a
// child was *not* terminated by a signal?
2016-02-01 20:23:58 -06:00
default:
retVal = EXIT_SUCCESS;
2016-02-01 20:23:58 -06:00
break;
}
return retVal;
2016-02-01 20:23:58 -06:00
}
void requestTermination(const Poco::Process::PID& pid)
{
try
{
Poco::Process::requestTermination(pid);
}
catch(const Poco::Exception& exc)
{
Log::warn("Util::requestTermination: Exception: " + exc.message());
}
}
int getMemoryUsage(const Poco::Process::PID nPid)
{
//TODO: Instead of RSS, return PSS
const auto cmd = "ps o rss= -p " + std::to_string(nPid);
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);
int nMem = -1;
try
{
nMem = std::stoi(sResponse);
}
catch(std::exception& e)
{
Log::warn() << "Trying to find memory of invalid/dead PID" << Log::end;
}
return nMem;
}
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<unsigned long>(s.c_str()), 0, 0, 0) != 0)
Log::syserror("Cannot set thread name to " + s + ".");
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */