blob: 7a771a8ff4bd0f747e1d6f6e37a3f16892cbb9a3 [file]
// Copyright 2021 The Emscripten Authors. All rights reserved.
// Emscripten is available under two separate licenses, the MIT license and the
// University of Illinois/NCSA Open Source License. Both these licenses can be
// found in the LICENSE file.
// This file defines the global state.
#include <emscripten/threading.h>
#include "memory_backend.h"
#include "paths.h"
#include "special_files.h"
#include "wasmfs.h"
#include "wasmfs_internal.h"
namespace wasmfs {
#ifdef WASMFS_CASE_INSENSITIVE
backend_t createIgnoreCaseBackend(std::function<backend_t()> createBackend);
#endif
// The below lines are included to make the compiler believe that the global
// constructor is part of a system header, which is necessary to work around a
// compilation error about using a reserved init priority less than 101. This
// ensures that the global state of the file system is constructed before
// anything else. ATTENTION: No other static global objects should be defined
// besides wasmFS. Due to # define _LIBCPP_INIT_PRIORITY_MAX
// __attribute__((init_priority(101))), we must use init priority 100 (reserved
// system priority) since wasmFS is a system level component.
// TODO: consider instead adding this in libc's startup code.
// WARNING: Maintain # n + 1 "wasmfs.cpp" 3 where n = line number.
# 29 "wasmfs.cpp" 3
__attribute__((init_priority(100))) WasmFS wasmFS;
# 31 "wasmfs.cpp"
// If the user does not implement this hook, do nothing.
__attribute__((weak)) extern "C" void wasmfs_before_preload(void) {}
// Set up global data structures and preload files.
WasmFS::WasmFS() : rootDirectory(initRootDirectory()), cwd(rootDirectory) {
wasmfs_before_preload();
preloadFiles();
}
// Manual integration with LSan. LSan installs itself during startup at the
// first allocation, which happens inside WasmFS code (since the WasmFS global
// object creates some data structures). As a result LSan's atexit() destructor
// will be called last, after WasmFS is cleaned up, since atexit() calls work
// are LIFO (like a stack). But that is a problem, since if WasmFS has shut
// down and deallocated itself then the leak code cannot actually print any of
// its findings, if it has any. To avoid that, define the LSan entry point as a
// weak symbol, and call it; if LSan is not enabled this can be optimized out,
// and if LSan is enabled then we'll check for leaks right at the start of the
// WasmFS destructor, which is the last time during which it is valid to print.
// (Note that this means we can't find leaks inside WasmFS code itself, but that
// seems fundamentally impossible for the above reasons, unless we made LSan log
// its findings in a way that does not depend on normal file I/O.)
__attribute__((weak)) extern "C" void __lsan_do_leak_check(void) {}
extern "C" void wasmfs_flush(void) {
// Flush musl libc streams.
fflush(0);
// Flush our own streams. TODO: flush all backends.
(void)SpecialFiles::getStdout()->locked().flush();
(void)SpecialFiles::getStderr()->locked().flush();
}
WasmFS::~WasmFS() {
// See comment above on this function.
//
// Note that it is ok for us to call it here, as LSan internally makes all
// calls after the first into no-ops. That is, calling it here makes the one
// time that leak checks are run be done here, or potentially earlier, but not
// later; and as mentioned in the comment above, this is the latest possible
// time for the checks to run (since right after this nothing can be printed).
__lsan_do_leak_check();
// TODO: Integrate musl exit() which would flush the libc part for us. That
// might also help with destructor priority - we need to happen last.
// (But we would still need to flush the internal WasmFS buffers, see
// wasmfs_flush() and the comment on it in the header.)
wasmfs_flush();
// Break the reference cycle caused by the root directory being its own
// parent.
rootDirectory->locked().setParent(nullptr);
}
// Special backends that want to install themselves as the root use this hook.
// Otherwise, we use the default backends.
__attribute__((weak)) extern "C" backend_t wasmfs_create_root_dir(void) {
#ifdef WASMFS_CASE_INSENSITIVE
return createIgnoreCaseBackend([]() { return createMemoryBackend(); });
#else
return createMemoryBackend();
#endif
}
std::shared_ptr<Directory> WasmFS::initRootDirectory() {
auto rootBackend = wasmfs_create_root_dir();
auto rootDirectory =
rootBackend->createDirectory(S_IRUGO | S_IXUGO | S_IWUGO);
auto lockedRoot = rootDirectory->locked();
// The root directory is its own parent.
lockedRoot.setParent(rootDirectory);
// If the /dev/ directory does not already exist, create it. (It may already
// exist in NODERAWFS mode, or if those files have been preloaded.)
auto devDir = lockedRoot.insertDirectory("dev", S_IRUGO | S_IXUGO);
if (devDir) {
auto lockedDev = devDir->locked();
lockedDev.mountChild("null", SpecialFiles::getNull());
lockedDev.mountChild("stdin", SpecialFiles::getStdin());
lockedDev.mountChild("stdout", SpecialFiles::getStdout());
lockedDev.mountChild("stderr", SpecialFiles::getStderr());
lockedDev.mountChild("random", SpecialFiles::getRandom());
lockedDev.mountChild("urandom", SpecialFiles::getURandom());
}
// As with the /dev/ directory, it is not an error for /tmp/ to already
// exist.
lockedRoot.insertDirectory("tmp", S_IRWXUGO);
return rootDirectory;
}
// Initialize files specified by the --preload-file option.
// Set up directories and files in wasmFS$preloadedDirs and
// wasmFS$preloadedFiles from JS. This function will be called before any file
// operation to ensure any preloaded files are eagerly available for use.
void WasmFS::preloadFiles() {
// Debug builds only: add check to ensure preloadFiles() is called once.
#ifndef NDEBUG
static std::atomic<int> timesCalled;
timesCalled++;
assert(timesCalled == 1);
#endif
// Ensure that files are preloaded from the main thread.
assert(emscripten_is_main_runtime_thread());
auto numFiles = _wasmfs_get_num_preloaded_files();
auto numDirs = _wasmfs_get_num_preloaded_dirs();
// If there are no preloaded files, exit early.
if (numDirs == 0 && numFiles == 0) {
return;
}
// Iterate through wasmFS$preloadedDirs to obtain a parent and child pair.
// Ex. Module['FS_createPath']("/foo/parent", "child", true, true);
for (int i = 0; i < numDirs; i++) {
char parentPath[PATH_MAX] = {};
_wasmfs_get_preloaded_parent_path(i, parentPath);
auto parsed = path::parseFile(parentPath);
std::shared_ptr<Directory> parentDir;
if (parsed.getError() ||
!(parentDir = parsed.getFile()->dynCast<Directory>())) {
emscripten_err(
"Fatal error during directory creation in file preloading.");
abort();
}
char childName[PATH_MAX] = {};
_wasmfs_get_preloaded_child_path(i, childName);
auto lockedParentDir = parentDir->locked();
if (lockedParentDir.getChild(childName)) {
// The child already exists, so we don't need to do anything here.
continue;
}
auto inserted =
lockedParentDir.insertDirectory(childName, S_IRUGO | S_IXUGO);
assert(inserted && "TODO: handle preload insertion errors");
}
for (int i = 0; i < numFiles; i++) {
char fileName[PATH_MAX] = {};
_wasmfs_get_preloaded_path_name(i, fileName);
auto mode = _wasmfs_get_preloaded_file_mode(i);
auto parsed = path::parseParent(fileName);
if (parsed.getError()) {
emscripten_err("Fatal error during file preloading");
abort();
}
auto& [parent, childName] = parsed.getParentChild();
auto created =
parent->locked().insertDataFile(std::string(childName), (mode_t)mode);
assert(created && "TODO: handle preload insertion errors");
created->locked().preloadFromJS(i);
}
}
} // namespace wasmfs