// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "nacl_io/kernel_object.h"
#include <assert.h>
#include <errno.h>
#include <fcntl.h>
#include <pthread.h>
#include <algorithm>
#include <map>
#include <string>
#include <vector>
#include "nacl_io/filesystem.h"
#include "nacl_io/kernel_handle.h"
#include "nacl_io/log.h"
#include "nacl_io/node.h"
#include "sdk_util/auto_lock.h"
#include "sdk_util/ref_object.h"
#include "sdk_util/scoped_ref.h"
namespace nacl_io {
KernelObject::KernelObject() : umask_(0) {
cwd_ = "/";
}
KernelObject::~KernelObject() {};
Error KernelObject::AttachFsAtPath(const ScopedFilesystem& fs,
const std::string& path) {
std::string abs_path = GetAbsParts(path).Join();
AUTO_LOCK(fs_lock_);
if (filesystems_.find(abs_path) != filesystems_.end()) {
LOG_ERROR("Can't mount at %s, it is already mounted.", path.c_str());
return EBUSY;
}
filesystems_[abs_path] = fs;
return 0;
}
Error KernelObject::DetachFsAtPath(const std::string& path,
ScopedFilesystem* out_fs) {
std::string abs_path = GetAbsParts(path).Join();
AUTO_LOCK(fs_lock_);
FsMap_t::iterator it = filesystems_.find(abs_path);
if (filesystems_.end() == it) {
LOG_TRACE("Can't unmount at %s, nothing is mounted.", path.c_str());
return EINVAL;
}
// It is only legal to unmount if there are no open references
if (it->second->RefCount() != 1) {
LOG_TRACE("Can't unmount at %s, refcount is != 1.", path.c_str());
return EBUSY;
}
*out_fs = it->second;
filesystems_.erase(it);
return 0;
}
// Uses longest prefix to find the filesystem for the give path, then
// acquires the filesystem and returns it with a relative path.
Error KernelObject::AcquireFsAndRelPath(const std::string& path,
ScopedFilesystem* out_fs,
Path* rel_parts) {
Path abs_parts = GetAbsParts(path);
out_fs->reset(NULL);
*rel_parts = Path();
AUTO_LOCK(fs_lock_);
// Find longest prefix
size_t max = abs_parts.Size();
for (size_t len = 0; len < abs_parts.Size(); len++) {
FsMap_t::iterator it = filesystems_.find(abs_parts.Range(0, max - len));
if (it != filesystems_.end()) {
rel_parts->Set("/");
rel_parts->Append(Path(abs_parts.Range(max - len, max)));
*out_fs = it->second;
return 0;
}
}
return ENOTDIR;
}
// Given a path, acquire the associated filesystem and node, creating the
// node if needed based on the provided flags.
Error KernelObject::AcquireFsAndNode(const std::string& path,
int oflags, mode_t mflags,
ScopedFilesystem* out_fs,
ScopedNode* out_node) {
Path rel_parts;
out_fs->reset(NULL);
out_node->reset(NULL);
Error error = AcquireFsAndRelPath(path, out_fs, &rel_parts);
if (error)
return error;
error = (*out_fs)->OpenWithMode(rel_parts, oflags, mflags, out_node);
if (error)
return error;
return 0;
}
Path KernelObject::GetAbsParts(const std::string& path) {
AUTO_LOCK(cwd_lock_);
Path abs_parts(cwd_);
return abs_parts.Append(Path(path));
}
std::string KernelObject::GetCWD() {
AUTO_LOCK(cwd_lock_);
std::string out = cwd_;
return out;
}
Error KernelObject::SetCWD(const std::string& path) {
std::string abs_path = GetAbsParts(path).Join();
ScopedFilesystem fs;
ScopedNode node;
Error error = AcquireFsAndNode(abs_path, O_RDONLY, 0, &fs, &node);
if (error)
return error;
if ((node->GetType() & S_IFDIR) == 0)
return ENOTDIR;
AUTO_LOCK(cwd_lock_);
cwd_ = abs_path;
return 0;
}
mode_t KernelObject::GetUmask() {
return umask_;
}
mode_t KernelObject::SetUmask(mode_t newmask) {
AUTO_LOCK(umask_lock_);
mode_t oldmask = umask_;
umask_ = newmask & S_MODEBITS;
return oldmask;
}
Error KernelObject::GetFDFlags(int fd, int* out_flags) {
AUTO_LOCK(handle_lock_);
if (fd < 0 || fd >= static_cast<int>(handle_map_.size()))
return EBADF;
*out_flags = handle_map_[fd].flags;
return 0;
}
Error KernelObject::SetFDFlags(int fd, int flags) {
AUTO_LOCK(handle_lock_);
if (fd < 0 || fd >= static_cast<int>(handle_map_.size()))
return EBADF;
// Only setting of FD_CLOEXEC is supported.
if (flags & ~FD_CLOEXEC)
return EINVAL;
handle_map_[fd].flags = flags;
return 0;
}
Error KernelObject::AcquireHandle(int fd, ScopedKernelHandle* out_handle) {
out_handle->reset(NULL);
AUTO_LOCK(handle_lock_);
if (fd < 0 || fd >= static_cast<int>(handle_map_.size()))
return EBADF;
Descriptor_t& desc = handle_map_[fd];
if (!desc.handle)
return EBADF;
*out_handle = desc.handle;
return 0;
}
Error KernelObject::AcquireHandleAndPath(int fd,
ScopedKernelHandle* out_handle,
std::string* out_path) {
out_handle->reset(NULL);
AUTO_LOCK(handle_lock_);
if (fd < 0 || fd >= static_cast<int>(handle_map_.size()))
return EBADF;
Descriptor_t& desc = handle_map_[fd];
if (!desc.handle)
return EBADF;
*out_handle = desc.handle;
*out_path = desc.path;
return 0;
}
int KernelObject::AllocateFD(const ScopedKernelHandle& handle,
const std::string& path) {
AUTO_LOCK(handle_lock_);
int id;
std::string abs_path = GetAbsParts(path).Join();
Descriptor_t descriptor(handle, abs_path);
// If we can recycle an FD, use that first
if (free_fds_.size()) {
// Force lower numbered FD to be available first.
// std::set is ordered, so begin() returns the lowest value.
id = *free_fds_.begin();
free_fds_.erase(id);
handle_map_[id] = descriptor;
} else {
id = handle_map_.size();
handle_map_.push_back(descriptor);
}
return id;
}
void KernelObject::FreeAndReassignFD(int fd,
const ScopedKernelHandle& handle,
const std::string& path) {
AUTO_LOCK(handle_lock_);
// If the required FD is larger than the current set, grow the set.
int sz = static_cast<int>(handle_map_.size());
if (fd >= sz) {
// Expand the handle map to include all the extra descriptors
// up to and including fd.
handle_map_.resize(fd + 1);
// Add all the new descriptors, except fd, to the free list.
for (; sz < fd; ++sz) {
free_fds_.insert(sz);
}
}
assert(handle != NULL);
free_fds_.erase(fd);
// This path will be from an existing handle, and absolute.
handle_map_[fd] = Descriptor_t(handle, path);
}
void KernelObject::FreeFD(int fd) {
AUTO_LOCK(handle_lock_);
handle_map_[fd].handle.reset(NULL);
free_fds_.insert(fd);
}
} // namespace nacl_io