haiku/src/add-ons/kernel/file_systems/ramfs/kernel_interface.cpp
Alexander von Gluck IV 3de080c186 change format of fs_volume to match current kernel filesystem interface
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@42016 a95241bf-73f2-0310-859d-f6bbb57e9c96
2011-06-07 18:07:25 +00:00

2191 lines
55 KiB
C++

// kernel_interface.cpp
//
// Copyright (c) 2003-2008, Axel Dörfler (axeld@pinc-software.de)
// Copyright (c) 2003, Ingo Weinhold (bonefish@cs.tu-berlin.de)
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
//
// You can alternatively use *this file* under the terms of the the MIT
// license included in this package.
#include <ctype.h>
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/stat.h>
#include <fs_index.h>
#include <fs_info.h>
#include <fs_interface.h>
#include <fs_query.h>
#include <fs_volume.h>
#include <KernelExport.h>
#include <NodeMonitor.h>
#include <TypeConstants.h>
#include <AutoDeleter.h>
#include "AllocationInfo.h"
#include "AttributeIndex.h"
#include "AttributeIterator.h"
#include "Debug.h"
#include "Directory.h"
#include "Entry.h"
#include "EntryIterator.h"
#include "File.h"
#include "Index.h"
#include "IndexDirectory.h"
#include "Locking.h"
#include "Misc.h"
#include "Node.h"
#include "Query.h"
#include "ramfs_ioctl.h"
#include "SymLink.h"
#include "Volume.h"
static const size_t kOptimalIOSize = 65536;
static const bigtime_t kNotificationInterval = 1000000LL;
// notify_if_stat_changed
void
notify_if_stat_changed(Volume *volume, Node *node)
{
if (volume && node && node->IsModified()) {
uint32 statFields = node->MarkUnmodified();
notify_stat_changed(volume->GetID(), node->GetID(), statFields);
}
}
// #pragma mark - FS
// ramfs_mount
static status_t
ramfs_mount(dev_t nsid, const char* /*device*/, uint32 flags,
const char* /*args*/, fs_volume* _volume, ino_t* rootID)
{
FUNCTION_START();
// parameters are ignored for now
// fail, if read-only mounting is requested
if (flags & B_MOUNT_READ_ONLY)
return B_BAD_VALUE;
// allocate and init the volume
Volume *volume = new(std::nothrow) Volume(_volume);
if (volume == NULL)
return B_NO_MEMORY;
status_t status = volume->Mount(nsid);
if (status != B_OK) {
delete volume;
RETURN_ERROR(status);
}
*rootID = volume->GetRootDirectory()->GetID();
_volume->private_volume = volume;
RETURN_ERROR(B_OK);
}
// ramfs_unmount
static status_t
ramfs_unmount(fs_volume* _volume)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = volume->Unmount();
if (error == B_OK)
delete volume;
if (error != B_OK)
REPORT_ERROR(error);
return error;
}
// ramfs_read_fs_info
static status_t
ramfs_read_fs_info(fs_volume* _volume, struct fs_info *info)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
info->flags = B_FS_IS_PERSISTENT | B_FS_HAS_ATTR | B_FS_HAS_MIME
| B_FS_HAS_QUERY;
info->block_size = volume->GetBlockSize();
info->io_size = kOptimalIOSize;
info->total_blocks = volume->CountBlocks();
info->free_blocks = volume->CountFreeBlocks();
info->device_name[0] = '\0';
strncpy(info->volume_name, volume->GetName(), sizeof(info->volume_name));
strcpy(info->fsh_name, "ramfs");
} else
SET_ERROR(error, B_ERROR);
return B_OK;
}
// ramfs_write_fs_info
static status_t
ramfs_write_fs_info(fs_volume* _volume, const struct fs_info *info, uint32 mask)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
if (mask & FS_WRITE_FSINFO_NAME)
error = volume->SetName(info->volume_name);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_sync
static status_t
ramfs_sync(fs_volume /*fs*/)
{
FUNCTION_START();
return B_OK;
}
// #pragma mark - VNodes
// ramfs_lookup
static status_t
ramfs_lookup(fs_volume* _volume, fs_vnode _dir, const char *entryName, ino_t *vnid,
int *type)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
FUNCTION(("dir: (%llu), entry: `%s'\n", (dir ? dir->GetID() : -1), entryName));
// check for non-directories
if (!dir)
RETURN_ERROR(B_NOT_A_DIRECTORY);
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
Node *node = NULL;
// special entries: "." and ".."
if (!strcmp(entryName, ".")) {
*vnid = dir->GetID();
if (volume->GetVNode(*vnid, &node) != B_OK)
error = B_BAD_VALUE;
} else if (!strcmp(entryName, "..")) {
Directory *parent = dir->GetParent();
if (parent && volume->GetVNode(parent->GetID(), &node) == B_OK)
*vnid = node->GetID();
else
error = B_BAD_VALUE;
// ordinary entries
} else {
// find the entry
error = dir->FindAndGetNode(entryName, &node);
SET_ERROR(error, error);
if (error == B_OK)
*vnid = node->GetID();
}
// get the node's type
if (error == B_OK)
*type = node->GetMode() & S_IFMT;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_read_vnode
static status_t
ramfs_read_vnode(fs_volume* _volume, ino_t vnid, fs_vnode *node, bool reenter)
{
// FUNCTION_START();
FUNCTION(("node: %Ld\n", vnid));
Volume* volume = (Volume*)_volume->private_volume;
Node *foundNode = NULL;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
error = volume->FindNode(vnid, &foundNode);
if (error == B_OK)
*node = foundNode;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_write_vnode
static status_t
ramfs_write_vnode(fs_volume /*fs*/, fs_vnode DARG(_node), bool /*reenter*/)
{
// DANGER: If dbg_printf() is used, this thread will enter another FS and
// even perform a write operation. The is dangerous here, since this hook
// may be called out of the other FSs, since, for instance a put_vnode()
// called from another FS may cause the VFS layer to free vnodes and thus
// invoke this hook.
// FUNCTION_START();
//FUNCTION(("node: %Ld\n", ((Node*)_node)->GetID()));
status_t error = B_OK;
RETURN_ERROR(error);
}
// ramfs_remove_vnode
static status_t
ramfs_remove_vnode(fs_volume* _volume, fs_vnode _node, bool /*reenter*/)
{
FUNCTION(("node: %Ld\n", ((Node*)_node)->GetID()));
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
volume->NodeRemoved(node);
delete node;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// #pragma mark - Nodes
// ramfs_ioctl
static status_t
ramfs_ioctl(fs_volume* _volume, fs_vnode /*node*/, fs_cookie /*cookie*/, uint32 cmd,
void *buffer, size_t /*length*/)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
switch (cmd) {
case RAMFS_IOCTL_GET_ALLOCATION_INFO:
{
if (buffer) {
if (VolumeReadLocker locker = volume) {
AllocationInfo *info = (AllocationInfo*)buffer;
volume->GetAllocationInfo(*info);
} else
SET_ERROR(error, B_ERROR);
} else
SET_ERROR(error, B_BAD_VALUE);
break;
}
case RAMFS_IOCTL_DUMP_INDEX:
{
if (buffer) {
if (VolumeReadLocker locker = volume) {
const char *name = (const char*)buffer;
PRINT((" RAMFS_IOCTL_DUMP_INDEX, `%s'\n", name));
IndexDirectory *indexDir = volume->GetIndexDirectory();
if (indexDir) {
if (Index *index = indexDir->FindIndex(name))
index->Dump();
else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ERROR);
} else
SET_ERROR(error, B_BAD_VALUE);
break;
}
default:
error = B_DEV_INVALID_IOCTL;
break;
}
RETURN_ERROR(error);
}
// ramfs_set_flags
static status_t
ramfs_set_flags(fs_volume /*fs*/, fs_vnode /*node*/, fs_cookie /*cookie*/,
int /*flags*/)
{
FUNCTION_START();
// TODO:...
return B_OK;
}
// ramfs_fsync
static status_t
ramfs_fsync(fs_volume /*fs*/, fs_vnode /*node*/)
{
FUNCTION_START();
return B_OK;
}
// ramfs_read_symlink
static status_t
ramfs_read_symlink(fs_volume* _volume, fs_vnode _node, char *buffer,
size_t *bufferSize)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// read symlinks only
if (!node->IsSymLink())
error = B_BAD_VALUE;
if (error == B_OK) {
if (SymLink *symLink = dynamic_cast<SymLink*>(node)) {
// copy the link contents
size_t toRead = min(*bufferSize,
symLink->GetLinkedPathLength());
if (toRead > 0)
memcpy(buffer, symLink->GetLinkedPath(), toRead);
*bufferSize = toRead;
} else {
FATAL(("Node %Ld pretends to be a SymLink, but isn't!\n",
node->GetID()));
error = B_BAD_VALUE;
}
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_create_symlink
static status_t
ramfs_create_symlink(fs_volume* _volume, fs_vnode _dir, const char *name,
const char *path, int mode)
{
FUNCTION(("name: `%s', path: `%s'\n", name, path));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
status_t error = B_OK;
// check name
if (!name || *name == '\0') {
SET_ERROR(error, B_BAD_VALUE);
// check directory
} else if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// directory deleted?
bool removed;
if (get_vnode_removed(volume->GetID(), dir->GetID(), &removed) != B_OK
|| removed) {
SET_ERROR(error, B_NOT_ALLOWED);
}
// check directory write permissions
error = dir->CheckPermissions(ACCESS_W);
Node *node = NULL;
if (error == B_OK) {
// check if entry does already exist
if (dir->FindNode(name, &node) == B_OK) {
SET_ERROR(error, B_FILE_EXISTS);
} else {
// entry doesn't exist: create a symlink
SymLink *symLink = NULL;
error = dir->CreateSymLink(name, path, &symLink);
if (error == B_OK) {
node = symLink;
// set permissions, owner and group
node->SetMode(mode);
node->SetUID(geteuid());
node->SetGID(getegid());
// put the node
volume->PutVNode(node->GetID());
}
}
}
NodeMTimeUpdater mTimeUpdater2(node);
// notify listeners
if (error == B_OK) {
notify_entry_created(volume->GetID(), dir->GetID(), name,
node->GetID());
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_link
static status_t
ramfs_link(fs_volume* _volume, fs_vnode _dir, const char *name, fs_vnode _node)
{
FUNCTION(("name: `%s'\n", name));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
Node *node = (Node*)_node;
status_t error = B_OK;
// check directory
if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// directory deleted?
bool removed;
if (get_vnode_removed(volume->GetID(), dir->GetID(), &removed) != B_OK
|| removed) {
SET_ERROR(error, B_NOT_ALLOWED);
}
// check directory write permissions
error = dir->CheckPermissions(ACCESS_W);
Entry *entry = NULL;
if (error == B_OK) {
// check if entry does already exist
if (dir->FindEntry(name, &entry) == B_OK) {
SET_ERROR(error, B_FILE_EXISTS);
} else {
// entry doesn't exist: create a link
error = dir->CreateEntry(node, name);
}
}
// notify listeners
if (error == B_OK) {
notify_entry_created(volume->GetID(), dir->GetID(), name,
node->GetID());
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_unlink
static status_t
ramfs_unlink(fs_volume* _volume, fs_vnode _dir, const char *name)
{
FUNCTION(("name: `%s'\n", name));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
status_t error = B_OK;
// check name
if (!name || *name == '\0' || !strcmp(name, ".") || !strcmp(name, "..")) {
SET_ERROR(error, B_BAD_VALUE);
// check node
} else if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// check directory write permissions
error = dir->CheckPermissions(ACCESS_W);
ino_t nodeID = -1;
if (error == B_OK) {
// check if entry exists
Node *node = NULL;
Entry *entry = NULL;
if (dir->FindAndGetNode(name, &node, &entry) == B_OK) {
nodeID = node->GetID();
// unlink the entry, if it isn't a non-empty directory
if (node->IsDirectory()
&& !dynamic_cast<Directory*>(node)->IsEmpty()) {
SET_ERROR(error, B_DIRECTORY_NOT_EMPTY);
} else
error = dir->DeleteEntry(entry);
volume->PutVNode(node);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
}
// notify listeners
if (error == B_OK)
notify_entry_removed(volume->GetID(), dir->GetID(), name, nodeID);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_rename
static status_t
ramfs_rename(fs_volume* _volume, fs_vnode _oldDir, const char *oldName,
fs_vnode _newDir, const char *newName)
{
Volume* volume = (Volume*)_volume->private_volume;
Directory *oldDir = dynamic_cast<Directory*>((Node*)_oldDir);
Directory *newDir = dynamic_cast<Directory*>((Node*)_newDir);
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
FUNCTION(("old dir: %Ld, old name: `%s', new dir: %Ld, new name: `%s'\n",
oldDir->GetID(), oldName, newDir->GetID(), newName));
NodeMTimeUpdater mTimeUpdater1(oldDir);
NodeMTimeUpdater mTimeUpdater2(newDir);
// target directory deleted?
bool removed;
if (get_vnode_removed(volume->GetID(), newDir->GetID(), &removed)
!= B_OK
|| removed) {
SET_ERROR(error, B_NOT_ALLOWED);
}
// check directory write permissions
if (error == B_OK)
error = oldDir->CheckPermissions(ACCESS_W);
if (error == B_OK)
error = newDir->CheckPermissions(ACCESS_W);
Node *node = NULL;
Entry *entry = NULL;
if (error == B_OK) {
// check if entry exists
if (oldDir->FindAndGetNode(oldName, &node, &entry) != B_OK) {
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else {
if (oldDir != newDir) {
// check whether the entry is a descendent of the target
// directory
for (Directory *parent = newDir;
parent;
parent = parent->GetParent()) {
if (parent == node) {
error = B_BAD_VALUE;
break;
} else if (parent == oldDir)
break;
}
}
}
// check the target directory situation
Node *clobberNode = NULL;
Entry *clobberEntry = NULL;
if (error == B_OK) {
if (newDir->FindAndGetNode(newName, &clobberNode,
&clobberEntry) == B_OK) {
if (clobberNode->IsDirectory()
&& !dynamic_cast<Directory*>(clobberNode)->IsEmpty()) {
SET_ERROR(error, B_NAME_IN_USE);
}
}
}
// do the job
if (error == B_OK) {
// temporarily acquire an additional reference to make
// sure the node isn't deleted when we remove the entry
error = node->AddReference();
if (error == B_OK) {
// delete the original entry
error = oldDir->DeleteEntry(entry);
if (error == B_OK) {
// create the new one/relink the target entry
if (clobberEntry)
error = clobberEntry->Link(node);
else
error = newDir->CreateEntry(node, newName);
if (error == B_OK) {
// send a "removed" notification for the clobbered
// entry
if (clobberEntry) {
notify_entry_removed(volume->GetID(),
newDir->GetID(), newName,
clobberNode->GetID());
}
} else {
// try to recreate the original entry, in case of
// failure
newDir->CreateEntry(node, oldName);
}
}
node->RemoveReference();
}
}
// release the entries
if (clobberEntry)
volume->PutVNode(clobberNode);
if (entry)
volume->PutVNode(node);
}
// notify listeners
if (error == B_OK) {
notify_entry_moved(volume->GetID(), oldDir->GetID(), oldName,
newDir->GetID(), newName, node->GetID());
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_access
static status_t
ramfs_access(fs_volume* _volume, fs_vnode _node, int mode)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
error = node->CheckPermissions(mode);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_read_stat
static status_t
ramfs_read_stat(fs_volume* _volume, fs_vnode _node, struct stat *st)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("node: %Ld\n", node->GetID()));
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
st->st_dev = volume->GetID();
st->st_ino = node->GetID();
st->st_mode = node->GetMode();
st->st_nlink = node->GetRefCount();
st->st_uid = node->GetUID();
st->st_gid = node->GetGID();
st->st_size = node->GetSize();
st->st_blksize = kOptimalIOSize;
st->st_atime = node->GetATime();
st->st_mtime = node->GetMTime();
st->st_ctime = node->GetCTime();
st->st_crtime = node->GetCrTime();
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_write_stat
static status_t
ramfs_write_stat(fs_volume* _volume, fs_vnode _node, const struct stat *st,
uint32 mask)
{
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("mask: %lx\n", mask));
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(node);
// check permissions
error = node->CheckPermissions(ACCESS_W);
// size
if (error == B_OK && (mask & B_STAT_SIZE))
error = node->SetSize(st->st_size);
if (error == B_OK) {
// permissions
if (mask & B_STAT_MODE) {
node->SetMode(node->GetMode() & ~S_IUMSK
| st->st_mode & S_IUMSK);
}
// UID
if (mask & B_STAT_UID)
node->SetUID(st->st_uid);
// GID
if (mask & B_STAT_GID)
node->SetGID(st->st_gid);
// mtime
if (mask & B_STAT_MODIFICATION_TIME)
node->SetMTime(st->st_mtime);
// crtime
if (mask & B_STAT_CREATION_TIME)
node->SetCrTime(st->st_crtime);
}
// notify listeners
if (error == B_OK)
notify_if_stat_changed(volume, node);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// #pragma mark - Files
// FileCookie
class FileCookie {
public:
FileCookie(int openMode) : fOpenMode(openMode), fLastNotificationTime(0) {}
inline int GetOpenMode() { return fOpenMode; }
inline bigtime_t GetLastNotificationTime()
{ return fLastNotificationTime; }
inline bool NotificationIntervalElapsed(bool set = false)
{
bigtime_t currentTime = system_time();
bool result = (currentTime - fLastNotificationTime
> kNotificationInterval);
if (set && result)
fLastNotificationTime = currentTime;
return result;
}
private:
int fOpenMode;
bigtime_t fLastNotificationTime;
};
// ramfs_create
static status_t
ramfs_create(fs_volume* _volume, fs_vnode _dir, const char *name, int openMode,
int mode, fs_cookie *_cookie, ino_t *vnid)
{
// FUNCTION_START();
FUNCTION(("name: `%s', open mode: %x, mode: %x\n", name, openMode, mode));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
status_t error = B_OK;
// check name
if (!name || *name == '\0') {
SET_ERROR(error, B_BAD_VALUE);
// check directory
} else if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// directory deleted?
bool removed;
if (get_vnode_removed(volume->GetID(), dir->GetID(), &removed) != B_OK
|| removed) {
SET_ERROR(error, B_NOT_ALLOWED);
}
// create the file cookie
FileCookie *cookie = NULL;
if (error == B_OK) {
cookie = new(nothrow) FileCookie(openMode);
if (!cookie)
SET_ERROR(error, B_NO_MEMORY);
}
Node *node = NULL;
if (error == B_OK) {
// check if entry does already exist
if (dir->FindNode(name, &node) == B_OK) {
// entry does already exist
// fail, if we shall fail, when the file exists
if (openMode & O_EXCL) {
SET_ERROR(error, B_FILE_EXISTS);
// don't create a file over an existing directory or symlink
} else if (!node->IsFile()) {
SET_ERROR(error, B_NOT_ALLOWED);
// the user must have write permission for an existing entry
} else if ((error = node->CheckPermissions(ACCESS_W))
== B_OK) {
// truncate, if requested
if (openMode & O_TRUNC)
error = node->SetSize(0);
// we ignore the supplied permissions in this case
// get vnode
if (error == B_OK) {
*vnid = node->GetID();
error = volume->GetVNode(node->GetID(), &node);
}
}
// the user must have dir write permission to create a new entry
} else if ((error = dir->CheckPermissions(ACCESS_W)) == B_OK) {
// entry doesn't exist: create a file
File *file = NULL;
error = dir->CreateFile(name, &file);
if (error == B_OK) {
node = file;
*vnid = node->GetID();
// set permissions, owner and group
node->SetMode(mode);
node->SetUID(geteuid());
node->SetGID(getegid());
}
}
// set result / cleanup on failure
if (error == B_OK)
*_cookie = cookie;
else if (cookie)
delete cookie;
}
NodeMTimeUpdater mTimeUpdater2(node);
// notify listeners
if (error == B_OK)
notify_entry_created(volume->GetID(), dir->GetID(), name, *vnid);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_open
static status_t
ramfs_open(fs_volume* _volume, fs_vnode _node, int openMode, fs_cookie *_cookie)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("node: %Ld\n", node->GetID()));
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// directory can be opened read-only
if (node->IsDirectory() && (openMode & O_RWMASK))
openMode &= ~O_RWMASK;
int accessMode = open_mode_to_access(openMode);
// truncating requires write permission
if (error == B_OK && (openMode & O_TRUNC))
accessMode |= ACCESS_W;
// check open mode against permissions
if (error == B_OK)
error = node->CheckPermissions(accessMode);
// create the cookie
FileCookie *cookie = NULL;
if (error == B_OK) {
cookie = new(nothrow) FileCookie(openMode);
if (!cookie)
SET_ERROR(error, B_NO_MEMORY);
}
// truncate if requested
if (error == B_OK && (openMode & O_TRUNC))
error = node->SetSize(0);
NodeMTimeUpdater mTimeUpdater(node);
// set result / cleanup on failure
if (error == B_OK)
*_cookie = cookie;
else if (cookie)
delete cookie;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_close
static status_t
ramfs_close(fs_volume* _volume, fs_vnode _node, fs_cookie /*cookie*/)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("node: %Ld\n", node->GetID()));
status_t error = B_OK;
// notify listeners
if (VolumeReadLocker locker = volume) {
notify_if_stat_changed(volume, node);
} else
SET_ERROR(error, B_ERROR);
return B_OK;
}
// ramfs_free_cookie
static status_t
ramfs_free_cookie(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie _cookie)
{
FUNCTION_START();
FileCookie *cookie = (FileCookie*)_cookie;
delete cookie;
return B_OK;
}
// ramfs_read
static status_t
ramfs_read(fs_volume* _volume, fs_vnode _node, fs_cookie _cookie, off_t pos,
void *buffer, size_t *bufferSize)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FileCookie *cookie = (FileCookie*)_cookie;
// FUNCTION(("((%lu, %lu), %Ld, %p, %lu)\n", node->GetDirID(),
// node->GetObjectID(), pos, buffer, *bufferSize));
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// don't read anything but files
if (!node->IsFile())
SET_ERROR(error, B_BAD_VALUE);
// check, if reading is allowed
int rwMode = cookie->GetOpenMode() & O_RWMASK;
if (error == B_OK && rwMode != O_RDONLY && rwMode != O_RDWR)
SET_ERROR(error, B_FILE_ERROR);
// read
if (error == B_OK) {
if (File *file = dynamic_cast<File*>(node))
error = file->ReadAt(pos, buffer, *bufferSize, bufferSize);
else {
FATAL(("Node %Ld pretends to be a File, but isn't!\n",
node->GetID()));
error = B_BAD_VALUE;
}
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_write
static status_t
ramfs_write(fs_volume* _volume, fs_vnode _node, fs_cookie _cookie, off_t pos,
const void *buffer, size_t *bufferSize)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FileCookie *cookie = (FileCookie*)_cookie;
// FUNCTION(("((%lu, %lu), %Ld, %p, %lu)\n", node->GetDirID(),
// node->GetObjectID(), pos, buffer, *bufferSize));
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
// don't write anything but files
if (!node->IsFile())
SET_ERROR(error, B_BAD_VALUE);
if (error == B_OK) {
// check, if reading is allowed
int rwMode = cookie->GetOpenMode() & O_RWMASK;
if (error == B_OK && rwMode != O_WRONLY && rwMode != O_RDWR)
SET_ERROR(error, B_FILE_ERROR);
if (error == B_OK) {
// reset the position, if opened in append mode
if (cookie->GetOpenMode() & O_APPEND)
pos = node->GetSize();
// write
if (File *file = dynamic_cast<File*>(node)) {
error = file->WriteAt(pos, buffer, *bufferSize,
bufferSize);
} else {
FATAL(("Node %Ld pretends to be a File, but isn't!\n",
node->GetID()));
error = B_BAD_VALUE;
}
}
}
// notify listeners
if (error == B_OK && cookie->NotificationIntervalElapsed(true))
notify_if_stat_changed(volume, node);
NodeMTimeUpdater mTimeUpdater(node);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// #pragma mark - Directories
// DirectoryCookie
class DirectoryCookie {
public:
DirectoryCookie(Directory *directory = NULL)
: fIterator(directory),
fDotIndex(DOT_INDEX),
// debugging
fIteratorID(atomic_add(&fNextIteratorID, 1)),
fGetNextCounter(0)
{
}
void Unset() { fIterator.Unset(); }
// EntryIterator *GetIterator() const { return &fIterator; }
status_t GetNext(ino_t *nodeID, const char **entryName)
{
fGetNextCounter++;
status_t error = B_OK;
if (fDotIndex == DOT_INDEX) {
// "."
Node *entry = fIterator.GetDirectory();
*nodeID = entry->GetID();
*entryName = ".";
fDotIndex++;
} else if (fDotIndex == DOT_DOT_INDEX) {
// ".."
Directory *dir = fIterator.GetDirectory();
if (dir->GetParent())
*nodeID = dir->GetParent()->GetID();
else
*nodeID = dir->GetID();
*entryName = "..";
fDotIndex++;
} else {
// ordinary entries
Entry *entry = NULL;
error = fIterator.GetNext(&entry);
if (error == B_OK) {
*nodeID = entry->GetNode()->GetID();
*entryName = entry->GetName();
}
}
PRINT(("EntryIterator %ld, GetNext() counter: %ld, entry: %p (%Ld)\n",
fIteratorID, fGetNextCounter, fIterator.GetCurrent(),
(fIterator.GetCurrent() ? fIterator.GetCurrent()->GetNode()->GetID() : -1)));
return error;
}
status_t Rewind()
{
fDotIndex = DOT_INDEX;
return fIterator.Rewind();
}
status_t Suspend() { return fIterator.Suspend(); }
status_t Resume() { return fIterator.Resume(); }
private:
enum {
DOT_INDEX = 0,
DOT_DOT_INDEX = 1,
ENTRY_INDEX = 2,
};
private:
EntryIterator fIterator;
uint32 fDotIndex;
// debugging
int32 fIteratorID;
int32 fGetNextCounter;
static vint32 fNextIteratorID;
};
vint32 DirectoryCookie::fNextIteratorID = 0;
// ramfs_create_dir
static status_t
ramfs_create_dir(fs_volume* _volume, fs_vnode _dir, const char *name, int mode)
{
FUNCTION(("name: `%s', mode: %x\n", name, mode));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
status_t error = B_OK;
// check name
if (!name || *name == '\0') {
SET_ERROR(error, B_BAD_VALUE);
// check directory
} else if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// directory deleted?
bool removed;
if (get_vnode_removed(volume->GetID(), dir->GetID(), &removed) != B_OK
|| removed) {
SET_ERROR(error, B_NOT_ALLOWED);
}
// check directory write permissions
error = dir->CheckPermissions(ACCESS_W);
Node *node = NULL;
if (error == B_OK) {
// check if entry does already exist
if (dir->FindNode(name, &node) == B_OK) {
SET_ERROR(error, B_FILE_EXISTS);
} else {
// entry doesn't exist: create a directory
Directory *newDir = NULL;
error = dir->CreateDirectory(name, &newDir);
if (error == B_OK) {
node = newDir;
// set permissions, owner and group
node->SetMode(mode);
node->SetUID(geteuid());
node->SetGID(getegid());
// put the node
volume->PutVNode(node->GetID());
}
}
}
NodeMTimeUpdater mTimeUpdater2(node);
// notify listeners
if (error == B_OK) {
notify_entry_created(volume->GetID(), dir->GetID(), name,
node->GetID());
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_remove_dir
static status_t
ramfs_remove_dir(fs_volume* _volume, fs_vnode _dir, const char *name)
{
FUNCTION(("name: `%s'\n", name));
Volume* volume = (Volume*)_volume->private_volume;
Directory *dir = dynamic_cast<Directory*>((Node*)_dir);
status_t error = B_OK;
// check name
if (!name || *name == '\0' || !strcmp(name, ".") || !strcmp(name, "..")) {
SET_ERROR(error, B_BAD_VALUE);
// check node
} else if (!dir) {
SET_ERROR(error, B_BAD_VALUE);
} else if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(dir);
// check directory write permissions
error = dir->CheckPermissions(ACCESS_W);
ino_t nodeID = -1;
if (error == B_OK) {
// check if entry exists
Node *node = NULL;
Entry *entry = NULL;
if (dir->FindAndGetNode(name, &node, &entry) == B_OK) {
nodeID = node->GetID();
if (!node->IsDirectory()) {
SET_ERROR(error, B_NOT_A_DIRECTORY);
} else if (!dynamic_cast<Directory*>(node)->IsEmpty()) {
SET_ERROR(error, B_DIRECTORY_NOT_EMPTY);
} else
error = dir->DeleteEntry(entry);
volume->PutVNode(node);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
}
// notify listeners
if (error == B_OK)
notify_entry_removed(volume->GetID(), dir->GetID(), name, nodeID);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_open_dir
static status_t
ramfs_open_dir(fs_volume /*fs*/, fs_vnode _node, fs_cookie *_cookie)
{
// FUNCTION_START();
// Volume *volume = (Volume*)fs;
Node *node = (Node*)_node;
FUNCTION(("dir: (%Lu)\n", node->GetID()));
// get the Directory
status_t error = (node->IsDirectory() ? B_OK : B_NOT_A_DIRECTORY);
Directory *dir = NULL;
if (error == B_OK) {
dir = dynamic_cast<Directory*>(node);
if (!dir) {
FATAL(("Node %Ld pretends to be a Directory, but isn't!\n",
node->GetID()));
error = B_NOT_A_DIRECTORY;
}
}
// create a DirectoryCookie
if (error == B_OK) {
DirectoryCookie *cookie = new(nothrow) DirectoryCookie(dir);
if (cookie) {
error = cookie->Suspend();
if (error == B_OK)
*_cookie = cookie;
else
delete cookie;
} else
SET_ERROR(error, B_NO_MEMORY);
}
FUNCTION_END();
RETURN_ERROR(error);
}
// ramfs_close_dir
static status_t
ramfs_close_dir(fs_volume /*fs*/, fs_vnode DARG(_node), fs_cookie _cookie)
{
FUNCTION_START();
FUNCTION(("dir: (%Lu)\n", ((Node*)_node)->GetID()));
// No locking needed, since the Directory is guaranteed to live at this
// time and for iterators there is a separate locking.
DirectoryCookie *cookie = (DirectoryCookie*)_cookie;
cookie->Unset();
return B_OK;
}
// ramfs_free_dir_cookie
static status_t
ramfs_free_dir_cookie(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie _cookie)
{
FUNCTION_START();
DirectoryCookie *cookie = (DirectoryCookie*)_cookie;
delete cookie;
return B_OK;
}
// ramfs_read_dir
static status_t
ramfs_read_dir(fs_volume* _volume, fs_vnode DARG(_node), fs_cookie _cookie,
struct dirent *buffer, size_t bufferSize, uint32 *count)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
DARG(Node *node = (Node*)_node; )
FUNCTION(("dir: (%Lu)\n", node->GetID()));
DirectoryCookie *cookie = (DirectoryCookie*)_cookie;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
error = cookie->Resume();
if (error == B_OK) {
// read one entry
ino_t nodeID = -1;
const char *name = NULL;
if (cookie->GetNext(&nodeID, &name) == B_OK) {
PRINT((" entry: `%s'\n", name));
size_t nameLen = strlen(name);
// check, whether the entry fits into the buffer,
// and fill it in
size_t length = (buffer->d_name + nameLen + 1) - (char*)buffer;
if (length <= bufferSize) {
buffer->d_dev = volume->GetID();
buffer->d_ino = nodeID;
memcpy(buffer->d_name, name, nameLen);
buffer->d_name[nameLen] = '\0';
buffer->d_reclen = length;
*count = 1;
} else {
SET_ERROR(error, B_BUFFER_OVERFLOW);
}
} else
*count = 0;
cookie->Suspend();
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_rewind_dir
static status_t
ramfs_rewind_dir(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie _cookie)
{
FUNCTION_START();
// No locking needed, since the Directory is guaranteed to live at this
// time and for iterators there is a separate locking.
DirectoryCookie *cookie = (DirectoryCookie*)_cookie;
// no need to Resume(), iterator remains suspended
status_t error = cookie->Rewind();
RETURN_ERROR(error);
}
// #pragma mark - Attribute Directories
// ramfs_open_attr_dir
static status_t
ramfs_open_attr_dir(fs_volume* _volume, fs_vnode _node, fs_cookie *cookie)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// check permissions
error = node->CheckPermissions(ACCESS_R);
// create iterator
AttributeIterator *iterator = NULL;
if (error == B_OK) {
iterator = new(nothrow) AttributeIterator(node);
if (iterator)
error = iterator->Suspend();
else
SET_ERROR(error, B_NO_MEMORY);
}
// set result / cleanup on failure
if (error == B_OK)
*cookie = iterator;
else
delete iterator;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_close_attr_dir
static status_t
ramfs_close_attr_dir(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie cookie)
{
FUNCTION_START();
// No locking needed, since the Node is guaranteed to live at this time
// and for iterators there is a separate locking.
AttributeIterator *iterator = (AttributeIterator*)cookie;
iterator->Unset();
return B_OK;
}
// ramfs_free_attr_dir_cookie
static status_t
ramfs_free_attr_dir_cookie(fs_volume /*fs*/, fs_vnode /*_node*/,
fs_cookie cookie)
{
FUNCTION_START();
// No locking needed, since the Node is guaranteed to live at this time
// and for iterators there is a separate locking.
AttributeIterator *iterator = (AttributeIterator*)cookie;
delete iterator;
return B_OK;
}
// ramfs_read_attr_dir
static status_t
ramfs_read_attr_dir(fs_volume* _volume, fs_vnode _node, fs_cookie cookie,
struct dirent *buffer, size_t bufferSize, uint32 *count)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
AttributeIterator *iterator = (AttributeIterator*)cookie;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
error = iterator->Resume();
if (error == B_OK) {
// get next attribute
Attribute *attribute = NULL;
if (iterator->GetNext(&attribute) == B_OK) {
const char *name = attribute->GetName();
size_t nameLen = strlen(name);
// check, whether the entry fits into the buffer,
// and fill it in
size_t length = (buffer->d_name + nameLen + 1) - (char*)buffer;
if (length <= bufferSize) {
buffer->d_dev = volume->GetID();
buffer->d_ino = -1; // attributes don't have a node ID
memcpy(buffer->d_name, name, nameLen);
buffer->d_name[nameLen] = '\0';
buffer->d_reclen = length;
*count = 1;
} else {
SET_ERROR(error, B_BUFFER_OVERFLOW);
}
} else
*count = 0;
iterator->Suspend();
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_rewind_attr_dir
static status_t
ramfs_rewind_attr_dir(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie cookie)
{
FUNCTION_START();
// No locking needed, since the Node is guaranteed to live at this time
// and for iterators there is a separate locking.
AttributeIterator *iterator = (AttributeIterator*)cookie;
// no need to Resume(), iterator remains suspended
status_t error = iterator->Rewind();
RETURN_ERROR(error);
}
// #pragma mark - Attributes
// AttributeCookie
class AttributeCookie {
public:
AttributeCookie() : fOpenMode(0), fLastNotificationTime(0) {}
status_t Init(const char* name, int openMode)
{
if (!fName.SetTo(name))
return B_NO_MEMORY;
fOpenMode = openMode;
return B_OK;
}
inline const char* GetName() const { return fName.GetString(); }
inline int GetOpenMode() const { return fOpenMode; }
inline bigtime_t GetLastNotificationTime() const
{ return fLastNotificationTime; }
inline bool NotificationIntervalElapsed(bool set = false)
{
bigtime_t currentTime = system_time();
bool result = (currentTime - fLastNotificationTime
> kNotificationInterval);
if (set && result)
fLastNotificationTime = currentTime;
return result;
}
private:
String fName;
int fOpenMode;
bigtime_t fLastNotificationTime;
};
// ramfs_create_attr
static status_t
ramfs_create_attr(fs_volume* _volume, fs_vnode _node, const char *name, uint32 type,
int openMode, fs_cookie *_cookie)
{
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
if (VolumeWriteLocker locker = volume) {
// try to find the attribute
Attribute *attribute = NULL;
node->FindAttribute(name, &attribute);
// in case the attribute exists we fail if required by the openMode
if (attribute && (openMode & O_EXCL))
RETURN_ERROR(B_FILE_EXISTS);
// creating and truncating require write permission
int accessMode = open_mode_to_access(openMode);
if (!attribute || (openMode & O_TRUNC))
accessMode |= ACCESS_W;
// check required permissions against node permissions
status_t error = node->CheckPermissions(accessMode);
if (error != B_OK)
RETURN_ERROR(error);
// create the cookie
AttributeCookie *cookie = new(nothrow) AttributeCookie();
if (!cookie)
return B_NO_MEMORY;
ObjectDeleter<AttributeCookie> cookieDeleter(cookie);
// init the cookie
error = cookie->Init(name, openMode);
if (error != B_OK)
RETURN_ERROR(error);
// if not existing yet, create the attribute and set its type
if (!attribute) {
error = node->CreateAttribute(name, &attribute);
if (error != B_OK)
RETURN_ERROR(error);
attribute->SetType(type);
notify_attribute_changed(volume->GetID(), node->GetID(), name,
B_ATTR_CREATED);
// else truncate if requested
} else if (openMode & O_TRUNC) {
error = attribute->SetSize(0);
if (error != B_OK)
return error;
notify_attribute_changed(volume->GetID(), node->GetID(), name,
B_ATTR_CHANGED);
}
NodeMTimeUpdater mTimeUpdater(node);
// success
cookieDeleter.Detach();
*_cookie = cookie;
} else
RETURN_ERROR(B_ERROR);
return B_OK;
}
// ramfs_open_attr
static status_t
ramfs_open_attr(fs_volume* _volume, fs_vnode _node, const char *name, int openMode,
fs_cookie *_cookie)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("node: %lld\n", node->GetID()));
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
// find the attribute
Attribute *attribute = NULL;
if (error == B_OK)
error = node->FindAttribute(name, &attribute);
// truncating requires write permission
int accessMode = open_mode_to_access(openMode);
if (error == B_OK && (openMode & O_TRUNC))
accessMode |= ACCESS_W;
// check open mode against permissions
if (error == B_OK)
error = node->CheckPermissions(accessMode);
// create the cookie
AttributeCookie *cookie = NULL;
if (error == B_OK) {
cookie = new(nothrow) AttributeCookie();
if (cookie) {
SET_ERROR(error, cookie->Init(name, openMode));
} else {
SET_ERROR(error, B_NO_MEMORY);
}
}
// truncate if requested
if (error == B_OK && (openMode & O_TRUNC)) {
error = attribute->SetSize(0);
if (error == B_OK) {
notify_attribute_changed(volume->GetID(), node->GetID(), name,
B_ATTR_CHANGED);
}
}
NodeMTimeUpdater mTimeUpdater(node);
// set result / cleanup on failure
if (error == B_OK)
*_cookie = cookie;
else if (cookie)
delete cookie;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_close_attr
static status_t
ramfs_close_attr(fs_volume* _volume, fs_vnode _node, fs_cookie cookie)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
FUNCTION(("node: %lld\n", node->GetID()));
status_t error = B_OK;
// notify listeners
if (VolumeReadLocker locker = volume) {
notify_if_stat_changed(volume, node);
} else
SET_ERROR(error, B_ERROR);
return B_OK;
}
// ramfs_free_attr_cookie
static status_t
ramfs_free_attr_cookie(fs_volume /*fs*/, fs_vnode /*_node*/, fs_cookie _cookie)
{
FUNCTION_START();
AttributeCookie *cookie = (AttributeCookie*)_cookie;
delete cookie;
return B_OK;
}
// ramfs_read_attr
static status_t
ramfs_read_attr(fs_volume* _volume, fs_vnode _node, fs_cookie _cookie, off_t pos,
void *buffer, size_t *bufferSize)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
AttributeCookie *cookie = (AttributeCookie*)_cookie;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// find the attribute
Attribute *attribute = NULL;
if (error == B_OK)
error = node->FindAttribute(cookie->GetName(), &attribute);
// check permissions
int accessMode = open_mode_to_access(cookie->GetOpenMode());
if (error == B_OK && !(accessMode & ACCESS_R))
SET_ERROR(error, B_NOT_ALLOWED);
// read
if (error == B_OK)
error = attribute->ReadAt(pos, buffer, *bufferSize, bufferSize);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_write_attr
static status_t
ramfs_write_attr(fs_volume* _volume, fs_vnode _node, fs_cookie _cookie, off_t pos,
const void *buffer, size_t *bufferSize)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
AttributeCookie *cookie = (AttributeCookie*)_cookie;
status_t error = B_OK;
// Don't allow writing the reserved attributes.
const char *name = cookie->GetName();
if (name[0] == '\0' || !strcmp(name, "name")
|| !strcmp(name, "last_modified") || !strcmp(name, "size")) {
//FUNCTION(("failed: node: %s, attribute: %s\n", node->GetName(), name));
RETURN_ERROR(B_NOT_ALLOWED);
}
if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(node);
// find the attribute
Attribute *attribute = NULL;
if (error == B_OK)
error = node->FindAttribute(cookie->GetName(), &attribute);
// check permissions
int accessMode = open_mode_to_access(cookie->GetOpenMode());
if (error == B_OK && !(accessMode & ACCESS_W))
SET_ERROR(error, B_NOT_ALLOWED);
// write the data
if (error == B_OK)
error = attribute->WriteAt(pos, buffer, *bufferSize, bufferSize);
// notify listeners
if (error == B_OK) {
notify_attribute_changed(volume->GetID(), node->GetID(), name,
B_ATTR_CHANGED);
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_read_attr_stat
static status_t
ramfs_read_attr_stat(fs_volume* _volume, fs_vnode _node, fs_cookie _cookie,
struct stat *st)
{
// FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
AttributeCookie *cookie = (AttributeCookie*)_cookie;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// find the attribute
Attribute *attribute = NULL;
if (error == B_OK)
error = node->FindAttribute(cookie->GetName(), &attribute);
// check permissions
int accessMode = open_mode_to_access(cookie->GetOpenMode());
if (error == B_OK && !(accessMode & ACCESS_R))
SET_ERROR(error, B_NOT_ALLOWED);
// read
if (error == B_OK) {
st->st_type = attribute->GetType();
st->st_size = attribute->GetSize();
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_rename_attr
static status_t
ramfs_rename_attr(fs_volume /*fs*/, fs_vnode /*_fromNode*/,
const char */*fromName*/, fs_vnode /*_toNode*/, const char */*toName*/)
{
// TODO:...
return B_BAD_VALUE;
}
// ramfs_remove_attr
static status_t
ramfs_remove_attr(fs_volume* _volume, fs_vnode _node, const char *name)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
Node *node = (Node*)_node;
status_t error = B_OK;
if (VolumeWriteLocker locker = volume) {
NodeMTimeUpdater mTimeUpdater(node);
// check permissions
error = node->CheckPermissions(ACCESS_W);
// find the attribute
Attribute *attribute = NULL;
if (error == B_OK)
error = node->FindAttribute(name, &attribute);
// delete it
if (error == B_OK)
error = node->DeleteAttribute(attribute);
// notify listeners
if (error == B_OK) {
notify_attribute_changed(volume->GetID(), node->GetID(), name,
B_ATTR_REMOVED);
}
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// #pragma mark - Indices
// IndexDirCookie
class IndexDirCookie {
public:
IndexDirCookie() : index_index(0) {}
int32 index_index;
};
// ramfs_open_index_dir
static status_t
ramfs_open_index_dir(fs_volume* _volume, fs_cookie *_cookie)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// check whether an index directory exists
if (volume->GetIndexDirectory()) {
IndexDirCookie *cookie = new(nothrow) IndexDirCookie;
if (cookie)
*_cookie = cookie;
else
SET_ERROR(error, B_NO_MEMORY);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_close_index_dir
static status_t
ramfs_close_index_dir(fs_volume /*fs*/, fs_cookie /*_cookie*/)
{
FUNCTION_START();
return B_OK;
}
// ramfs_free_index_dir_cookie
static status_t
ramfs_free_index_dir_cookie(fs_volume /*fs*/, fs_cookie _cookie)
{
FUNCTION_START();
IndexDirCookie *cookie = (IndexDirCookie*)_cookie;
delete cookie;
return B_OK;
}
// ramfs_read_index_dir
static status_t
ramfs_read_index_dir(fs_volume* _volume, fs_cookie _cookie, struct dirent *buffer,
size_t bufferSize, uint32 *count)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
IndexDirCookie *cookie = (IndexDirCookie*)_cookie;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// get the next index
Index *index = volume->GetIndexDirectory()->IndexAt(
cookie->index_index++);
if (index) {
const char *name = index->GetName();
size_t nameLen = strlen(name);
// check, whether the entry fits into the buffer,
// and fill it in
size_t length = (buffer->d_name + nameLen + 1) - (char*)buffer;
if (length <= bufferSize) {
buffer->d_dev = volume->GetID();
buffer->d_ino = -1; // indices don't have a node ID
memcpy(buffer->d_name, name, nameLen);
buffer->d_name[nameLen] = '\0';
buffer->d_reclen = length;
*count = 1;
} else {
SET_ERROR(error, B_BUFFER_OVERFLOW);
}
} else
*count = 0;
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_rewind_index_dir
static status_t
ramfs_rewind_index_dir(fs_volume /*fs*/, fs_cookie _cookie)
{
FUNCTION_START();
IndexDirCookie *cookie = (IndexDirCookie*)_cookie;
cookie->index_index = 0;
return B_OK;
}
// ramfs_create_index
static status_t
ramfs_create_index(fs_volume* _volume, const char *name, uint32 type,
uint32 /*flags*/)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
// only root is allowed to manipulate the indices
if (geteuid() != 0) {
SET_ERROR(error, B_NOT_ALLOWED);
} else if (VolumeWriteLocker locker = volume) {
// get the index directory
if (IndexDirectory *indexDir = volume->GetIndexDirectory()) {
// check whether an index with that name does already exist
if (indexDir->FindIndex(name)) {
SET_ERROR(error, B_FILE_EXISTS);
} else {
// create the index
AttributeIndex *index;
error = indexDir->CreateIndex(name, type, &index);
}
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_remove_index
static status_t
ramfs_remove_index(fs_volume* _volume, const char *name)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
// only root is allowed to manipulate the indices
if (geteuid() != 0) {
SET_ERROR(error, B_NOT_ALLOWED);
} else if (VolumeWriteLocker locker = volume) {
// get the index directory
if (IndexDirectory *indexDir = volume->GetIndexDirectory()) {
// check whether an index with that name does exist
if (Index *index = indexDir->FindIndex(name)) {
// don't delete a special index
if (indexDir->IsSpecialIndex(index)) {
SET_ERROR(error, B_BAD_VALUE);
} else
indexDir->DeleteIndex(index);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// ramfs_read_index_stat
static status_t
ramfs_read_index_stat(fs_volume* _volume, const char *name, struct stat *st)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
status_t error = B_OK;
if (VolumeReadLocker locker = volume) {
// get the index directory
if (IndexDirectory *indexDir = volume->GetIndexDirectory()) {
// find the index
if (Index *index = indexDir->FindIndex(name)) {
st->st_type = index->GetType();
if (index->HasFixedKeyLength())
st->st_size = index->GetKeyLength();
else
st->st_size = kMaxIndexKeyLength;
st->st_atime = 0; // TODO: index times
st->st_mtime = 0; // ...
st->st_ctime = 0; // ...
st->st_crtime = 0; // ...
st->st_uid = 0; // root owns the indices
st->st_gid = 0; //
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ENTRY_NOT_FOUND);
} else
SET_ERROR(error, B_ERROR);
RETURN_ERROR(error);
}
// #pragma mark - Queries
// Query implementation by Axel Dörfler. Slightly adjusted.
// ramfs_open_query
static status_t
ramfs_open_query(fs_volume* _volume, const char *queryString, uint32 flags,
port_id port, uint32 token, fs_cookie *cookie)
{
FUNCTION_START();
PRINT(("query = \"%s\", flags = %lu, port_id = %ld, token = %ld\n", queryString, flags, port, token));
Volume* volume = (Volume*)_volume->private_volume;
// lock the volume
VolumeReadLocker locker(volume);
if (!locker.IsLocked())
RETURN_ERROR(B_ERROR);
// parse the query expression
Expression *expression = new Expression((char *)queryString);
if (expression == NULL)
RETURN_ERROR(B_NO_MEMORY);
ObjectDeleter<Expression> expressionDeleter(expression);
if (expression->InitCheck() < B_OK) {
WARN(("Could not parse query, stopped at: \"%s\"\n",
expression->Position()));
RETURN_ERROR(B_BAD_VALUE);
}
// create the query
Query *query = new Query(volume, expression, flags);
if (query == NULL)
RETURN_ERROR(B_NO_MEMORY);
expressionDeleter.Detach();
// TODO: The Query references an Index, but nothing prevents the Index
// from being deleted, while the Query is in existence.
if (flags & B_LIVE_QUERY)
query->SetLiveMode(port, token);
*cookie = (void *)query;
return B_OK;
}
// ramfs_close_query
static status_t
ramfs_close_query(fs_volume /*fs*/, fs_cookie /*cookie*/)
{
FUNCTION_START();
return B_OK;
}
// ramfs_free_query_cookie
static status_t
ramfs_free_query_cookie(fs_volume* _volume, fs_cookie cookie)
{
FUNCTION_START();
Volume* volume = (Volume*)_volume->private_volume;
// lock the volume
VolumeReadLocker locker(volume);
if (!locker.IsLocked())
RETURN_ERROR(B_ERROR);
Query *query = (Query *)cookie;
Expression *expression = query->GetExpression();
delete query;
delete expression;
return B_OK;
}
// ramfs_read_query
static status_t
ramfs_read_query(fs_volume* _volume, fs_cookie cookie, struct dirent *buffer,
size_t bufferSize, uint32 *count)
{
FUNCTION_START();
Query *query = (Query *)cookie;
Volume* volume = (Volume*)_volume->private_volume;
// lock the volume
VolumeReadLocker locker(volume);
if (!locker.IsLocked())
RETURN_ERROR(B_ERROR);
status_t status = query->GetNextEntry(buffer, bufferSize);
if (status == B_OK)
*count = 1;
else if (status == B_ENTRY_NOT_FOUND)
*count = 0;
else
return status;
return B_OK;
}
// TODO: status_t (*rewind_query)(fs_volume fs, fs_cookie cookie);
// #pragma mark - Module Interface
static status_t
ramfs_std_ops(int32 op, ...)
{
switch (op) {
case B_MODULE_INIT:
{
init_debugging();
PRINT(("ramfs_std_ops(): B_MODULE_INIT\n"));
return B_OK;
}
case B_MODULE_UNINIT:
PRINT(("ramfs_std_ops(): B_MODULE_UNINIT\n"));
exit_debugging();
return B_OK;
default:
return B_ERROR;
}
}
static file_system_module_info sRamFSModuleInfo = {
{
"file_systems/ramfs" B_CURRENT_FS_API_VERSION,
0,
ramfs_std_ops,
},
"ramfs", // short_name
"RAM File System", // pretty_name
B_DISK_SYSTEM_SUPPORTS_WRITING, // DDM flags
// scanning
NULL, // identify_partition()
NULL, // scan_partition()
NULL, // free_identify_partition_cookie()
NULL, // free_partition_content_cookie()
&ramfs_mount,
&ramfs_unmount,
&ramfs_read_fs_info,
&ramfs_write_fs_info,
&ramfs_sync,
/* vnode operations */
&ramfs_lookup,
NULL, // &ramfs_get_vnode_name,
&ramfs_read_vnode,
&ramfs_write_vnode,
&ramfs_remove_vnode,
/* VM file access */
NULL, // &ramfs_can_page,
NULL, // &ramfs_read_pages,
NULL, // &ramfs_write_pages,
NULL, // &ramfs_get_file_map,
&ramfs_ioctl,
&ramfs_set_flags,
NULL, // &ramfs_select,
NULL, // &ramfs_deselect,
&ramfs_fsync,
&ramfs_read_symlink,
&ramfs_create_symlink,
&ramfs_link,
&ramfs_unlink,
&ramfs_rename,
&ramfs_access,
&ramfs_read_stat,
&ramfs_write_stat,
NULL, // &ramfs_preallocate,
/* file operations */
&ramfs_create,
&ramfs_open,
&ramfs_close,
&ramfs_free_cookie,
&ramfs_read,
&ramfs_write,
/* directory operations */
&ramfs_create_dir,
&ramfs_remove_dir,
&ramfs_open_dir,
&ramfs_close_dir,
&ramfs_free_dir_cookie,
&ramfs_read_dir,
&ramfs_rewind_dir,
/* attribute directory operations */
&ramfs_open_attr_dir,
&ramfs_close_attr_dir,
&ramfs_free_attr_dir_cookie,
&ramfs_read_attr_dir,
&ramfs_rewind_attr_dir,
/* attribute operations */
&ramfs_create_attr,
&ramfs_open_attr,
&ramfs_close_attr,
&ramfs_free_attr_cookie,
&ramfs_read_attr,
&ramfs_write_attr,
&ramfs_read_attr_stat,
NULL, // &ramfs_write_attr_stat,
&ramfs_rename_attr,
&ramfs_remove_attr,
/* index directory & index operations */
&ramfs_open_index_dir,
&ramfs_close_index_dir,
&ramfs_free_index_dir_cookie,
&ramfs_read_index_dir,
&ramfs_rewind_index_dir,
&ramfs_create_index,
&ramfs_remove_index,
&ramfs_read_index_stat,
/* query operations */
&ramfs_open_query,
&ramfs_close_query,
&ramfs_free_query_cookie,
&ramfs_read_query,
NULL, // &ramfs_rewind_query,
};
module_info *modules[] = {
(module_info *)&sRamFSModuleInfo,
NULL,
};