/* -*- 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/. */ /* * A very simple, single threaded helper to efficiently pre-init and * spawn lots of kits as children. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Kit.hpp" #include #include #include #include #include #include #include using Poco::Process; using Poco::Thread; #ifndef KIT_IN_PROCESS static bool NoCapsForKit = false; static bool NoSeccomp = false; #endif static bool DisplayVersion = false; static std::string UnitTestLibrary; static std::string LogLevel; static std::atomic ForkCounter(0); static std::map childJails; #ifndef KIT_IN_PROCESS int ClientPortNumber = DEFAULT_CLIENT_PORT_NUMBER; std::string MasterLocation; #endif /// Dispatcher class to demultiplex requests from /// WSD and handles them. class CommandDispatcher : public IoUtil::PipeReader { public: CommandDispatcher(const int pipe) : PipeReader("wsd_pipe_rd", pipe) { } /// Polls WSD commands and handles them. bool pollAndDispatch() { std::string message; const int ready = readLine(message, [](){ return SigUtil::getTerminationFlag(); }); if (ready <= 0) { // Termination is done via SIGTERM, which breaks the wait. if (ready < 0) { if (SigUtil::getTerminationFlag()) { LOG_INF("Poll interrupted in " << getName() << " and TerminationFlag is set."); } // Break. return false; } // Timeout. return true; } LOG_INF("ForKit command: [" << message << "]."); try { std::vector tokens = LOOLProtocol::tokenize(message); if (tokens.size() == 2 && tokens[0] == "spawn") { const int count = std::stoi(tokens[1]); if (count > 0) { LOG_INF("Setting to spawn " << tokens[1] << " child" << (count == 1 ? "" : "ren") << " per request."); ForkCounter = count; } else { LOG_WRN("Cannot spawn " << tokens[1] << " children as requested."); } } else if (tokens.size() == 3 && tokens[0] == "setconfig") { // Currently only rlimit entries are supported. if (!Rlimit::handleSetrlimitCommand(tokens)) { LOG_ERR("Unknown setconfig command: " << message); } } else { LOG_ERR("Unknown command: " << message); } } catch (const std::exception& exc) { LOG_ERR("Error while processing forkit request [" << message << "]: " << exc.what()); } return true; } }; #ifndef KIT_IN_PROCESS static bool haveCapability(cap_value_t capability) { cap_t caps = cap_get_proc(); if (caps == nullptr) { LOG_SFL("cap_get_proc() failed."); return false; } char *cap_name = cap_to_name(capability); cap_flag_value_t value; if (cap_get_flag(caps, capability, CAP_EFFECTIVE, &value) == -1) { if (cap_name) { LOG_SFL("cap_get_flag failed for " << cap_name << "."); cap_free(cap_name); } else { LOG_SFL("cap_get_flag failed for capability " << capability << "."); } return false; } if (value != CAP_SET) { if (cap_name) { LOG_FTL("Capability " << cap_name << " is not set for the loolforkit program."); cap_free(cap_name); } else { LOG_ERR("Capability " << capability << " is not set for the loolforkit program."); } return false; } if (cap_name) { LOG_INF("Have capability " << cap_name); cap_free(cap_name); } else { LOG_INF("Have capability " << capability); } return true; } static bool haveCorrectCapabilities() { bool result = true; // Do check them all, don't shortcut with && if (!haveCapability(CAP_SYS_CHROOT)) result = false; if (!haveCapability(CAP_MKNOD)) result = false; if (!haveCapability(CAP_FOWNER)) result = false; return result; } #endif /// Check if some previously forked kids have died. static void cleanupChildren() { std::vector jails; Process::PID exitedChildPid; int status; // Reap quickly without doing slow cleanup so WSD can spawn more rapidly. while ((exitedChildPid = waitpid(-1, &status, WUNTRACED | WNOHANG)) > 0) { const auto it = childJails.find(exitedChildPid); if (it != childJails.end()) { LOG_INF("Child " << exitedChildPid << " has exited, will remove its jail [" << it->second << "]."); jails.emplace_back(it->second); childJails.erase(it); if (childJails.empty() && !SigUtil::getTerminationFlag()) { // We ran out of kits and we aren't terminating. LOG_WRN("No live Kits exist, and we are not terminating yet."); } } else { LOG_ERR("Unknown child " << exitedChildPid << " has exited"); } } // Now delete the jails. for (const auto& path : jails) { LOG_INF("Removing jail [" << path << "]."); FileUtil::removeFile(path, true); } } static int createLibreOfficeKit(const std::string& childRoot, const std::string& sysTemplate, const std::string& loTemplate, const std::string& loSubPath, bool queryVersion = false) { // Generate a jail ID to be used for in the jail path. const std::string jailId = Util::rng::getFilename(16); // Used to label the spare kit instances static size_t spareKitId = 0; ++spareKitId; LOG_DBG("Forking a loolkit process with jailId: " << jailId << " as spare loolkit #" << spareKitId << "."); const Process::PID pid = fork(); if (!pid) { // Child // Close the pipe from loolwsd close(0); #ifndef KIT_IN_PROCESS UnitKit::get().postFork(); #endif if (std::getenv("SLEEPKITFORDEBUGGER")) { const size_t delaySecs = std::stoul(std::getenv("SLEEPKITFORDEBUGGER")); if (delaySecs > 0) { std::cerr << "Kit: Sleeping " << delaySecs << " seconds to give you time to attach debugger to process " << Process::id() << std::endl; Thread::sleep(delaySecs * 1000); } } #ifndef KIT_IN_PROCESS lokit_main(childRoot, jailId, sysTemplate, loTemplate, loSubPath, NoCapsForKit, NoSeccomp, queryVersion, DisplayVersion, spareKitId); #else lokit_main(childRoot, jailId, sysTemplate, loTemplate, loSubPath, true, true, queryVersion, DisplayVersion, spareKitId); #endif } else { // Parent if (pid < 0) { LOG_SYS("Fork failed."); } else { LOG_INF("Forked kit [" << pid << "]."); childJails[pid] = childRoot + jailId; } #ifndef KIT_IN_PROCESS UnitKit::get().launchedKit(pid); #endif } return pid; } void forkLibreOfficeKit(const std::string& childRoot, const std::string& sysTemplate, const std::string& loTemplate, const std::string& loSubPath, int limit) { // Cleanup first, to reduce disk load. cleanupChildren(); #ifndef KIT_IN_PROCESS (void) limit; #else if (limit > 0) ForkCounter = limit; #endif if (ForkCounter > 0) { // Create as many as requested. const size_t count = ForkCounter; LOG_INF("Spawning " << count << " new child" << (count == 1 ? "." : "ren.")); const size_t retry = count * 2; for (size_t i = 0; ForkCounter > 0 && i < retry; ++i) { if (ForkCounter-- <= 0 || createLibreOfficeKit(childRoot, sysTemplate, loTemplate, loSubPath) < 0) { LOG_ERR("Failed to create a kit process."); ++ForkCounter; } } } } #ifndef KIT_IN_PROCESS static void printArgumentHelp() { std::cout << "Usage: loolforkit [OPTION]..." << std::endl; std::cout << " Single-threaded process that spawns lok instances" << std::endl; std::cout << " Note: Running this standalone is not possible. It is spawned by loolwsd" << std::endl; std::cout << " and is controlled via a pipe." << std::endl; std::cout << "" << std::endl; } int main(int argc, char** argv) { if (!hasCorrectUID("loolforkit")) { return EX_SOFTWARE; } if (std::getenv("SLEEPFORDEBUGGER")) { const size_t delaySecs = std::stoul(std::getenv("SLEEPFORDEBUGGER")); if (delaySecs > 0) { std::cerr << "Forkit: Sleeping " << delaySecs << " seconds to give you time to attach debugger to process " << Process::id() << std::endl; Thread::sleep(delaySecs * 1000); } } #ifndef FUZZER SigUtil::setFatalSignals(); SigUtil::setTerminationSignals(); #endif Util::setThreadName("forkit"); // Initialization const bool logToFile = std::getenv("LOOL_LOGFILE"); const char* logFilename = std::getenv("LOOL_LOGFILENAME"); const char* logLevel = std::getenv("LOOL_LOGLEVEL"); const char* logColor = std::getenv("LOOL_LOGCOLOR"); std::map logProperties; if (logToFile && logFilename) { logProperties["path"] = std::string(logFilename); } Log::initialize("frk", "trace", logColor != nullptr, logToFile, logProperties); LogLevel = logLevel ? logLevel : "trace"; if (LogLevel != "trace") { LOG_INF("Setting log-level to [trace] and delaying setting to configured [" << LogLevel << "] until after Forkit initialization."); } std::string childRoot; std::string loSubPath; std::string sysTemplate; std::string loTemplate; #if ENABLE_DEBUG static const char* clientPort = std::getenv("LOOL_TEST_CLIENT_PORT"); if (clientPort) ClientPortNumber = std::stoi(clientPort); static const char* masterPort = std::getenv("LOOL_TEST_MASTER_PORT"); if (masterPort) MasterLocation = masterPort; #endif for (int i = 0; i < argc; ++i) { char *cmd = argv[i]; char *eq; if (std::strstr(cmd, "--losubpath=") == cmd) { eq = std::strchr(cmd, '='); loSubPath = std::string(eq+1); } else if (std::strstr(cmd, "--systemplate=") == cmd) { eq = std::strchr(cmd, '='); sysTemplate = std::string(eq+1); } else if (std::strstr(cmd, "--lotemplate=") == cmd) { eq = std::strchr(cmd, '='); loTemplate = std::string(eq+1); } else if (std::strstr(cmd, "--childroot=") == cmd) { eq = std::strchr(cmd, '='); childRoot = std::string(eq+1); } else if (std::strstr(cmd, "--clientport=") == cmd) { eq = std::strchr(cmd, '='); ClientPortNumber = std::stoll(std::string(eq+1)); } else if (std::strstr(cmd, "--masterport=") == cmd) { eq = std::strchr(cmd, '='); MasterLocation = std::string(eq+1); } else if (std::strstr(cmd, "--version") == cmd) { std::string version, hash; Util::getVersionInfo(version, hash); std::cout << "loolforkit version details: " << version << " - " << hash << std::endl; DisplayVersion = true; } else if (std::strstr(cmd, "--rlimits") == cmd) { eq = std::strchr(cmd, '='); const std::string rlimits = std::string(eq+1); std::vector tokens = LOOLProtocol::tokenize(rlimits, ';'); for (const std::string& cmdLimit : tokens) { const std::pair pair = Util::split(cmdLimit, ':'); std::vector tokensLimit = { "setconfig", pair.first, pair.second }; if (!Rlimit::handleSetrlimitCommand(tokensLimit)) { LOG_ERR("Unknown rlimits command: " << cmdLimit); } } } #if ENABLE_DEBUG // this process has various privileges - don't run arbitrary code. else if (std::strstr(cmd, "--unitlib=") == cmd) { eq = std::strchr(cmd, '='); UnitTestLibrary = std::string(eq+1); } #endif // we are running in a lower-privilege mode - with no chroot else if (std::strstr(cmd, "--nocaps") == cmd) { LOG_ERR("Security: Running without the capability to enter a chroot jail is ill advised."); NoCapsForKit = true; } // we are running without seccomp protection else if (std::strstr(cmd, "--noseccomp") == cmd) { LOG_ERR("Security :Running without the ability to filter system calls is ill advised."); NoSeccomp = true; } } if (loSubPath.empty() || sysTemplate.empty() || loTemplate.empty() || childRoot.empty()) { printArgumentHelp(); return EX_USAGE; } if (!UnitBase::init(UnitBase::UnitType::Kit, UnitTestLibrary)) { LOG_ERR("Failed to load kit unit test library"); return EX_USAGE; } // Setup & check environment const std::string layers( "xcsxcu:${BRAND_BASE_DIR}/share/registry " "res:${BRAND_BASE_DIR}/share/registry " "bundledext:${${BRAND_BASE_DIR}/program/lounorc:BUNDLED_EXTENSIONS_USER}/registry/com.sun.star.comp.deployment.configuration.PackageRegistryBackend/configmgr.ini " "sharedext:${${BRAND_BASE_DIR}/program/lounorc:SHARED_EXTENSIONS_USER}/registry/com.sun.star.comp.deployment.configuration.PackageRegistryBackend/configmgr.ini " "userext:${${BRAND_BASE_DIR}/program/lounorc:UNO_USER_PACKAGES_CACHE}/registry/com.sun.star.comp.deployment.configuration.PackageRegistryBackend/configmgr.ini " #if ENABLE_DEBUG // '*' denotes non-writable. "user:*file://" DEBUG_ABSSRCDIR "/loolkitconfig.xcu " #else "user:*file://" LOOLWSD_CONFIGDIR "/loolkitconfig.xcu " #endif ); // No-caps tracing can spawn eg. glxinfo & other oddness. unsetenv("DISPLAY"); ::setenv("CONFIGURATION_LAYERS", layers.c_str(), 1 /* override */); if (!std::getenv("LD_BIND_NOW")) // must be set by parent. LOG_INF("Note: LD_BIND_NOW is not set."); if (!NoCapsForKit && !haveCorrectCapabilities()) { std::cerr << "FATAL: Capabilities are not set for the loolforkit program." << std::endl; std::cerr << "Please make sure that the current partition was *not* mounted with the 'nosuid' option." << std::endl; std::cerr << "If you are on SLES11, please set 'file_caps=1' as kernel boot option." << std::endl << std::endl; return EX_SOFTWARE; } // Set various options we need. std::string options = "unipoll"; if (Log::logger().trace()) options += ":profile_events"; // options += ":sc_no_grid_bg"; // leave ths disabled for now, merged-cells needs more work. ::setenv("SAL_LOK_OPTIONS", options.c_str(), 0); // Initialize LoKit if (!globalPreinit(loTemplate)) { LOG_FTL("Failed to preinit lokit."); Log::shutdown(); std::_Exit(EX_SOFTWARE); } if (Util::getProcessThreadCount() != 1) LOG_ERR("Error: forkit has more than a single thread after pre-init"); LOG_INF("Preinit stage OK."); // We must have at least one child, more are created dynamically. // Ask this first child to send version information to master process and trace startup. ::setenv("LOOL_TRACE_STARTUP", "1", 1); Process::PID forKitPid = createLibreOfficeKit(childRoot, sysTemplate, loTemplate, loSubPath, true); if (forKitPid < 0) { LOG_FTL("Failed to create a kit process."); Log::shutdown(); std::_Exit(EX_SOFTWARE); } // No need to trace subsequent children. ::unsetenv("LOOL_TRACE_STARTUP"); if (LogLevel != "trace") { LOG_INF("Forkit initialization complete: setting log-level to [" << LogLevel << "] as configured."); Log::logger().setLevel(LogLevel); } CommandDispatcher commandDispatcher(0); LOG_INF("ForKit process is ready."); while (!SigUtil::getTerminationFlag()) { UnitKit::get().invokeForKitTest(); if (!commandDispatcher.pollAndDispatch()) { LOG_INF("Child dispatcher flagged for termination."); break; } forkLibreOfficeKit(childRoot, sysTemplate, loTemplate, loSubPath); } int returnValue = EX_OK; UnitKit::get().returnValue(returnValue); #if 0 int status = 0; waitpid(forKitPid, &status, WUNTRACED); #endif LOG_INF("ForKit process finished."); Log::shutdown(); std::_Exit(returnValue); } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */