// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef LIBRARIES_NACL_IO_NODE_H_
#define LIBRARIES_NACL_IO_NODE_H_
#include <stdarg.h>
#include <string>
#include "nacl_io/error.h"
#include "nacl_io/event_listener.h"
#include "nacl_io/log.h"
#include "nacl_io/osdirent.h"
#include "nacl_io/osstat.h"
#include "nacl_io/ostermios.h"
#include "sdk_util/ref_object.h"
#include "sdk_util/scoped_ref.h"
#include "sdk_util/simple_lock.h"
#define S_IRALL (S_IRUSR | S_IRGRP | S_IROTH)
#define S_IWALL (S_IWUSR | S_IWGRP | S_IWOTH)
#define S_IXALL (S_IXUSR | S_IXGRP | S_IXOTH)
#define S_MODEBITS 07777
namespace nacl_io {
class Filesystem;
class Node;
struct HandleAttr;
typedef sdk_util::ScopedRef<Node> ScopedNode;
// NOTE: The KernelProxy is the only class that should be setting errno. All
// other classes should return Error (as defined by nacl_io/error.h).
class Node : public sdk_util::RefObject {
protected:
explicit Node(Filesystem* filesystem);
virtual ~Node();
protected:
virtual Error Init(int open_flags);
virtual void Destroy();
public:
// Return true if the node permissions match the given open mode.
virtual bool CanOpen(int open_flags);
// Returns the emitter for this Node if it has one, if not, assume this
// object can not block.
virtual EventEmitter* GetEventEmitter();
virtual uint32_t GetEventStatus();
// Normal OS operations on a node (file), can be called by the kernel
// directly so it must lock and unlock appropriately. These functions
// must not be called by the filesystem.
virtual Error FSync();
// It is expected that the derived Node will fill with 0 when growing
// the file.
virtual Error FTruncate(off_t length);
// Assume that |out_bytes| is non-NULL.
virtual Error GetDents(size_t offs,
struct dirent* pdir,
size_t count,
int* out_bytes);
// Assume that |stat| is non-NULL.
virtual Error GetStat(struct stat* stat);
// Assume that |arg| is non-NULL.
Error Ioctl(int request, ...);
virtual Error VIoctl(int request, va_list args);
// Assume that |buf| and |out_bytes| are non-NULL.
virtual Error Read(const HandleAttr& attr,
void* buf,
size_t count,
int* out_bytes);
// Assume that |buf| and |out_bytes| are non-NULL.
virtual Error Write(const HandleAttr& attr,
const void* buf,
size_t count,
int* out_bytes);
// Assume that |addr| and |out_addr| are non-NULL.
virtual Error MMap(void* addr,
size_t length,
int prot,
int flags,
size_t offset,
void** out_addr);
virtual Error Tcflush(int queue_selector);
virtual Error Tcgetattr(struct termios* termios_p);
virtual Error Tcsetattr(int optional_actions,
const struct termios* termios_p);
virtual Error Futimens(const struct timespec times[2]);
virtual Error Fchmod(mode_t mode);
virtual int GetLinks();
virtual int GetMode();
virtual void SetMode(int mode);
virtual int GetType();
virtual void SetType(int type);
// Assume that |out_size| is non-NULL.
virtual Error GetSize(off_t* out_size);
// Returns 0 if node is a TTY
virtual Error Isatty();
virtual bool IsaDir();
virtual bool IsaFile();
virtual bool IsaSock();
virtual bool IsSeekable();
// Number of children for this node (directory)
virtual int ChildCount();
protected:
// Directory operations on the node are done by the Filesystem. The
// filesystem's lock must be held while these calls are made.
// Adds or removes a directory entry updating the link numbers and refcount
// Assumes that |node| is non-NULL.
virtual Error AddChild(const std::string& name, const ScopedNode& node);
virtual Error RemoveChild(const std::string& name);
// Find a child and return it without updating the refcount
// Assumes that |out_node| is non-NULL.
virtual Error FindChild(const std::string& name, ScopedNode* out_node);
// Update the link count
virtual void Link();
virtual void Unlink();
// Update the a/m/c time in stat_.
enum {
UPDATE_ATIME = 1,
UPDATE_MTIME = 2,
UPDATE_CTIME = 4,
};
void UpdateTime(int update_bits);
protected:
struct stat stat_;
sdk_util::SimpleLock node_lock_;
// We use a pointer directly to avoid cycles in the ref count.
// TODO(bradnelson) We should change this so it's unnecessary for the node
// to track it's parent. When a node is unlinked, the filesystem should do
// any cleanup it needs.
Filesystem* filesystem_;
friend class DevFs;
friend class DirNode;
friend class Filesystem;
friend class FuseFs;
friend class Html5Fs;
friend class HttpFs;
friend class MemFs;
};
} // namespace nacl_io
#endif // LIBRARIES_NACL_IO_NODE_H_